前言
本篇我们继续讲解本章其余的部分:构造函数、只读字段、匿名类型、结构详解、部分类、静态类、Object类、扩展方法,等。
01
类
构造函数
构造函数是一种特殊的方法:
- 与类同名
- 没有返回值,甚至不能写void(但可以写修饰符public,private,protected)
- 未指定构造函数时,编译器会自动创建默认构造函数;一旦手工创建了构造函数,编译器就不会再创建默认构造函数
- 构造函数可重载,同方法一样
- 静态构造函数:构造函数加static关键字;它只执行一次,一般用于初始化静态字段或者属性;静态构造函数执行时机并不确定,因此有顺序依赖的代码不要写在静态构造函数中(比如一个程序集中有10个静态构造函数,哪个先执行并无定数,我简单测试的结果是:在访问类的静态字段/属性时,静态构造函数会被执行),其中的代码逻辑尽量保持简单和单一;静态构造函数无访问修饰符、无参数,因为没有意义;静态构造函数只能访问静态成员,不能访问实例成员
- 构造函数的相互调用
当我们把构造函数看成方法后,构造方法间的互调用,我们可能会认为和方法一样,比如:
TestMain(){
}
TestMain(int arg){
this(); //构造方法互调用
int a = 10;
}
但实际上,这种方法只适用于Java(且构造方法的调用必须在首行)。在.Net中,使用的是叫做“构造函数初始化器”的语法:
Test()
: base() //调用基类的初始化器
{
}
Test(int arg)
:this(arg, 5) //初始化器
{
arg = 10;
}
Test(int arg1, int arg2)
:this() //初始化器
{
}
构造函数初始化器在构造函数函数体执行之前执行。this为类的多个构造函数的互调用,base为对基类的构造函数的调用。
只读字段
只读字段(readonly)是对常量的补充。
private readonly int Column5;
它的应用场景是:允许把一个字段设置为常量,但还需要执行一些计算,以确定它的初始值,这是常量无法做到的。
它的规则是:可以在构造函数中给只读字段赋值,但不能在其他地方赋值。
匿名类型
之前的章节,见过使用var关键字实现“类型推断”,var同时还应用于创建匿名类型。所谓匿名类型,是是一个继承自Object且没有名称的类。
var obj = new
{
Name = "daixf",
Age = 40
};
我们可以看到,在定义匿名类型时,需要指定属性名称和值。注意匿名类型的成员是属性而不是字段,可以想想为什么这么设计?我想,是因为遵循设计规范,属性一般是public的,而字段一般是private,两者各司其职。显然匿名类的成员需要被访问,所以应该是public,所以应该定义为属性。
但不是任何情况下,都需要指定创建的匿名类型的属性名的,如果数据来自于其他类的属性,匿名类型会默认使用相同属性名。
var obj1 = new
{
obj.Name,
obj.Age
};
因为匿名类型并没有特定的类型名,编译器为其指定了一个临时的类型名,因此在匿名类型上无法执行反射功能。
结构
上一篇已经讲过,结构和类非常像。结构的特点:
- 结构的定义和类类似,结构也是派生自Object,可定义字段、属性、函数
- 结构是值类型。因为是值类型,所以可以采用值类型的赋值方式而不采用new
- 限制:结构不能继承;结构默认了一个无参构造函数,且不允许显式的定义无参构造函数
- 和类中定义public会员采用属性不同,因为结构本来是小对象,因此多数程序员也倾向于不使用属性,而采用public字段作为成员。这似乎违反了C#通用编程规范(属性public,字段private),但它可以看作规范的例外情况。
部分类(partial)
部分类似乎离我们很远,我在践行“基于Repository框架”的时候,曾经必须用它,因为大量的类需要通过工具软件生成,而工具生成的类又有一部分功能需要手工书写,为了避免手工写的代码被工具软件覆盖,所以就采用了部分类的方式,一个类分为两个文件。但抛弃了Repository框架之后,我就再也没有主动用过部分类。
采用部分类的原因一般有以下几种:
- 避免代码生成工具生成的代码和手写代码的冲突,使用多个部分类代码文件
- 方便多个程序员协同开发,大家修改同一个类却可以锁定不同的文件
- 一个类的功能代码较多,使用部分类对功能进行分类,不同的功能组分布到不同的代码文件中
使用部分类的一个典型就是Winform窗体。一些初学者可能还不知道Form窗体使用了部分类。一个Form窗体就有两个cs代码文件,designer.cs就是由界面设计工具生成的:
使用部分类遵循的规则:
- 关键字的用法:把partial放在class,struct,interface等关键字的前面
- 多个部分类文件,必须定义为相同的namespace
- 多个部分类文件,必须使用相同的修饰符关键字,比如同为public,private
除了部分类,还有“部分方法”的概念,用途差不多,这里先不详述。
静态类(static)
是否定义类为静态类,似乎并不是那么清楚。因为一般我们声明一个静态字段/属性,或者实现一个静态方法,也不代表其所在的类必须是静态类。那么我们什么情况下认为类应该定义为静态类呢?我认为是否定义为静态类,更多的依据编程规范,而不是强制要求。
- 首先明确的一点是,如果类中定义了扩展方法,那么类必须定义为静态类,否则会编译错误
- 如果类的所有成员都是静态成员,那么此类应该声明为静态类(因为已经没有实例成员,不声明为静态类没有实际意义)
- 如果确定不会设计类的实例成员,那么应该声明类为静态类,因为这样编译器会进行强制检查,避免程序员不经意间定义不需要的实例成员。
Object类
所有.Net类都派生自Object类,包括结构(结构首先派生自类System.ValueType,而ValueType类又派生自Object类)。结构也派生自类,这也就好理解为什么struct的定义和class这么像了。
Object作为基类,它的一些成员是都是非常重要的,而且非常常用,简单介绍:
- ToString(),转字符串,常见的是基础类型转string
- GetHashTable(),用于散列表和字典
- Equals()和RefrenceEquals(),和比较运算符“==”一起,三者应用场景值得仔细比较
- Finalize(),类似于C++的析构函数。它和终结器,和Dispose()的关系和区别后面细讲
- GetType(),获取类类型,此功能往往和反射有关
- MemberwiseClone(),浅表复制。所谓浅表复制,是指复制类实例中,所有值类型的“值”,以及所有引用类型的“引用”。
扩展方法
扩展方法应用于:想给某个类增加方法,但却没有类的源码,或者没有对类的修改权限。此时可以使用扩展方法。扩展方法是静态类中的静态方法。对于扩展方法,第一个参数是要扩展的类型,它放在this关键字的后面。这告诉编译器,这个方法是扩展类型的一部分。在这个例子中,string是要扩展的类型。
public static bool IsNullOrEmpty(this string str)
调用:"abc".IsNullOrEmpty()
第三章到此为止就结束了,本章总体上讲了类的组成成员,以及其字段、属性、方法,和各种特性。下章将讲解《继承》。
觉得文章有意义的话,请动动手指,分享给朋友一起来共同学习进步。
欢迎关注本人微信公众号,更及时的关注最新文章(每周三篇原创文章,以及多篇专题文章):
附文:
C#部分类与部分方法