[CLR via C#]7. 常量和字段

原文:[CLR via C#]7. 常量和字段

7.1 常量

  常量(constant)是一个特殊的值,它是一个从不变化的值。

  在定义常量时,它的值必须在编译时确定。确定之后,编译器将常量的值保存到程序集的元数据中。这就意味着只能为编译器认定的基元类型定义常量。

 

  C#是允许定义一个非基元类型的常量变量(constant variable),但这个值应设为null。

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

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

  代码引用一个常量符号时,编译器会在定义常量的程序集的元数据中查找该符号,提取常量的值,并将值嵌入生成的IL代码中。由于常量的值直接嵌入代码中,所以运行时不需要为常量额外分配内存。

  除此之外,不能获取常量的地址,也不能以传递引用的方式传递常量。

 

  基于上一条,假如A程序集只依赖于B程序集中的常量,那么编译后,即使删除B程序集,A程序集也不会受到影响,也是能找到B程序集中定义的常量值。

 

7.2 字段

 

  字段(field)是一种数据成员,其中容纳了一个值类型的实例或者对一个引用类型的引用。

 

  下表总结了应用于字段的修饰符:

CLR术语 C#术语 说明
Static static 这种字段是类型状态的一部分,而不是对象状态的一部分
Instance 默认 这种字段与类型的一个实例关联,而不是与类型本身关联
InitOnly readOnly 这种字段只能由一个构造器方法中的代码写入
Volatile volatile 看到访问这种字段的代码,编译器、CLR或硬件就不会执行一些"线程不安全"的优化措施

   

  CLR支持类型(静态)字段和实例(非静态)字段。对于类型字段,用于容纳字段数据的动态内存是在类型对象中分配的,而类型对象是在类型加载到一个AppDomain时创建的。什么时候要将类型加载到一个AppDimain中呢?通常是在引用了该类型的任何方法首次进行JIT编译的时候。

 

  对于实例字段,用于容纳字段数据的动态内存则是在构造类型的一个实例时分配的。

 

  由于字段存储在动态内存中,所有它们的值在运行时才能获取。字段还解决了常量存在的版本控制的问题。此外,字段可以是任何数据类型。

 

  CLR支持readonly字段和read/write字段。大多数字段是read/write字段,这意味着在代码执行过程中,字段可以多次改变。但是,readonly字段只能在一个构造器方法中写入(在这有人会有疑问,我可以直接定义readonly的值啊?比如private readonly int ss = 123;不必从构造器方法中写入?这问题,以后会提到)。注意,可以利用反射来修改readonly字段。

 

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

public sealed class SomeType {
    // 这是一个静态readonly字段:在运行时对Random类进行初始化
    // 它的值会被计算并存储到内存中
    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字段。在后面会讲到,C#实际是在构造器中对字段进行初始化的,字段的内联初始化只是一种语法上的简化而已。另外,在C#中使用内联初始化,有一些性能问题需要考虑,这些以后会着重讲解。

  注意,当某个字段是引用类型,并且该字段标记为readonly时,那么不可改变的引用,而非字段引用的值。

internal static class ReadOnlyReferences {
   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 引用别的什么东西
         //AType.InvalidChars = new Char[] { 'X', 'Y', 'Z' };
      }
   }
}

 

时间: 2024-11-03 11:34:59

[CLR via C#]7. 常量和字段的相关文章

CLR笔记:7.常量和字段

1.常量 常量是永远不会改变的符号.它的值必须在编译时就确定.编译后,CLR将常量的值保存在Assembly的 元数据中,这意味着常量必须是基元类型. 常量是类型的一部分,总是被当成静态成员,但并不显示声明为static. public const Int32 MaxEntriesList = 100; 当代码引用常量时,CLR在元数据中查找该符号,将提取的常量值嵌入到IL中,所以常量没有地址以及 相应的分配内存,而且不能通过引用传递变量,也就是说,在导入一个外部的DLL时,就已经将其中的常 量

C# 类型和成员基础以及常量、字段、属性

类型和成员基础 在C#中,一个类型内部可以定义多种成员:常量.字段.实例构造器.类型构造器(静态构造器).方法.操作符重载.转换操作符.属性.事件.类型. 类型的可见性有public和internal(默认)两种,前者定义的类型对所有程序集中的所有类型都可见,后者定义的类型只对同一程序集内部的所有类型可见: public class PublicClass { } //所有处可见 internal class ExplicitlyInternalClass { } //程序集内可见 class

c# 常量和字段_C#教程

它的值是在编译时确定的.编译器将常量保存到程序集的元数据中,所有只能是编译器认识的基元类型作为常量. 常量被看成类的一部分,是看出静态成员. 代码引用一个常量符号,会在定义常量的元数据中查找该符号,提取之,并嵌入代码,生成的IL中是值本身. 在c#中使用的是const关键字. 字段:已一种数据成员,可以容纳任何的数据类型,不仅仅想常量一样只能存储基元类型. CLR支持类型字段和实例字段 类型字段:用于容纳字段数据的动态内存是在类型对象中分配的, 而类对象是在类型加载到一个AppDomain中创建

CLR笔记系文章目录索引

CLR笔记:18.可空值类型 CLR笔记:17.自定义属性 CLR笔记:16.泛型 CLR笔记:15.委托 CLR笔记:14.接口 CLR笔记:13.数组 CLR笔记:12.枚举类型和位标志 CLR笔记:11.字符串 CLR笔记:10.事件 CLR笔记:9.Property CLR笔记:8.方法 CLR笔记:7.常量和字段 CLR笔记:6.类型和成员基础 CLR笔记:5.基元,引用和值类型 CLR笔记:4.类型基础 CLR笔记:3.共享程序集合强命名程序集 CLR笔记:2.生成,打包,部署,管理

[CLR via C#]6. 类型和成员基础

原文:[CLR via C#]6. 类型和成员基础 6.1 类型的各种成员 在一个类型中,可以定义0个或多个以下种类的成员: 1)常量    常量就是指出数据值恒定不变的符号.这些符号通常用于使代码更容易阅读和维护.常量通常与类型关联,而不与类型的实例关联.从逻辑上讲,常量始终是静态成员. 2)字段    字段表示一个只读或可读/写的数据值.字段可以是静态的,这时是类型状态的一部分:字段也可以是实例(非静态)的,这时字段是对象状态的一部分.强烈建议将字段声明成为私有字段,防止类型或对象状态被外部

[CLR via C#]13. 接口

原文:[CLR via C#]13. 接口 一.类和接口继承 在Microsoft.Net Framwork中,有一个名为System.Object的类,它定义了4个公共实例方法:ToString, Equals, GetHashCode和GetType.该类是其他所有类的根或者说最终基类.换言之,所有类都继承了Object的4个实例方法.这还意味着能操作Object类的实例的代码实际能操作任何类的实例. 在CLR中,任何类都肯定从一个类也只能从一个类(而且只能从Objetc派生的类)派生的.这

CLR编程

问题描述 请问CLR编程里怎么样处理静态字段和静态函数呢?我在C#下可以用static声明一个静态字段或者函数但是在CLR下什么静态字段时候却出现了以下的错误:错误1由于safe程序集'TD'中的类型'TopServer'具有静态字段'AddString',CREATEASSEMBLY失败.safe程序集中静态字段的属性在VisualC#中必须标记为readonly,在VisualBasic中必须标记为ReadOnly,或者在VisualC++和中间语言中标记为initonly.TD请问怎么处理

针对 Java 开发人员的 C# 编程语言

编程 本文讨论 Java 和 C# 之间的异同点,目的在于当迁移到 .NET 时,让 Java 开发人员掌握所涉及的一些知识.Java 和 C# 之间的主要相似点是: • Java 和 C# 都源于 C++,并且共有 C++ 的一些特征. • 两种语言都需要编译成中间代码,而不是直接编译成纯机器码.Java 编译成 Java 虚拟机 (Java Virtual Machine, JVM) 字节码,而 C# 则编译成公共中间语言 (Common Intermediate Language, CIL

《.net编程先锋C#》第四章 C#类型(转)

编程 第四章 C#类型 既然你知道了怎样创建一个简单的C#程序,我将会给你介绍C#的类型系统.在这一章中,你学到如何使用不同的值和引用类型,加框和消框机制能为你作些什么.尽管这一章的不侧重于例子,但你可以学到很多重要的信息,关于如何创建现成类型的程序.4.1 值类型 各种值类型总是含有相应该类型的一个值.C#迫使你初始化变量才能使用它们进行计算-变量没有初始化不会出问题,因为当你企图使用它们时,编译器会告诉你. 每当把一个值赋给一个值类型时,该值实际上被拷贝了.相比,对于引用类型,仅是引用被拷贝