在C#中常量和只读之间有什么区别?

你什么时候会用一个代替另一个?


当前回答

const是编译时常量,而readonly允许在运行时计算值并在构造函数或字段初始值设定项中设置。因此,“const”总是常量,但“readonly”在赋值后是只读的。

C#团队的埃里克·里佩尔(Eric Lippert)有更多关于不同类型不变性的信息。

其他回答

常量:无法在任何地方更改。

只读:此值只能在构造函数中更改。无法在正常功能中更改。

有一个只读的小陷阱。只读字段可以在构造函数中设置多次。即使在两个不同的链式构造函数中设置了该值,它仍然是允许的。

public class Sample {
    private readonly string ro;

    public Sample() {
        ro = "set";
    }

    public Sample(string value) : this() {
        ro = value; // this works even though it was set in the no-arg ctor
    }
}

还有一个陷阱:只读值可以通过反射被“狡猾”的代码改变。

var fi = this.GetType()
             .BaseType
             .GetField("_someField", 
                       BindingFlags.Instance | BindingFlags.NonPublic);
fi.SetValue(this, 1);

我可以使用反射更改C#中的私有只读继承字段吗?

常量成员是在编译时定义的,不能在运行时更改。常量使用const关键字声明为字段,并且必须在声明时进行初始化。

public class MyClass
{
    public const double PI1 = 3.14159;
}

只读成员就像一个常量,因为它代表一个不变的值。不同之处在于,只读成员可以在运行时在构造函数中初始化,也可以在声明时初始化。

public class MyClass1
{
     public readonly double PI2 = 3.14159;

     //or

     public readonly double PI3;

     public MyClass2()
     {
         PI3 = 3.14159;
     }
}

常量

它们不能声明为静态(它们是隐式静态的)常数的值在编译时计算常量仅在声明时初始化

只读的

它们可以是实例级的,也可以是静态的该值在运行时计算readonly可以在声明中初始化,也可以通过构造函数中的代码初始化

何时使用常量或只读常量编译时常量:绝对常量,值在声明期间设置,在IL代码本身中只读的运行时常量:可以通过配置文件(即App.config)在构造函数/init中设置,但一旦初始化就不能更改