第 7 章 常量和字段

本章内容:

  • 常量
  • 字段

本章介绍如何向类型添加数据成员,具体就是常量和字段。

7.1 常量

常量是值从不变化的符号。定义常量符号时,它的值必须能在编译时确定。确定后,编译器将常量值保存到程序集元数据中。这意味着只能定义编译器识别的基元类型的常量。在 C# 中,以下类型是基元类型,可用于定义常量:BooleanCharByteSByteInt16UInt16Int32Uint32Int64UInt64SingleDoubleDecimalString。然而,C# 也允许定义非基元类型的常量变量(constant variable),前提是把值设为null:

1
2
3
4
5
6
7
using System;

public sealed class SomeType {
    // SomeType 不是基元类型,但 C# 允许
    // 值为 null 的这种类型的常量变量
    public const SomeType Empty = null;
}

由于常量值从不变化,所以常量总是被视为类型定义的一部分。换言之,常量总是被视为静态成员,而不是实例成员。定义常量将导致创建元数据。

代码引用常量符号时,编译器在定义常量的程序集的元数据中查找该符号,提取常量的值,将值签入生成的 IL 代码中。由于常量的值直接嵌入代码,所以在运行时不需要为常量分配任何内存。除此之外,不能获取常量的地址,也不能以传引用的方式传递常量。这些限制意味着常量不能很好地支持跨程序集的版本控制。因此,只有确定一个符号的值从不变化才应定义常量。(将 MaxInt6定义为32767就是一个很好的例子)。下面来演示我刚才所说的内容。首先,请输入以下代码,并将其编译成一个 DLL 程序集。

1
2
3
4
5
6
7
using System;

pulbic sealed class SomeLibraryType {
    // 注意:C# 不允许为常量指定 static 关键字,
    // 因为常量总是隐式为 static
    public const Int32 MaxEntriesInList = 50;
}

接着用以下代码生成一个应用程序程序集:

用 csc.exe 的 /r 开关来引用刚才的.dll 文件。

1
2
3
4
5
6
7
using System;

public sealed class Program {
    public static void Main() {
        Console.WriteLine("Max entries supported in list: " + SomeLibraryType.MaxEntriesInList);
    }
}

注意代码引用了在 SomeLibraryType 类中定义的 MaxEntriesInList 常量。编译器生成引用程序代码时,会注意到 MaxEntriesInList 是值为 50 的常量符号,所以会将 Int3250 嵌入应用程序的 IL 代码,如下所示。事实上 ,在生成了应用程序程序集之后,运行时根本不会加载 DLL 程序集,可以把它从磁盘上删除。

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
.method public hidebysig static void Main() cil managed 
{
    entrypoint 
    // Code size            25 (0x19)

    .maxstack 8 
    IL_0000: nop 
    IL_0001: ldstr          "Max entries supported in list: "
    IL_0006: ldc.i4.s       50
    IL_0008: box            [mscorlib]System.Int32
    IL_000d: call           string [mscorlib]System.String::Concat(object, object)
    IL_0012: call           void [mscorlib]System.Console::WriteLine(string)
    IL_0017: nop 
    IL_0018: ret
} // end of method Program::Main

这个例子清楚地展示了版本控制问题。如果开发人员将常量 MaxEntriesInList 的值更改为 1000,并且只是重写生成 DLL 1000,并且只是重写生成 DLL 程序集,那么应用程序集不受任何影响。应用程序集不受任何影响。应用程序要获得新值,也必须重新编译。如果希望在运行时从一个程序集中提取另一个程序集中的值,那么不应该使用常量,而应该使用 readonly 字段,详情参见下一节。

7.2 字段

字段是一种数据成员,其中容纳了一个值类型的实例或者对一个引用类型的引用。表 7-1 总结了可应用于字段的修饰符。

表 7-1 字段修饰符

CLR 术语 C#术语 说明
Static static 这种字段是类型状态的一部分,而不是对象状态的一部分
Instance (默认) 这种字段与类型的一个实例关联,而不是类型本身关联
InitOnly readonly 这种字段只能由一个构造器方法中的代码写入
Volatile volatile 编译器、CLR 和硬件不会对访问这种字段的代码执行“线程不安全”的优化措施。只有以下类型才能标记为 volatile:所有引用类型,SingleBooleanByteSBtypeInt16UInt16Int32UInt32Char,以及基础类型为ByteSByteInt16UInt16Int32UInt32的所有枚举类型。volatile 字段将在第 29 章“基元线程同步构造”讨论

文档将 volatile 翻译为“可变”。其实它是“短暂存在”、“易变”的意思,因为可能有多个线程想都对这种字段进行修改,所以“易变”或“易失”更佳。——译注

如表 7-1 所示, CLR 支持类型(静态)字段和实例(非静态)字段。如果是类型字段,容纳字段数据所需的动态内存是在类型对象中分配的,而类型对象是在类型加载到一个 AppDomain 时创建的(参见第 22 章 “CLR 寄宿和 AppDomain”)。那么,什么时候将类型加载到一个 AppDomain 中呢?这通常是在引用了该类型的任何方法首次进行 JIT 斌阿姨的时候,如果是实例字段,容纳字段数据所需的动态内存是在构造类型的实例时分配的。

由于字段存储在动态内存中,所以它们的值在运行时才能获取。字段还解决了常量存在的版本控制问题。此外,字段可以是任何数据类型,不像常量那样仅仅局限于编译器内置的基元类型。

CLR 支持 readonly 字段和 read/write 字段。大多数字段都是 read/write字段,意味着在代码执行过程中,字段值可多次改变。但readonly 字段只能在构造器方法中写入。(构造器方法只能调用一次,即对象首次创建时。)编译器和验证机制确保 readonly 字段不会被构造器以外的任何方法写入。注意,可利用反射来修改readonly字段。

现在,让我们以 7.1 节“常量”的代码为例,使用一个静态 readonly 字段来修正版本控制问题。下面是新版本 DLL 程序集的代码:

1
2
3
4
5
6
using System;

public sealed class SomeLibraryType {
    // 字段和类型关联必须使用 static 关键字
    public static readonly Int32 MaxEntriesInList = 50;
}

这是唯一需要修改的,应用程序的代码不必修改。但是,为了观察新的行为,必须重新生成它。当应用程序的 Main 方法运行时,CLR 将加载DLL 程序集(现在运行时需要该程序集了),并从分配给它的动态内存中提取 MaxEntriesInList 字段的值。当然,该值是 50。假设 DLL 程序集的开发人员将 50 改为 1000,并重新生成程序集,当应用程序代码重新执行时,它将自定提取字段的新值 1000。应用程序不需要重新生成,可以直接运行(尽管性能会受到一点影响)。要注意的是,当前假定的是 DLL 程序集的新版本没有进行强命名,而且应用程序的版本策略是让CLR 加载这个新版本。

下例演示了如何定义一个与类型本身关联的 readonly 静态字段。还定义了 read/write 静态字段,以及 readonlyread/write 实例字段。

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
public sealed class SomeType {
    // 这是一个静态 readonly 字段:在运行时对这个类进行初始化时,
    // 它的值会被计算并存储到内存中
    public static readonly Random s_random = new Random();

    // 这是一个静态 read/write 字段
    private static Int32 s_numberOfWrites = 0;

    // 这是一个实例 readonly 字段
    public readonly String Pathname = "Untitled";

    // 这是一个实例 read/write 字段
    private System.IO.FileStream m_fs;

    public SomeType(String pathname) {
        // 改行修改只读字段 Pathname,
        // 在构造器中可以这样做
        this.Pathname = pathname;
    }

    public String DoSomething() {
        // 该行读写静态 read/write 字段
        s_numberOfWrites = s_numberOfWrites + 1;

        // 改行读取 readonly 实例字段
        return Pathname;
    }
}

在上述代码中,许多字段都是内联初始化的。C#允许使用这种简便的内联初始化语法来初始化类的常量、read/write 字段和 readonly 字段。第 8 章“方法”会讲到,C# 实际是在构造器中对字段进行初始化的,字段的内联初始化只是一种语法上的简化。另外,在 C# 中初始化字段时,如果使用内联语法,而不是在构造器中赋值,有一些性能问题需要考虑。这些性能问题也将在第 8 章讨论。

内联(inline)初始化是指在代码中直接赋值来初始化,而不是将对构造器的调用写出来。——译注

重要提示 当某个字段是引用类型,并且该字段被标记为 readonly 时,不可改变的是引用,而非字段引用的对象。以下代码对此进行了演示:

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
public sealed class AType {
    // InvalidChars 总是引用同一个数组对象
    public static readonly Char[] InvalidChars = new Char[] { 'A', 'B', 'C' };
}

public sealed class AnotherType {
    public static void M() {
        // 下面三行代码是合法的,可通过编译,并可成功
        // 修改 InvalidChars 数组中的字符
        AType.InvalidChars[0] = 'X';
        AType.InvalidChars[1] = 'Y';
        AType.InvalidChars[2] = 'Z';

        // 下一行代码是非法的,无法通过编译,
        // 因为不能让 InvalidChars 引用别的什么东西
        // A static readonly field cannot be assigned to(except in a static constructor or a variable initializer)
        AType.InvalidChars = new Char[] { 'X', 'Y', 'Z' };
    }
}