Effective C# 使用成员初始化器而不是赋值语句

一般情况下,一个类都会有多个构造函数。随着时间的推移,成员变量、构造函数不断增加。为了处理这种情况最方便的办法就是:在声明变量的时候进行初始化,而不是在每个构造函数中进行。无论是类成员(静态变量)合适实例变量,我们都应该充分利用初始化器的语法。

C#编程在,一般在声明一个变量的同时我们会对其进行初始化:


代码如下:

1 class Employee
2 {
3 private List<Employee> empList = new List<Employee>();
4 }

不论我们为Employee类添加了多少个构造函数,empList变量都能够被正确的初始化,这是因为:

编译器会在所有的构造函数(包括默认构造函数)的最开始位置生成代码来为所以的实例成员变量定义初始化器(进行初始化);所以我们不需要再构造函数中为每一个定义的成员变量添加初始化代码——直接在声明的时候初始化即可。

初始化器可以看做是构造函数中初始化语句的另一种表示。初始化器生成的代码会插入到构造函数代码前面执行。初始化器将在为类型执行调用基类构造函数之前执行,其顺序与类成员变量声明的顺序一样。

  C#的初始化器语法是一种最简单的、能够避免类型中存在未初始化变量的解决办法。但是在下面的三种情况下应该避免使用初始化器:

1.初始化对象为0或null时
  因为系统默认的初始化会将会在所有代码执行前把一切都设置成0或者null(值类型和引用类型)。而且这一步的操作是位于很底层的实现,我们也可以直接将对象赋值设置为0或null,但是显然这是多余的。

2.对同一个变量执行不同的初始化方式
  使用初始化语句的一个前提是:所有得构造函数都将为该变量设置同样的值。我们看下面的示例代码:


代码如下:

class Employee
{
//声明变量的同时进行初始化
private List<Employee> empList = new List<Employee>();

public Employee()
{
}

public Employee(int size)
{
empList = new List<Employee>(size);
}
}

在上面的代码中,当我们调用第二个构造函数创建初始化指定大小的泛型集合时 ,实际上时创建了两个List<Employee>。第一个创建后马上变成了垃圾——这是由于初始化器将在所有构造函数之前执行。编译器生成的代码类似于下面的这段代码:


代码如下:

class Employee
{
//声明变量
private List<Employee> empList;

public Employee()
{
empList = new List<Employee>();
}

public Employee(int size)
{
empList = new List<Employee>();
empList = new List<Employee>(size);
}
}

我们可以看到这样做会影响程序的效率,创建了不必要的对象,所以如果需要在不同的构造函数中执行不同的初始化方式时正确的做法应该是不适用初始化器,而是先声明变量,然后在构造函数中进行成员变量的初始化,如下:


代码如下:

class Employee
{
//声明变量
private List<Employee> empList;

public Employee()
{
empList = new List<Employee>();
}

public Employee(int size)
{
empList = new List<Employee>(size);
}
}

3.需要进行异常处理
  初始化器无法被try语句包裹。所以对象初始化器执行的过程中发生异常都会被传递到对象之外。如果在初始化对象的时候可能会抛出异常时我们应该将这部分代码放到构造函数中,对其进行异常处理。这样才能够实现必要的恢复性代码,以创建类型实例并以更友好的方式处理异常。

小节:
  成员初始化器是保证类型中成员变量都被初始化的最简单方法——在声明变量时就对其进行初始化,无论调用的是那个构造函数,初始化器都将会在所有构造函数之前执行。这种语法也避免了在添加新的构造函数时遗漏掉重要的初始化代码。所以,如果对于所有的构造函数某个成员变量的初始化值是一样的,那么就应该尽量使用初始化器语法。

(0)

相关推荐

  • C#两个相同属性的类赋值方法

    最近有遇到两个类之间的赋值问题,两个类的属性几乎都一样的,所以写了个通过反射获取属性的然后赋值的方法,把一个类的属性的值赋值给另一个类. 框架是.net 4.5 public static D Mapper<D, S>(S s) { D d = Activator.CreateInstance<D>(); try { var sType = s.GetType(); var dType = typeof(D); foreach (PropertyInfo sP in sType.G

  • Question:基于C#连续赋值的面试题(解答)

    题目在这里:Question:基于C#连续赋值的面试题介绍在msdn中,对=号操作符的说明如下: 赋值运算符 ( =) 将右操作数的值存储在左操作数表示的存储位置.属性或索引器中,并将值作为结果返回.操作数的类型必须相同(即右操作数必须可以隐式转换为左操作数的类型).首先来看int x,y,z;X = y = z = 1;.csharpcode, .csharpcode pre{font-size: small;color: black;font-family: consolas, "Couri

  • Question:基于C#连续赋值的面试题介绍

    我们知道C#中 a=b=c;是合法的,例如: int i,j,k;i=j=k=1; 下面有两道题: 第一题: 复制代码 代码如下: const int x=1;    short y;    object z;    z=y=x; //请问下面的输出是什么?    Console.WriteLine(z.GetType().ToString()); 第二题: 复制代码 代码如下: class C {     private string x;     public string X     { 

  • c#3.0实现延迟赋值示例

    延迟赋值主要有两点: 1.一个参数可能或可能没被赋值. 2.一个参数在一个函数中每次使用时可能被赋值. 如下面的这种情况: 复制代码 代码如下: int Add(int x, int y){    return (2 + 1) + (1);} 使用Func<T>,我们轻松实现,看代码: 复制代码 代码如下: /// <summary>/// LazyExpression/// </summary>/// <typeparam name="T"&

  • C#实现利用反射简化给类字段赋值的方法

    本文实例讲述了C#实现利用反射简化给类字段赋值的方法.分享给大家供大家参考.具体分析如下: 说明:这个例子主要的思路是建立一个类和数据库查询语句的字段结构是一致的 然后利用反射,直接用数据字段名称进行拼凑,给类对象的字段进行赋值   1.类的定义 namespace CCB_Donet.ClassFolder { public class FieldRuleInfo { public string gStrFNo; public string gStrFName; public string g

  • Effective C# 使用成员初始化器而不是赋值语句

    一般情况下,一个类都会有多个构造函数.随着时间的推移,成员变量.构造函数不断增加.为了处理这种情况最方便的办法就是:在声明变量的时候进行初始化,而不是在每个构造函数中进行.无论是类成员(静态变量)合适实例变量,我们都应该充分利用初始化器的语法. C#编程在,一般在声明一个变量的同时我们会对其进行初始化: 复制代码 代码如下: 1 class Employee 2 { 3 private List<Employee> empList = new List<Employee>(); 4

  • 成员初始化列表与构造函数体中的区别详细解析

    论坛中回答一个别人问题 C++ Primer中在讲构造函数初始化列表的时候有这么一段话:无论是在构造函数初始化列表中初始化成员,还是在构造函数体中对它们赋值,最终结果是相同的.不同之处在于,使用构造函数初始化列表的版本初始化数据成员,没有定义初始化列表的构造函数版本在构造函数体中对数据成员赋值. 请问这里的初始化数据成员与对数据成员赋值的含义是什么?有什么区别? 我知道在数据成员有默认构造函数时是有不同的,但对其他类型的成员呢?其他类型成员的初始化和赋值有区别吗?================

  • 关于C++类的成员初始化列表的相关问题

    在以下四中情况下,要想让程序顺利编译,必须使用成员初始化列表(member initialization list): 1,初始化一个引用成员(reference member): 2,初始化一个常量对象(const member); 3,调用一个基类的构造函数,且该基类的构造函数有一组参数: 4,调用一个成员类(member class)的构造函数,且该构造函数有一组参数 这四种情况程序可以正常编译,但是效率有所欠缺(下面会具体说到). class Word{ String _name; in

  • C#特性-对象集合初始化器介绍

    C# 3.0为你提供了对象集合初始化器: /// <summary> /// 图书类 /// </summary> public class Book { /// <summary> /// 图书名称 /// </summary> public string Title { get; set; } /// <summary> /// 单价 /// </summary> public float Price { get; set; }

  • 简单了解Java类成员初始化顺序

    这篇文章主要介绍了简单了解Java类成员初始化顺序,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 类中包含7中成员: 1.静态变量 static 2.final静态常量 final static 3.静态代码块 static{} //多个代码块顺序执行 4.普通变量 5.普通代码块 {} //多个代码块顺序执行 6.构造方法 7.普通方法 类中成员初始化方式(排出7): 非继承类中: 首先执行1/2/3,执行顺序为顺序执行(谁在前先执行谁).

  • c# 类成员初始化顺序的特殊情况

    这里直接给出C#类成员一般初始化顺序: 子类静态字段 子类静态构造 子类实例字段 父类静态字段 父类静态构造 父类实例字段 父类实例构造 子类实例构造 为什么说是"一般"初始化顺序呢?因为根据类结构的不同,类成员的初始化顺序并不是一成不变的.但是这个顺序是基础,可以推导出其他特殊的初始化顺序.下面我们就来看两种特殊的情况: static void Main(string[] args) { Console.WriteLine("---------------一般初始化顺序--

  • c# 9.0新特性——模块初始化器

    作者:MarkKang 出处:https://www.cnblogs.com/markkang/ 1 背景动机 关于模块或者程序集初始化工作一直是C#的一个痛点,微软内部外部都有大量的报告反应很多客户一直被这个问题困扰,这还不算没有统计上的客户.那么解决这个问题,还有基于什么样的考虑呢? 在库加载的时候,能以最小的开销.无需用户显式调用任何接口,使客户做一些期望的和一次性的初始化. 当前静态构造函数方法的一个最大的问题是运行时会对带有静态构造函数的类型做一些额外的检查.这是因为要决定静态构造函数

  • C++类成员初始化的三种方式

    目录 一.初始化方式 1.初始化方式一:初始化列表 2.初始化方式二:构造函数初始化 3.初始化方式三:声明时初始化(也称就地初始化,c++11后支持) 二.声明时初始化->初始化列表->构造函数初始化 1.声明时初始化的使用场景 2.列表初始化的使用场景 3.构造函数初始化的使用场景 前言: 在C++98中,支持了在类声明中使用等号"="加初始值的方式,来初始化类中静态成员常量.这种声明方式我们也称之为"就地"声明.就地声明在代码编写时非常便利,不过C

  • C++成员初始化列表

    文章转自: 公众号:Coder梁(ID:Coder_LT) 除了可以使用构造函数Classy::Classy(int n, int m): mem1(n), mem2(0), mem3(n*m+2) {    ...};类成员进行初始化之外,C++还提供了另外一种初始化的方法,叫做成员初始化列表. 我们假设Classy是一个类,而mem1,mem2和mem3都是这个类的数据成员,那么类构造函数可以写成: Classy::Classy(int n, int m): mem1(n), mem2(0)

  • SpringBoot深入浅出分析初始化器

    如有错误,望指正: SpringBoot可以有三种方式定义初始化器,来为容器中增加自定义的对象,具体如下: 1.定义在spring.factories文件中,被SpringFactoriesLoader发现注册: 在resources下建立META-INF文件夹,新建spring.factories文件,添加自定义的初始化器:org.springframework.context.ApplicationContextInitializer=com.mooc.sb2.initializer.Ini

随机推荐