C#中const和readonly有什么区别?


1361

constreadonlyC#和有什么不一样?

您什么时候可以使用另一个?


我不得不看一些答案才能找到此链接,但这是一个很好的链接。 埃里克·利珀特(Eric Lippert)对C#不变性的看法
Frank Bryce 2015年

2
@donstack,实际上根据C#参考可以在字段声明和构造函数中多次分配和重新分配只读字段。
马克斯,

Answers:


1289

除了明显的区别

  • 可以在定义constVS readonly值时声明值,该值可以动态计算,但需要在构造函数退出之前分配。在冻结之后。
  • 'const'是隐式的static。您使用一种ClassName.ConstantName符号来访问它们。

有细微的差别。考虑在中定义的类AssemblyA

public class Const_V_Readonly
{
  public const int I_CONST_VALUE = 2;
  public readonly int I_RO_VALUE;
  public Const_V_Readonly()
  {
     I_RO_VALUE = 3;
  }
}

AssemblyBAssemblyA在代码中引用并使用这些值。编译后,

  • const值的情况下,就像是一个查找替换,值2被“烘焙”到AssemblyBIL中。这意味着如果明天我将I_CONST_VALUE在未来更新到20。AssemblyB在我重新编译之前仍然有2个
  • 在该readonly值的情况下,它类似于ref存储位置。该值未烘焙到AssemblyB的IL中。这意味着如果内存位置已更新,AssemblyB则无需重新编译即可获取新值。因此,如果I_RO_VALUE将其更新为30,则只需要构建即可AssemblyA。不需要重新编译所有客户端。

因此,如果您确信常量的值不会改变,请使用const

public const int CM_IN_A_METER = 100;

但是,如果您有一个可能会更改的常量(例如,精度),或者有疑问,请使用readonly

public readonly float PI = 3.14;

更新:Aku需要得到提及,因为他首先指出了这一点。另外,我需要在我学到的知识上加点文字。. 有效的C#-Bill Wagner


77
static点似乎是最重要和有用的点consts are implicitly static
LCJ 2013年

28
关于参考值的部分是最重要的部分。常量值可以优化掉。
CodingBarfield 2013年

22
readonly可以在构造函数外部更改变量(反射)。只是编译器试图阻止您在构造函数之外修改var。
Bitterblue 2013年

12
readonly构造函数完成后,即使通过反射也不允许更改@ mini-me 变量。运行时碰巧不执行此操作。运行时也碰巧不强制您不要更改string.Empty"Hello, world!",但是我仍然不会声称这是string.Empty可修改的,或者代码不应该假定该string.Empty字符串始终为零长度。

7
blogs.msmvps.com/jonskeet/2014/07/16/…是一个有趣的只读只读开销
CAD bloke,2014年

275

有一个const的陷阱!如果从另一个程序集中引用一个常量,则其值将直接编译到调用程序集中。这样,当您更新引用程序集中的常量时,在调用程序集中就不会更改它!


8
在反编译(Reflector,ILSpy,..)时,无论是同一程序集还是另一程序集,任何人都不会引用一个常量,因此您根本无法分析常量在已编译代码中的使用。
springy76

159

常数

  • 常量默认为静态
  • 它们在编译时必须有一个值(例如,可以有3.14 * 2,但不能调用方法)
  • 可以在函数中声明
  • 复制到使用它们的每个程序集中(每个程序集都获取值的本地副本)
  • 可以在属性中使用

只读实例字段

  • 在构造函数退出时必须具有设置值
  • 创建实例时进行评估

静态只读字段

  • 当代码执行达到类引用时(在创建新实例或执行静态方法时)进行评估
  • 静态构造函数完成时必须具有评估值
  • 不建议将ThreadStaticAttribute放在这些属性上(静态构造函数将仅在一个线程中执行,并将为其线程设置值;所有其他线程都将未初始化此值)

58

只需添加一下,对于引用类型的ReadOnly仅会使引用变为只读,而不是值。例如:

public class Const_V_Readonly
{
  public const int I_CONST_VALUE = 2;
  public readonly char[] I_RO_VALUE = new Char[]{'a', 'b', 'c'};

  public UpdateReadonly()
  {
     I_RO_VALUE[0] = 'V'; //perfectly legal and will update the value
     I_RO_VALUE = new char[]{'V'}; //will cause compiler error
  }
}

除了string可以用作常量之外,还有其他引用类型吗?
springy76

您可以const使用除字符串以外的其他引用类型,但常量只能具有value null
Mike Rosoft


32

const:无法在任何地方更改。

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


26

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


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
    }
}

26

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

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

readonly构件是一样的,因为它代表了一个不变的值的常数。区别在于readonly成员可以在运行时在构造函数中初始化,也可以在声明成员时初始化。

public class MyClass1
{
     public readonly double PI2 = 3.14159;

     //or

     public readonly double PI3;

     public MyClass2()
     {
         PI3 = 3.14159;
     }
}

const

  • 它们不能被声明为static(它们是隐式静态的)
  • 在编译时评估常量的值
  • 常量仅在声明时初始化

只读

  • 它们可以是实例级别的,也可以是静态的
  • 该值在运行时评估
  • 只读可以在声明中初始化,也可以在构造函数中通过代码初始化

6
它们不能是静态的,它们是静态的。您应该清楚表明自己是否可以宣布static const int i = 0;
nawfal 2013年

您能解释为什么const不能在方法内部进行声明吗?
Minh Tran

21

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

C#小组的Eric Lippert拥有有关不同类型不变性的更多信息


15

这是另一个链接,说明const如何不是版本安全的,或者与引用类型无关。

总结

  • const属性的值是在编译时设置的,在运行时不能更改
  • 常量不能被标记为静态-关键字表示它们是静态的,与只读字段不同。
  • 常量除值(原始)类型外不能为其他任何东西
  • readonly关键字将字段标记为不可更改。但是,可以在类的构造函数中更改属性
  • readonly only关键字也可以与static组合使用,以使其与const(至少在表面上)具有相同的作用。当您查看两者之间的IL时,有明显的区别
  • const字段在IL中被标记为“文字”,而readonly是“ initonly”

11

只读:可以在运行时通过Ctor更改值。但不是通过成员函数

常量:通过消除静态。不能在任何地方(Ctor,Function,运行时等,无处)更改值


感谢您不要让我只为这两个要点阅读4段...
Don Cheadle



5

我们办公室的一位团队成员就何时使用const,static和readonly提供了以下指导:

  • 当您拥有一个可以在运行时知道的类型的变量(字符串文字,int,double,enums ...)时,请使用const,您希望类的所有实例或使用者都可以访问该值不应更改的位置。
  • 当您拥有希望类的所有实例或使用者可以访问值可以更改的位置的数据时,请使用static
  • 当您拥有一个在运行时(对象)时不知道的类型的变量,并且希望类的所有实例或使用者都可以访问该值不应更改的位置时,请使用静态只读
  • 当您有一个实例级别的变量,该变量在对象创建时会知道,该变量不应更改,请使用readonly

最后一点:const字段是静态的,但反之则不正确。


1
我认为您的意思是“逆向”。相反将是“非常量字段不是静态的”。这可能是正确的,也可能不是。相反,“静态字段是(总是)const”是不正确的。
Michael Blackburn

5

它们都是常量,但是在编译时也可以使用const。这意味着差异的一方面是您可以将const变量用作属性构造函数的输入,但不能使用只读变量。

例:

public static class Text {
  public const string ConstDescription = "This can be used.";
  public readonly static string ReadonlyDescription = "Cannot be used.";
}

public class Foo 
{
  [Description(Text.ConstDescription)]
  public int BarThatBuilds {
    { get; set; }
  }

  [Description(Text.ReadOnlyDescription)]
  public int BarThatDoesNotBuild {
    { get; set; }
  }
}

5
  • 何时使用constreadonly

    • const

      • 编译时常量:绝对常量,值在声明期间设置,在IL代码本身中
    • readonly

      • 运行时常量:可以通过config文件(即)在构造函数/ init中设置App.config,但一旦初始化,就无法更改

4

标记为const的变量只不过是强类型的#define宏,在编译时,const变量引用将被内联文字值替换。结果,只能以这种方式使用某些内置的原始值类型。标记为只读的变量可以在构造函数中在运行时设置,并且它们的只读性也可以在运行时强制执行。与此相关的性能开销很小,但这意味着您可以对任何类型(甚至是引用类型)使用只读。

同样,const变量本质上是静态的,而如果需要,只读变量可以是实例特定的。


补充说,const是强类型的 #define宏。否则,我们可能会吓跑所有C或C ++人员。:-)
杰森·贝克

4

CONST

  1. const关键字可以应用于字段或局部变量
  2. 我们必须在声明时分配const字段
  3. 未分配内存,因为编译后const值已嵌入IL代码本身中。就像找到所有出现的const变量并用其值替换一样。因此,编译后的IL代码将具有硬编码的值来代替const变量
  4. C#中的常量默认情况下是静态的。
  5. 该值对于所有对象都是恒定的
  6. 存在dll版本控制问题-这意味着每当我们更改公共const变量或属性时,(实际上,理论上就不应更改它),必须重新构建使用此变量的任何其他dll或程序集。
  7. 只能将C#内置类型声明为常量
  8. 常量字段不能作为ref或out参数传递

只读

  1. readonly关键字仅适用于字段,而不适用于局部变量
  2. 我们可以在声明时或在构造函数中分配只读字段,而不能在任何其他方法中分配。
  3. 为只读字段分配的动态内存,我们可以在运行时获取该值。
  4. 只读属于创建的对象,因此只能通过类的实例进行访问。为了使其成为类成员,我们需要在只读之前添加static关键字。
  5. 该值可能会有所不同,具体取决于所使用的构造函数(因为它属于类的对象)
  6. 如果将非原始类型(引用类型)声明为只读,则仅引用是不可变的,而不是其中包含的对象。
  7. 由于该值是在运行时获取的,因此只读字段/属性没有dll版本控制问题。
  8. 我们可以在构造函数上下文中将readonly字段作为ref或out参数传递。

3

另一个陷阱

由于const实际上仅适用于基本数据类型,因此,如果要使用类,则可能会感到“被迫”使用ReadOnly。但是,请当心陷阱!ReadOnly表示您不能用另一个对象替换该对象(您不能使其引用另一个对象)。但是任何引用该对象的进程都可以自由修改该对象内部的值!

因此,不要混淆以为ReadOnly意味着用户不能更改任何东西。C#中没有简单的语法来防止类的实例更改其内部值(据我所知)。


是的,那更像是一个一般主题。如果您具有仅公开数组列表的get属性,则仍可以修改该数组列表。您不能为该属性设置另一个arraylist,但是不能阻止用户更改arraylist。
Gishu

3

A const必须是硬编码的readonly可以在类的构造函数设置


3

C#.Net中的const和readonly字段之间存在显着差异

const默认情况下是静态的,需要使用常量值进行初始化,以后将无法对其进行修改。构造函数中也不允许更改值。并非所有数据类型都可以使用它。对于exDateTime。不能与DateTime数据类型一起使用。

public const DateTime dt = DateTime.Today;  //throws compilation error
public const string Name = string.Empty;    //throws compilation error
public readonly string Name = string.Empty; //No error, legal

readonly可以声明为静态的,但不是必需的。声明时无需初始化。可以使用构造函数来分配或更改其值。因此,当用作实例类成员时,它具有优势。两种不同的实例可能具有不同的只读字段值。对于前-

class A
{
    public readonly int Id;

    public A(int i)
    {
        Id = i;
    }
}

然后可以使用即时特定值初始化readonly字段,如下所示:

A objOne = new A(5);
A objTwo = new A(10);

在这里,实例objOne的readonly字段值为5,而objTwo的值为10。使用const不可能。


2

常量将作为文字值编译到使用者中,而静态字符串将用作对所定义值的引用。

作为练习,尝试创建一个外部库并在控制台应用程序中使用它,然后更改该库中的值并重新编译(不重新编译使用者程序),将DLL放到目录中并手动运行EXE,您应该找到常量字符串不变。


我真心怀疑那是真的...我去检查一下。
ljs

这是对50层具体的方法来提高你的C#一个- amazon.co.uk/Effective-Specific-Ways-Improve-Your/dp/0321245660/...
拉斯凸轮


@Andrew Hare-是的,我刚刚检查了一下。我很惊讶,这是一个真正的陷阱,对此我感到非常惊讶,惊讶的是……!
ljs

但是,我确实反对在这里使用单词指针。这不是一个指针,它是一个参考,而且在C#中的差异,你可以操纵在不安全模式下非托管的指针,因此两者之间的区别是很重要的。
ljs

2

不变

定义时,我们需要将值提供给const字段。然后,编译器将常量的值保存在程序集的元数据中。这意味着只能为原始类型(如boolean,char,byte等)定义常量。常量始终被视为静态成员,而不是实例成员。

只读

只读字段只能在运行时解析。这意味着我们可以使用声明字段类型的构造函数为值定义一个值。验证由编译器完成,除了构造方法外,只读字段未由任何其他方法写入。

本文在此处详细介绍了两者


1

原则上; 您可以在运行时将静态只读字段的值分配给非常数值,而必须为const分配常数。


1

常量和只读是相似的,但是它们并不完全相同。const字段是编译时常量,这意味着可以在编译时计算该值。只读字段可启用其他方案,在该方案中,必须在类型构造期间运行某些代码。构造后,只读字段无法更改。

例如,const成员可用于定义成员,例如:

struct Test
{
    public const double Pi = 3.14;
    public const int Zero = 0;
}

因为像3.14和0这样的值是编译时常量。但是,请考虑定义一种类型并想提供一些预制实例的情况。例如,您可能想要定义一个Color类,并为常见的颜色(如黑色,白色等)提供“常量”。使用const成员是不可能的,因为右侧不是编译时常量。可以使用常规静态成员执行此操作:

public class Color
{
    public static Color Black = new Color(0, 0, 0);
    public static Color White = new Color(255, 255, 255);
    public static Color Red = new Color(255, 0, 0);
    public static Color Green = new Color(0, 255, 0);
    public static Color Blue = new Color(0, 0, 255);
    private byte red, green, blue;

    public Color(byte r, byte g, byte b) {
        red = r;
        green = g;
        blue = b;
    }
}

但是,没有什么可以阻止Color的客户使用它了,也许可以通过交换Black和White值来避免。不用说,这会使Color类的其他客户端感到震惊。“只读”功能解决了这种情况。通过在声明中简单地引入readonly关键字,我们保留了灵活的初始化,同时防止了客户端代码的混乱。

public class Color
{
    public static readonly Color Black = new Color(0, 0, 0);
    public static readonly Color White = new Color(255, 255, 255);
    public static readonly Color Red = new Color(255, 0, 0);
    public static readonly Color Green = new Color(0, 255, 0);
    public static readonly Color Blue = new Color(0, 0, 255);
    private byte red, green, blue;

    public Color(byte r, byte g, byte b) {
        red = r;
        green = g;
        blue = b;
    }
}

有趣的是,const成员始终是静态的,而readonly成员可以是静态的,也可以不是静态的,就像常规字段一样。

可以将单个关键字用于这两个目的,但这会导致版本问题或性能问题。假设一会儿,我们为此关键字(const)使用了一个关键字,并且开发人员写道:

public class A
{
    public static const C = 0;
}

另一位开发人员编写了依赖于A的代码:

public class B
{
    static void Main() {
        Console.WriteLine(A.C);
    }
}

现在,生成的代码是否可以依赖AC是编译时常量的事实?即,AC的使用是否可以简单地替换为值0?如果对此说“是”,则意味着A的开发人员无法更改AC初始化的方式-未经许可,这会束缚A的开发人员的双手。如果您对这个问题说“否”,那么将错过重要的优化。也许A的作者肯定AC将始终为零。const和readonly的使用允许A的开发人员指定意图。这使得更好的版本控制行为和更好的性能。


1

ReadOnly:仅从类的构造函数初始化一次值。
const:可以在任何函数中初始化,但只能初始化一次


1

不同之处在于,静态只读字段的值是在运行时设置的,因此对于程序的不同执行,它可以具有不同的值。但是,const字段的值设置为编译时间常数。

切记:对于引用类型,在两种情况下(静态和实例),readonly修饰符仅防止您为该字段分配新引用。具体来说,它不会使引用指向的对象不可变。

有关详细信息,请参阅有关此主题的C#常见问题解答:http : //blogs.msdn.com/csharpfaq/archive/2004/12/03/274791.aspx


1

常量变量在编译时声明和初始化。病房之后不能更改该值。只读变量将仅从类的静态构造函数中初始化。只读仅在我们要在运行时分配值时使用。


1

常量:应用程序生命周期内的绝对常数。

只读:可以在运行时更改。


1
您可以更改的“只读”定义存在缺陷。我猜“更改”是指“设置”,例如“可以在运行时设置”。
艾哈迈德(Ahmed)

0

人们上面所说的补充一件事。如果您有一个包含只读值的程序集(例如,readonly MaxFooCount = 4;),则可以通过发送具有不同值的该程序集的新版本来更改调用程序集所看到的值(例如,只读MaxFooCount = 5;)。

但是使用const时,将在编译调用方时将其折叠到调用方的代码中。

如果您已达到C#熟练程度,则可以阅读Bill Wagner的书《有效的C#:改善C#的50种特定方法》,该书将详细回答此问题(以及其他49项)。

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.