C#中字段、属性、只读、构造函数赋值、反射赋值的问题

作者:无名 - C#教程 -

C#中字段、属性和构造函数赋值的问题提出问题如下所述:

首先提出几个问题:

1、如何实现自己的注入框架?

2、字段和自动属性的区别是什么?

3、字段和自动属性声明时的直接赋值和构造函数赋值有什么区别?

4、为什么只读字段和只读自动属性(只有get没有set访问器)都可以在构造函数中进行赋值?

5、反射可以给只读字段或者只读属性进行赋值吗?

6、自动属性和普通属性的区别?

这些问题是我在试着写自己的注入实现时遇到的问题。这些问题应该在学习C#时的第一节课就应该学到了,我看网上还有人分享说他在面试时遇到面试官问为什么只读字段和只读自动属性可以在构造函数中进行赋值,他没有回答上来,然后他写文章探讨这个问题,却没有得出一个明显的答案,实在可惜。网上关于只读属性有些是写ReadOnly特性的,读到这些文章直接跳过吧,老版本的C#现在看也没什么帮助。

给出答案

2、属性比字段多了get/set访问器;字段是在内存中声明的一个内存空间,可以实实在在的存储值;属性像字段一样使用,却可以有自己的代码段,能赋值取值,是因为访问属性就是调用属性的get/set方法对字段进行取值赋值(或者不操作字段);在MSDN上,建议字段作为类的私有变量使用private/protected修饰,属性则往往作为共有属性使用public修饰;字段的读取和操作都是直接操作内存,属性是调用get/set访问器,所以字段比属性快。

3、准确来说,没有区别。区别仅仅是直接赋值先执行,构造函数赋值后执行。在生成的IL中间语言(C#代码先编译成IL代码,然后才编译成汇编语言)中,字段直接赋值和构造函数赋值是在同一个代码段中(构造函数中)的。

4、这个问题可以和上面的问题联合起来回答。构造函数作为实例化一个类的入口,是最先访问的。字段的直接赋值其实也是放在构造函数中执行的,所以才说直接赋值和构造函数赋值没有区别。“只读”的限制只是由C#编译器(CLR)维护的,我觉得全名应该叫做“除构造函数外只读”更加准确,这是C#语法的规则记住就行(这是当然,直接赋值其实是放在构造函数中进行赋值的,如果构造函数不能赋值那只读字段没有值和没有声明一样);

5、这个问题又可以和上面的问题联系起来一起回答。通过反射可以给自读字段赋值但是无法给只读属性进行赋值(不相信的可以试一下)。对只读字段的赋值是因为绕过了C#编译器(CLR)的只读显示,对只读属性赋值的话是还是调用set访问器对字段进行赋值,因为没有set访问器所以允许后会报错。那么问题来了,那为什么只读自动属性没有set访问器还可以在构造函数中赋值呢?其实只读自动属性在构造函数中进行赋值,实质上是对字段进行赋值,和属性的get/set访问器没有关系。

6、区别是什么?上面一直强调自动属性,是因为自动属性和普通属性不一样,比如只读普通属性(没有set访问器)无法在构造函数中赋值。在没有自动属性之前,普通属性使用步骤是首先声明一个字段如_id,然后声明一个属性Id,在get和set访问器中做一些操作,这些操作大多数是对字段_id的操作,但是有时候和字段没有关系。普通属性可以像字段一样通过“.”的方式调用,但又像方法一样具有代码段(普通属性从来不开辟内存空间)。

但是C#3.0之后引入了自动属性,声明方式如public int id { get; set; },C#6.0之后又有了public string FirstName { get; set; } = "Jane"。自动属性肯定开辟了内存空间然后才有了自动属性的直接赋值。其实在类中声明自动属性会在编译成IL中间语言中声明一个隐藏字段,然后生成隐藏字段的get/set方法,然后生成get/set访问器。这里可以解释为什么只读普通属性无法在构造函数中赋值(和直接赋值)而只读自动属性可以在构造函数中赋值(和直接赋值),因为不论直接赋值还是在构造函数中赋值,生成的IL代码中的构造函数中,操作的都是隐藏字段,并没有访问属性的set访问器。(注意这里只是说的类中的自动属性,接口中也是可以有自动属性的,但是接口的自动属性并不会生成隐藏字段只是定义get/set访问器)

开始解释

通过C#生成的IL中间语言代码可以知道的更清楚

 public class User
 {
  public int id = 0;
  public int age { get; set; } = 1;
  public User()
  {
   id = 2;
   age = 3;
  }
 }

可以看到,自动属性会生成一个名称为 '<age>k__BackingField' 的隐藏私有字段+私有字段的get/set方法+属性代码段;

可以看到IL代码生成了User的构造函数 .ctor,ctor是构造函数(Constructor)。

不论直接赋值还是构造函数赋值,都是在.ctor中执行的,并且操作的都是字段,自动属性的赋值操作的是隐藏字段。

 public interface IUser
 {
 int id { get; set; }
 }

可以看到,接口中的自动属性并没有生成隐藏字段。

其他说明

1、上文中提到“反射可以给只读字段进行赋值但是无法给只读属性进行赋值”。无法给只读属性进行赋值是因为没有set访问器。但是我们已经知道了可以给字段赋值,并且只读属性会生成隐藏字段,那我们是不是可以通过给隐藏字段进行赋值间接达到给自动属性赋值的目的呢?答案是可以的!

定义User的只读自动属性

 public class User
 {
  public int age { get; } = 1;
  public User()
  {
   age = 3;
  }
 }

控制台的反射赋值代码:   

 var user = new User();
   try { typeof(User).GetProperty("age").SetValue(user, 9); }
   catch{ Console.WriteLine("只读属性赋值失败");}
   typeof(User).GetField("<age>k__BackingField", BindingFlags.Instance | BindingFlags.NonPublic).SetValue(user,9);
   Console.WriteLine(user.age);
   Console.Read();

运行

2、因为隐藏字段是私有的,所以取到隐藏字段需要  BindingFlags.NonPublic

3、只读自动属性说明不想被访问到那为什么还要给它赋值呢?这个问题……做着玩,项目中我觉得也没有什么用到的机会……

总结

以上所述是IT人知识库小编给大家介绍的C#中字段、属性、只读、构造函数赋值、反射赋值的问题 ,希望对大家有所帮助,如果大家有任何疑问欢迎给我留言,IT人知识库小编会及时回复大家的!

IT人知识库 该篇知识地址:http://www.itpeo.net/12809/519265.html





WPF实现控件拖动的示例代码

实现控件拖动的基本原理是对鼠标位置的捕获,同时根据鼠标按键的按下、释放确定控件移动的幅度和时机。 简单示例: 在... ...

rfedfre

WPF 在image控件用鼠标拖拽出矩形的实现方法

今天有小伙伴问我一个问题,在image控件用鼠标拖拽出矩形,本文告诉大家如何使用鼠标画出矩形 做出来的效果先请大家看一下... ...

rfedfre

怎么使用C#代码创建快捷方式文件详解

前言 快捷方式是一种特殊的文件,扩展名为 lnk。有很多种方式来创建快捷方式,首先我们看一下快捷方式是什么。对快捷方式点... ...

C# 7.2中结构体性能问题的解决方案

前言 在某些使用了readonly关键字的情况下,C#编译器会创建出结构体的防御副本。虽然这个问题已经众所周知并被记录下... ...

.NET Core开发之配置详解

熟悉ASP.NET的开发者一定对web.config文件不陌生。在ASP.NET环境中,要想添加配置参数,一般也都会在此... ...

rfedfre

WPF Slider滑动条的颜色修改方法

效果如下: 鄙人虽然开发WPF有些时间,但之前一直是一些简单Template和Style改改之类的工作,并没有深入研究... ...

C#实现文章添加内链的方法

为什么文章要添加内链?  1.有利于读者      &nb... ...

C#公众号开发之给用户发红包

红包功能简单介绍: 1、商户调用接口时,通过指定发送对象以及发送金额的方式发放红包,这样的方式,允许商户灵活的应用于各种... ...

C# 获取 PC 序列号的方法示例

在 C++ 需要使用 GetSystemFirmwareTable 的方法来获得 PC 的序列号,需要写的代码很多,但是... ...

C#程序异常关闭时的捕获

本文主要以一个简单的小例子,描述C# Winform程序异常关闭时,如何进行捕获,并记录日志。 概述 有时在界面的事件... ...

C#获取日期的星期名称实例代码

具体代码如下所示: private string GetWeekName(DayOfWeek week) {... ...

rfedfre

C# 7.0中解构功能详解

本文为大家分享了C# 7.0中的解构功能,供大家参考,具体内容如下 解构元组   C#7.0新增了诸多功能,其中有一项是... ...

C#实现ini文件读写操作

本文实例为大家分享了C#语言实现ini文件读写操作的具体代码,供大家参考,具体内容如下 1、ini文件是什么? 见百度百... ...

rfedfre

C# WPF 父控件通过使用可视化树找到子控件的示例代码

在我们使用WPF设计前台界面时,经常会重写数据模板,或者把控件放到数据模板里。但是一旦将控件放到数据模板中,在后台就没有... ...

C#实现金额转换成中文大写金额

本文实例为大家分享了C#金额转换成中文大写金额的具体代码,供大家参考,具体内容如下 /// <summary&... ...

rfedfre

C#利用VS中插件打包并发布winfrom程序

本文为大家分享了C#利用VS中插件打包并发布winfrom程序,供大家参考,具体内容如下 1.先在VS 的扩展更新中搜索... ...

微信小程序支付C#后端源码

本文实例为大家分享了微信小程序支付C#后端源码,供大家参考,具体内容如下 using System; using S... ...

rfedfre

C# 添加PDF页眉/页脚的示例代码

概述 页眉页脚是一篇完整、精致的文档的重要组成部分。在页眉页脚处,可以呈现的内容很多,如公司名称、页码、工作表名、日期... ...

C#怎么生成唯一订单号

本文实例为大家分享了C#生成唯一订单号的具体代码,供大家参考,具体内容如下 根据GUID+DateTime.Now.Ti... ...

C# 中用 Sqlparameter 的两种用法

新建一个表: create table abc ( id int IDENTITY(1,1) NOT NULL, n... ...