本文借鉴《编写高质量的C#代码:改善C#程序的157个建议》,算是对自己学习的总结,也希望分享下所学知识~~
什么是成员?
指字段和方法。
1.不要为抽象类提供 public 的构造方法。
抽象类如果没有指定构造方法,编辑器会生成一个 protected 的构造方法。
抽象类不设计的本意是能让子类继承,不是用于生成对象实例的,所以这是没必要的。
2.多使用属性
属性和字段的本质区别:属性是方法。
针对一个属性
public string Name
{
get;set;
}
编辑器会生成一个字段和两个方法。
使用属性可以添加代码控制,比如触发事件。
当然,如果某个属性仅仅对内部可见,也可以考虑为字段。
3.应该初始化主要属性和字段
如果字段没有在初始化器中设置初始值,那么它就应该在构造方法中设置初始值。
初始化器如下:
A a = new A()
{
Name = "nameA"
};
也属于编译器的语法糖,编译后在构造方法的最开始处执行。
4.不要在构造方法中调用虚函数
public class Person
{
public Person()
{
Test();
}
public virtual void Test()
{
}
}
public class Chinese : Person
{
List<int> list;
public Chinese() : base()
{
list = new List<int>();
}
public override void Test()
{
Console.WriteLine(list.Count);
}
}
//for test
Chinese chinese = new Chinese();
这里会报错,因为首先调用基类的构造函数,调用了虚方法,转到子类的虚方法,但是还没有调用到子类的构造函数进行初始化,所以会报出空指针的错误。
5.重写时不应使用子类参数
public class Employee { }
public class Manager : Employee { }
public class Salary
{
public void SetSalary(Employee e)
{
Console.WriteLine("职员被设置了薪水");
}
}
public class ManagerSalary : Salary
{
public void SetSalary(Manager e)
{
Console.WriteLine("经理被设置了薪水");
}
}
//for test
ManagerSalary m = new ManagerSalary();
m.SetSalary(new Employee()); // 职员被设置了薪水
本意是设置经理的薪水,但是实例调用代码却设置了员工的薪水。
这是因为子类中的方法已经被当作一个独立于基类的方法了。
重写时使用子类参数有一定风险,应该仍然使用 Employee 并且使用 new 关键字。
6.静态方法和实例方法的区别
a).某个类型 Person,就是一个类型对象,类型对象会加载到 加载堆
上。
类型对象包括:自身的指针、自身的同步索引块、静态字段,静态方法记录项和实例方法记录项。
b).其 new 出来的对象,叫做实例对象,实例对象会加载到 垃圾回收堆
上 。
实例对象包括:类型对象指针,同步索引块,实例字段。
本质上是没有区别的,使用上的区别是使用实例方法需要先有实例对象。
如果一个方法只跟类型本身有关系,就应该被设计成静态方法,反之亦然。
7.使用扩展方法
a.扩展 sealed 密封类型。
b.扩展第三方程序集中的类型。
c.避免不必要的继承体系。
使用规则:
a.必须在静态类中,而不能是一个嵌套类。
b.扩展方法必须是静态的。
c.扩展方法的第一个参数必须是要扩展的类型,加上 this 关键字。
d.不支持扩展属性、事件。
public class Salary
{
public string Num
{
get; set;
}
public void SetSalary(Employee e)
{
Console.WriteLine("职员被设置了薪水");
}
}
public static class SalaryExtension //不能是嵌套类
{
public static string GetSalary(this Salary salary)
{
return salary.Num;
}
}