c#中字段和属性的区别,委托和事件的区别

IDE眼里的字段和属性

csharp 复制代码
class Test
{
	public int age1 = 12;
	public int Age2 { get; set; } = 18;

	public void Show()
	{
		Console.WriteLine(age1++);
		Console.WriteLine(Age2++);
	} 
}

很多新人发现在类中定义变量时,有些人会在后面写上get,set

这种写法定义出来的变量,在使用的时候看起来和普通的变量没有区别。

所以不理解这样做有什么意义。

首先,我们去掉Age2。只关注age1.

用你的IDE对他点右键,快速重构中会出现两个选项,封装字段并使用字段/并使用属性。

csharp 复制代码
class Test
{
	private int age1 = 12;

	public int Age1 { get => age1; set => age1 = value; }

	public void Show()
	{
		Console.WriteLine(age1++);//如果选择使用属性,那么此句会使用Age1
	}
}

无论选择哪一种,类中都会出现一个Age1,并且他的内容是相同的。

然后,根据你选择的使用属性 / 仍使用字段 ,下面对这个变量的调用会替换为age1 / Age1

得出结论,没有get,set的东西是字段,带有get,set的东西叫做属性。

然后,对Age1点右键,快速重构为自动属性.

csharp 复制代码
class Test
{
	public int Age1 { get; set; } = 12;

	public void Show()
	{
		Console.WriteLine(Age1++);
		//如果选择使用属性,那么此句会使用Age1
	}
}

这说明,{ get; set; }写法的叫自动属性,是一种简略的写法。

他的完整写法应该包含一个字段,并在get,set中添加逻辑内容去控制一个字段。

匿名字段

在使用自动属性时,编译器会自动创建一个你无法访问的字段,这称为匿名字段。

由于无法访问到匿名字段,所以无法为他决定初始值。因此,自动属性改为直接在属性上赋值初始值。
public int Age1 { get; set; } = 12;

但是,对于完整属性,public int Age1 { get => age1; set => age1 = value; }是不能这样写的。

事件

事件和属性类似,同样是一种缩略写法,同样引入了匿名字段。

csharp 复制代码
class Test
{
	public Action action = () => { };
	public event Action Action = () => { };

	public void Show()
	{
		action();
		Action();
	}
}

事件和委托的使用方式一样,所以看不出来区别。但如果将事件写全,那么事件就不能赋值或调用。

csharp 复制代码
class Test
{
	public Action action = () => { };
	public event Action Action { add => action += value; remove => action -= value; } //= () => { };

	public void Show()
	{
		action(); 
		//Action();
	}
}

继承中的字段和属性

程序里的东西可以分类为储存的值,或可以执行的指令。

在类中,属性和事件是可以设置为虚拟,抽象,重写的。这说明属性和事件更与方法类似。

在接口中,不能存在字段,但是可以存在属性和事件。这说明属性和事件有别于字段。

csharp 复制代码
interface ITest
{
	public event Action Action;
	public int Age { get; set; }
}
abstract class BTest : ITest
{
	public abstract int Age { get; set; }

	public abstract event Action Action;
}
class DTest : BTest
{
	public override int Age { get { return default; } set { } }

	public override event Action Action { add { } remove { } }
}

并且,在这个抽象类中,不能为属性或事件赋值初始值,也不能像使用委托一样调用这个事件。

这再次说明了,属性和事件不储存值。我们平常对他们的调用都是转为调用一个匿名字段的。

反射眼里的字段和属性

声明两个类,其中一个属性和事件写上空的逻辑。

csharp 复制代码
class Test1
{
	public int Age { get; set; }
	public event Action Action;
}
class Test2
{
	public int Age { get => default; set { } }
	public event Action Action { add { } remove { } }
}

使用反射查看这两个类里面所有的东西。

csharp 复制代码
Type t1=typeof(Test1);
Type t2=typeof(Test2);

foreach (var item in t1.GetMembers((BindingFlags)(-1)))
{
    Console.WriteLine(item.Name);
}
Console.WriteLine("=========");
foreach (var item in t2.GetMembers((BindingFlags)(-1)))
{
	Console.WriteLine(item.Name);
}
get_Age
set_Age
add_Action
remove_Action
.ctor
Age
Action
<Age>k__BackingField
Action
=========
get_Age
set_Age
add_Action
remove_Action
.ctor
Age
Action

在这里面Age是属性,get_Age和set_Age分别是属性里面的get访问器和set访问器。

同样的,Action,add_Action,remove_Action这三个东西都是事件生成的东西。

然后.ctor是构造器 / 构造方法 / 构造函数。

最后,上面的类型多出来了一个<Age>k__BackingField和一个Action。

这两个东西就是匿名字段。验证代码如下。

csharp 复制代码
Type t1 = typeof(Test1);
FieldInfo field1 = t1.GetField("<Age>k__BackingField", (BindingFlags)(-1));

Test1 test = new Test1();
Console.WriteLine(test.Age);

field1.SetValue(test, 666);
Console.WriteLine(test.Age);

最后,在类型上。反射出来的属性和事件,可以获取他们的访问器。此方法在名字上,和返回值的类型上,

都认为访问器是一种方法。也就是说属性和事件是包含方法的东西。

csharp 复制代码
Type type = null;//此代码仅展示类型,运行会有异常。
MethodInfo method = type.GetMethod("");//获取方法

PropertyInfo property1 = type.GetProperty("");//获取属性
MethodInfo get = property1.GetGetMethod();//获取get访问器,以方法形式
MethodInfo set = property1.GetSetMethod();//获取set访问器,以方法形式

EventInfo eventInfo = type.GetEvent("");
MethodInfo add = eventInfo.GetAddMethod();
MethodInfo remove = eventInfo.GetRemoveMethod();

并且在属性实例和事件实例这两个类型里,也确实没有任何能获取包含的值的方法。

所以,属性和事件的本质工作并不包含储存一个值。

相关推荐
向宇it1 小时前
【从零开始入门unity游戏开发之——C#篇25】C#面向对象动态多态——virtual、override 和 base 关键字、抽象类和抽象方法
java·开发语言·unity·c#·游戏引擎
向宇it3 小时前
【从零开始入门unity游戏开发之——C#篇24】C#面向对象继承——万物之父(object)、装箱和拆箱、sealed 密封类
java·开发语言·unity·c#·游戏引擎
坐井观老天7 小时前
在C#中使用资源保存图像和文本和其他数据并在运行时加载
开发语言·c#
pchmi10 小时前
C# OpenCV机器视觉:模板匹配
opencv·c#·机器视觉
黄油饼卷咖喱鸡就味增汤拌孜然羊肉炒饭11 小时前
C#都可以找哪些工作?
开发语言·c#
boligongzhu13 小时前
Dalsa线阵CCD相机使用开发手册
c#
向宇it1 天前
【从零开始入门unity游戏开发之——C#篇23】C#面向对象继承——`as`类型转化和`is`类型检查、向上转型和向下转型、里氏替换原则(LSP)
java·开发语言·unity·c#·游戏引擎·里氏替换原则
sukalot1 天前
windows C#-命名实参和可选实参(下)
windows·c#
小码编匠1 天前
.NET 下 RabbitMQ 队列、死信队列、延时队列及小应用
后端·c#·.net