In the Design Guidelines for Developing Class Libraries, Microsoft say:
Do not assign instances of mutable types to read-only fields.
The objects created using a mutable type can be modified after they are created. For example, arrays and most collections are mutable types while Int32, Uri, and String are immutable types. For fields that hold a mutable reference type, the read-only modifier prevents the field value from being overwritten but does not protect the mutable type from modification.
This simply restates the behaviour of readonly without explaining why it's bad to use readonly. The implication appears to be that many people do not understand what "readonly" does and will wrongly expect readonly fields to be deeply immutable. In effect it advises using "readonly" as code documentation indicating deep immutability - despite the fact that the compiler has no way to enforce this - and disallows its use for its normal function: to ensure that the value of the field doesn't change after the object has been constructed.
I feel uneasy with this recommendation to use "readonly" to indicate something other than its normal meaning understood by the compiler. I feel that it encourages people to misunderstand the meaning of "readonly", and furthermore to expect it to mean something that the author of the code might not intend. I feel that it precludes using it in places it could be useful - e.g. to show that some relationship between two mutable objects remains unchanged for the lifetime of one of those objects. The notion of assuming that readers do not understand the meaning of "readonly" also appears to be in contradiction to other advice from Microsoft, such as FxCop's "Do not initialize unnecessarily" rule, which assumes readers of your code to be experts in the language and should know that (for example) bool fields are automatically initialised to false, and stops you from providing the redundancy that shows "yes, this has been consciously set to false; I didn't just forget to initialize it".
So, first and foremost, why do Microsoft advise against use of readonly for references to mutable types? I'd also be interested to know:
- Do you follow this Design Guideline in all your code?
- What do you expect when you see "readonly" in a piece of code you didn't write?
I agree with you completely, and I do sometimes use
readonly
in my code for mutable reference types.As an example: I might have some
private
orprotected
member -- say, aList<T>
-- which I use within a class's methods in all its mutable glory (callingAdd
,Remove
, etc.). I may simply want to put a safeguard in place to ensure that, no matter what, I am always dealing with the same object. This protects both me and other developers from doing something stupid: namely, assigning the member to a new object.To me, this is often a preferable alternative to using a property with a private
set
method. Why? Becausereadonly
means the value cannot be changed after instantiation, even by the base class.In other words, if I had this:
Then I could still set
InternalList = new List<T>();
at any arbitrary point in code in my base class. (This would require a very foolish error on my part, yes; but it would still be possible.)On the other hand, this:
Makes it unmistakably clear that
_internalList
can only ever refer to one particular object (the one to which_internalList
is set in the constructor).So I am on your side. The idea that one should refrain from using
readonly
on a mutable reference type is frustrating to me personally, as it basically presupposes a misunderstanding of thereadonly
keyword.I had a quick look in the Framework Design Guidelines book (pages 161-162), and it basically states what you've already noticed yourself. There's an additional comment by Joe Duffy that explains the guideline's raison-d'être:
I personally think that the keyword
readonly
was named badly. The fact that it only specifies the const-ness of the reference, and not of the const-ness of the referenced object, easily creates misleading situations.I think it would have been preferable if
readonly
made referenced objects immutable, too, and not just the reference, because that is what the keyword implies.To remedy this unfortunate situation, the guideline was made up. While I think that its advice is sound from the human point of view (it's not always obvious which types are mutable and which aren't without looking up their definition, and the word suggests deep immutability), I sometimes wish that, when it comes to declaring const-ness, C# would offer a freedom similar to that offered by C++, where you can define
const
either on the pointer, or on the pointed-to-object, or on both or nothing.Microsoft has a few such peculiar advices. The other one that immediately springs to mind is not to nest generic types in public members, like
List<List<int>>
. I try to avoid these constructs where easily possible, but ignore the newbie-friendly advice when I feel the use is justified.As for readonly fields - I try to avoid public fields as such, instead going for properties. I think there was a piece of advice about that too, but more importantly there are cases now and then when a field doesn't work while a property does (mostly it has to do with databinding and/or visual designers). By making all public fields properties I avoid any potential problems.
In the end they are just guidelines. I know for a fact that the people at Microsoft often don't follow all of the guidelines.
The syntax you are looking for is supported by the C++/CLI language:
The first const makes the referenced object immutable, the 2nd makes the reference immutable. The latter is equivalent to the readonly keyword in C#. Attempts to evade it produce a compile error:
It is however smoke and mirrors. The immutability is verified by the compiler, not by the CLR. Another managed language could modify both. Which is the root of the problem, the CLR just doesn't have support for it.
It seems natural that if a field is readonly, you would expect to not be able to change the value or anything having to do with it. If I knew that Bar was a readonly field of Foo, I could obviously not say
But I can get away with saying
If the object backing Bar is, in fact, mutable. Microsoft is simply recommending against that subtle, counterintuitive behavior by suggesting that readonly fields be backed by immutable objects.