In this 7th installment of C# Basics, I’m going to cover the differences between the modifiers const, readonly, and static, specifically in reference to class members where the developer wishes to use the fields or properties in client code that cannot change the value of that member.
To illustrate and discuss the differences between each modifier, I’ve put together a contrived set of classes with the least amount of code I can think to add in order to review and to examine the differences in the IL (intermediate language) taken from Ildasm.exe. (Visit this nice page to learn how to use Ildasm.exe from with Visual Studio.)
using System;
namespace ReadOnlyConstant
{
public class MyConstOnly
{
// assigned at declaration and used at compile time only
public const int Age = 20;
}
public class MyReadOnly
{
// can only be assigned in declaration or constructor
public readonly int Age = 0;
public MyReadOnly()
{
Age = 20;
}
}
public static class MyStaticOnly
{
// can be assigned within class but "readonly" for client
private static int _age = 20;
public static int Age { get { return _age; } }
}
}
The goal in each class above is to create a public field or member that can be read by client code but cannot be changed by client code. Looking through the IL produced by compiling this code can also be instructive even if you do not fully understand each and every IL instruction. Take a look here at these classes under the compiled covers and notice that the MyConstOnly class does not have a getter method to retrieve Age nor is it's value set in the .ctor but only noted by the compiler in the .field definition for use by the compiler later should client code use it. Then read through to the client code and see its IL code as well.
// =============== CLASS MEMBERS DECLARATION ===================
.class public auto ansi beforefieldinit ReadOnlyConstant.MyConstOnly
extends [mscorlib]System.Object
{
.field public static literal int32 Age = int32(0x00000014)
.method public hidebysig specialname rtspecialname
instance void .ctor() cil managed
{
// Code size 7 (0x7)
.maxstack 8
IL_0000: ldarg.0
IL_0001: call instance void [mscorlib]System.Object::.ctor()
IL_0006: ret
} // end of method MyConstOnly::.ctor
} // end of class ReadOnlyConstant.MyConstOnly
.class public auto ansi beforefieldinit ReadOnlyConstant.MyReadOnly
extends [mscorlib]System.Object
{
.field public initonly int32 Age
.method public hidebysig specialname rtspecialname
instance void .ctor() cil managed
{
// Code size 25 (0x19)
.maxstack 8
IL_0000: ldarg.0
IL_0001: ldc.i4.0
IL_0002: stfld int32 ReadOnlyConstant.MyReadOnly::Age
IL_0007: ldarg.0
IL_0008: call instance void [mscorlib]System.Object::.ctor()
IL_000d: nop
IL_000e: nop
IL_000f: ldarg.0
IL_0010: ldc.i4.s 20
IL_0012: stfld int32 ReadOnlyConstant.MyReadOnly::Age
IL_0017: nop
IL_0018: ret
} // end of method MyReadOnly::.ctor
} // end of class ReadOnlyConstant.MyReadOnly
.class public abstract auto ansi sealed beforefieldinit ReadOnlyConstant.MyStaticOnly
extends [mscorlib]System.Object
{
.field private static int32 _age
.method public hidebysig specialname static
int32 get_Age() cil managed
{
// Code size 11 (0xb)
.maxstack 1
.locals init ([0] int32 CS$1$0000)
IL_0000: nop
IL_0001: ldsfld int32 ReadOnlyConstant.MyStaticOnly::_age
IL_0006: stloc.0
IL_0007: br.s IL_0009
IL_0009: ldloc.0
IL_000a: ret
} // end of method MyStaticOnly::get_Age
.method private hidebysig specialname rtspecialname static
void .cctor() cil managed
{
// Code size 8 (0x8)
.maxstack 8
IL_0000: ldc.i4.s 20
IL_0002: stsfld int32 ReadOnlyConstant.MyStaticOnly::_age
IL_0007: ret
} // end of method MyStaticOnly::.cctor
.property int32 Age()
{
.get int32 ReadOnlyConstant.MyStaticOnly::get_Age()
} // end of property MyStaticOnly::Age
} // end of class ReadOnlyConstant.MyStaticOnly
// =============================================================
You can read the MSDN full explanations of each modifier but here’s the basics:
const
Can only be assigned a value in declaration and can only be a value type or string. Use the const modifier when you KNOW the value won’t change. If you think it might change at a later date and your assembly is distributed as a compiled library, consider one of the other modifiers to assure that you don’t have a value you didn’t expect in your client code. (See use code sample below.)
readonly
Can assign a value at declaration or in the class constructor. It is important to note that if you use a reference type with modifiable members, your client code can still modify those members even if it cannot assign a value to the readonly reference. Note in the IL above that the initialization of the declared value occurs in the .ctor before the assignment in the .ctor, so if you are wondering which would be better, now you have some insight into that question.
static
Can assign the value of the private member anywhere within the class code. Note the initialization of the value in the static .ctor of the class. You could also assign the value in some other method later but with the public property implementing only a get, the client code cannot assign a value.
And here is the client code and it’s IL just below it. The most important point to note in the IL is that the client code is compiled with the const’s literal value, NOT a get to the class. This is why you must watch for the use of a const that could change with a new library. Make sure you compile your client code against that new library when you get it or you could be very sorry when the library is using one const compiled value and you’re using another.
namespace TestConsole
{
class Program
{
static void Main(string[] args)
{
// compiler will replace with constant value
// If the referenced assembly is changed to 40 and this is
// not compiled again against that new assembly, the value
// for mcoAge will still be 20. (See IL below.)
int mcoAge = MyConstOnly.Age;
MyReadOnly mro = new MyReadOnly();
int mroAge = mro.Age;
int msoAge = MyStaticOnly.Age;
Console.WriteLine("{0} {1} {2}", mcoAge, mroAge, msoAge);
}
}
}
// output: 20 20 20
// and here is the IL with some of my own comments
// =============== CLASS MEMBERS DECLARATION ===================
.class private auto ansi beforefieldinit TestConsole.Program
extends [mscorlib]System.Object
{
.method private hidebysig static void Main(string[] args) cil managed
{
.entrypoint
// Code size 53 (0x35)
.maxstack 4
.locals init ([0] int32 mcoAge,
[1] class [ReadOnlyConstant]ReadOnlyConstant.MyReadOnly mro,
[2] int32 mroAge,
[3] int32 msoAge)
IL_0000: nop
IL_0001: ldc.i4.s 20 //NOTE: literal value assigned - no mention of MyConstOnly class
IL_0003: stloc.0
IL_0004: newobj instance void [ReadOnlyConstant]ReadOnlyConstant.MyReadOnly::.ctor()
IL_0009: stloc.1
IL_000a: ldloc.1
IL_000b: ldfld int32 [ReadOnlyConstant]ReadOnlyConstant.MyReadOnly::Age
IL_0010: stloc.2
IL_0011: call int32 [ReadOnlyConstant]ReadOnlyConstant.MyStaticOnly::get_Age()
IL_0016: stloc.3
IL_0017: ldstr "{0} {1} {2}"
IL_001c: ldloc.0
IL_001d: box [mscorlib]System.Int32
IL_0022: ldloc.2
IL_0023: box [mscorlib]System.Int32
IL_0028: ldloc.3
IL_0029: box [mscorlib]System.Int32
IL_002e: call void [mscorlib]System.Console::WriteLine(string,
object,
object,
object)
IL_0033: nop
IL_0034: ret
} // end of method Program::Main
.method public hidebysig specialname rtspecialname
instance void .ctor() cil managed
{
// Code size 7 (0x7)
.maxstack 8
IL_0000: ldarg.0
IL_0001: call instance void [mscorlib]System.Object::.ctor()
IL_0006: ret
} // end of method Program::.ctor
} // end of class TestConsole.Program
// =============================================================