C# · 12月 21, 2021

关于“属性”的几个问题,也许面试会问到哦~

       这些天太忙了都没更新博客了,这篇我们继续聊聊“属性”,大家都知道,属性其实分两种,无参属性和有参属性,顾名思义

无参属性就是我们平时用到的 “属性”,有参属性就是我们所说的 “索引器”,

Age { ; [ i] { { i + .Empty; } }

乍一看这两个还是蛮像的,本质上来说这两个都是getXXX,setXXX方法,只是编译器为了提高我们的开发效率而做的语法糖。

好,下面回答几个小问题,当然是我自己的个人见解。

一般来说,一个类中都存在一个描述类的状态数据,我们也可以认为是元数据,这些元数据是不可以被轻易修改的,一但

      被错误的修改,就会导致类的破坏,所以建议在字段中加一层壳,由属性来提供高层访问。

   举个例子:Person的Age字段不能设为150的非法数据,这个时候我们就可以在属性的set方法上进行过滤了。

(value age = }

}

其实这个问题问的好,如果你是平时用用而没有用IL看一下的话,可能还真被蒙到了,既然说到了IL,那就用IL看一下。

从IL上可以清楚的看到其实编译器给我们生成了一个私有的k__backingField 字段。

b = Bird { Name = ,Age =

要看有没有区别,我们得要看到底这个“类型初始化器”到底干了些什么?老规矩,我们看看IL代码。

从IL上可以看出,两个nop之间,我们调用了构造函数(ctor),并且先后调用了set_Name,set_Age方法,所以本质上来说,

“类型初始化器”只是一个语法糖,跟我们手工在构造函数中初始化一样。

这个问题好,其实你可以发现,我们在定义一个索引器的时候,根本就没有定义索引器的名字,而是直接用this,重点

    就在这里,我们知道this表示当前实例的上下文,导致我们的[]只能用到类型的实例上,也就做不了将[]用到类型上。

[ { i + }

[ { i + get_Item( }

这个问题也是蛮有意思的,最常见的做法就是手工修改我们自己定义的方法名,但是我们这里可不可以另辟蹊径呢?我们在写

      WCF的时候,可能会遇到给方法标记别名的情况,然后我们就用OperationContract给方法换一个名字,现在估计就有人想到

     了我是不是也可以给“索引器”加上别名?确实可以这样,在这里我们可以用IndexerName来完成。

[ { i + }

然后我们再看看IL代码,就这样成功的修改了索引器的方法名。