建议6: 区别readonly和const的使用方法

建议6: 区别readonly和const的使用方法

很多初学者分不清readonly和const的使用场合。在我看来,要使用const的理由只有一个,那就是效率。但是,在大部分应用情况下,“效率”并没有那么高的地位,所以我更愿意采用readonly,因为readonly赋予代码更多的灵活性。const和readonly的本质区别如下:

const是一个编译期常量,readonly是一个运行时常量。

const只能修饰基元类型、枚举类型或字符串类型,readonly没有限制。

关于第一个区别,因为const是编译期常量,所以它天然就是static的,不能手动再为const增加一个static修饰符,下面的代码将会导致编译通不过:

 
 
  1. static const int ConstValue = 100; //常量“ConstValue”不能标记为 static 

而之所以说const变量的效率高,是因为经过编译器编译后,我们在代码中引用const变量的地方会用const变量所对应的实际值来代替,如:

 
 
  1. Console.WriteLine(ConstValue); 

和下面的代码生成的IL代码是一致的:

 
 
  1. Console.WriteLine(100); 

readonly变量是运行时变量,其赋值行为发生在运行时。readonly的全部意义在于,它在运行时第一次被赋值后将不可以改变。当然,“不可以改变”分为两层意思:

1)对于值类型变量,值本身不可改变(readonly,只读)。

2)对于引用类型变量,引用本身(相当于指针)不可改变。

来看值类型变量,查看如下代码:

 
 
  1. class Sample  
  2. {  
  3.     public readonly int ReadOnlyValue ;  
  4.  
  5.     public Sample( int value)  
  6.     {  
  7.         ReadOnlyValue = value;  
  8.     }  

Sample的实例ReadOnlyValue在构造方法中被赋值后就不可以改变,下面的代码将不会编译通过:

 
 
  1. Sample sample = new Sample(200);  
  2. sample.ReadOnlyValue = 300;         //无法对只读的字  
  3. //段赋值(构造函数或变量初始值指定项中除外) 

针对引用类型变量,查看如下代码:

 
 
  1. class Sample2  
  2. {  
  3.     public readonly Student ReadOnlyValue;  
  4.  
  5.     public Sample2(Student value)  
  6.     {  
  7.         ReadOnlyValue = value;  
  8.     }  

Sample2的ReadOnlyValue是一个引用类型变量,赋值后,变量不能再指向任何其他的Student实例,所以,下面的代码将不会编译通过:

 
 
  1. Sample2 sample2 = new Sample2(new Student() { Age = 10 });  
  2. sample2.ReadOnlyValue = new Student() { Age = 20 };  
  3.     //无法对只读的字段赋值(构造函数或变量初始值指定项中除外) 

但是,我们之前已经提到过了,引用本身不可改变,引用所指的实例的值,却是可以改变的,下面的代码将会被允许:
 
 
  1. Sample2 sample2 = new Sample2(new Student() { Age = 10 });  
  2. sample2.ReadOnlyValue.Age = 20

readonly所代表的运行时含义有一个重要的作用,就是可以为每个类的实例指定一个readonly的变量。以Sample这个类为例,可以在运行时生成多个实例,而同时,又可以为每个实例生成自己的readonly变量,如下:
 
 
  1. Sample sample1 = new Sample(100);  
  2. Sample sample2 = new Sample(200);  
  3. Sample sample3 = new Sample(300); 

这就是readonly变量所代表的运行时含义,也正是readonly变量的灵活之处。

注意 有人说,readonly变量不能被重新赋值是不正确的。下面的代码表明了它可以被改变:

 
 
  1. class Program  
  2. {  
  3.     static void Main(string[] args)  
  4.     {  
  5.         Sample sample = new Sample(200);  
  6.         Console.WriteLine(sample.ReadOnlyValue);  
  7.     }  
  8. }  
  9.  
  10. class Sample  
  11. {  
  12.     public readonly int ReadOnlyValue = 100;  
  13.  
  14.     public Sample(int value)  
  15.     {  
  16.         ReadOnlyValue = value;  
  17.     }  

ReadOnlyValue首先在初始值指定项(也称为初始化器)中被赋值为100,后来,在构造方法中又被赋值为200。实际上,应该把初始化器理解成构造方法的一部分,它其实是一个语法糖。在构造方法内,我们确实可以多次对readonly赋值。

转自:《编写高质量代码改善C#程序的157个建议》陆敏技

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值