用静态构造函数初始化静态成员_

原创 2012年03月28日 19:32:50
编写程序的时候难免要使用到静态成员,由于静态成员的访问是脱离类型对象的,所以使用非静态构造函数,私有方法或者一些其他方法都是不合理的。.Net提供了成员初始化语句和静态构造函数来初始化静态成员。

  根据上一个单元,我们可以知道静态成员的初始化语句会早于静态构造函数执行,其次还知道成员初始化语句的好处和一些限制。这里要提的一点就是,静态构造函数和实例构造函数之间的区别,因为静态构造函数是由CLR调用执行的,所以静态构造函数只能是一个,同时不能还有参数。那么静态构造函数相对于成员初始化的一些弊端都不复存在。

  此外,对于成员初始化最大的问题,就是异常无法捕获,可能对于普通成员来说,还可以在构造类型对象的时候加上try-catch;而对于静态成员来说,有时无法确定哪一个类型访问语句会先执行,如果对每个使用类型的地方都加上try-catch,会降低程序可读性。但如果把这种异常散发出去,会导致整个程序崩溃。那么使用静态构造函数就能比较好的捕获异常,并进行异常处理。

  对于静态成员初始化语句和静态构造函数一个对比大致如下。

  简便性 安全性 异常处理
静态成员初始化 最简便 比较安全 非常难
静态构造函数 比较简便 最安全 容易

  那么到底用什么方法来初始化静态成员呢,给出以下两点建议:

  第一就是简单静态成员,例如类型为值类型等,使用成员初始化语句来完成;

  第二静态成员初始化比较复杂,或者有可能出现异常,那么用静态构造函数来完成。

  本单元的内容基本上讲完了,现在说说静态成员最主要的一个应用“Singleton”,就是保证此类型的对象唯一性。其实实现很简单,如下就举一个简单的例子。

public class MySingleton
{
 private static readonly MySingleton _theOneAndOnly;
 static MySingleton()
 {
  _theOneAndOnly = new MySingleton();
 }
 public static MySingleton TheOnly
 {
  get{ return _theOneAndOnly; }
 }

 /// <summary>
 /// Private constructor to avoid object created from outside
 /// </summary>

 private MySingleton()
 {}
}

  对于静态成员何时被初始化,其实仔细看我上一个单元所说的对象构造步骤就可以分析出来。可能看起来比较抽象,尤其在分析嵌套关系的时候,现在就举一个例子来进行说明。

class A
{
 public static int X = B.Y;
 static A()
 {
  ++X;
 }
}

class B
{
 public static int Y = A.X;
 static B()
 {
  ++Y;
 }
}

  那么大家先猜猜如下调用输出结果是什么。

Debug.WriteLine( A.X.ToString() );
Debug.WriteLine( B.Y.ToString() );

  其结果是“2,1”,也就是A.X的值为2,而B.Y的值为1。

  分析此类问题,只要记住三点就行了。
  
  第一代码的执行顺序,代码在前的先执行;

  第二静态成员初始化语句要先于静态构造函数执行;

  第三静态成员初始化语句与静态构造函数只执行一次。

  如果了解这三点,接下来就分析为什么会出现上面的结果。

  当调用到第一条语句的时候,

Debug.WriteLine( A.X.ToString() );


  首先是访问A这个类型,那么要对A这个类型的静态成员进行初始化,其次如果有静态构造函数,需要调用它。

  对于A的静态成员只有“X”,按照上一单元的过程,先给其分配空间,并辅以0来初始化,其次调用其对应的成员初始化语句来初始化这个静态成员。

  那么它的成员初始化语句是“X = B.Y”,那么需要访问“B.Y”来初始化X这个静态成员。

  对于“B.Y”的访问,就是访问B类型,也是和访问A一样,首先对这个类型的静态成员进行初始化,其次如果有静态构造函数,需要调用它。而B的静态成员只有“Y”,先给其分配空间,并辅以0来初始化,其次调用其对应的成员初始化语句来初始化这个静态成员。

  那么对于“Y = A.X”成员初始化语句,由于此时不是第一次访问类型A,所以不再进行静态成员初始化和静态构造函数的调用,对于“A.X”的访问是直接访问。此时“A.X”的值为0,那么Y的值也为0;接着执行B的静态构造函数,这样处理后Y的值为1。

  那么对于A中的成员初始化语句“X = B.Y”,到此就执行完了,此时A类型中的X与B类型中的Y都是一样的,值为1。不过B中的静态成员初始化语句和静态构造函数都执行过了,而A中的静态构造函数还未执行。因此经过A的静态构造函数处理,A的X值为2,这也就是最后显示的结果。

  分析过程看起来很绕,其实只要把握我前面所说的三个原则,那么在复杂的问题也一样分析

 

 

 

静态构造函数自动被调用,不能被显式调用。虽然提供了许多约束条件,但是静态构造函数执行的确切时间和顺序是不确定的:
一个类的静态构造函数在这个类的任何实例被创建前执行。
一个类的静态构造函数在类的任何静态成员被引用前执行。
一个类的静态构造函数在它的所有派生类的静态构造函数执行之后执行。
一个类的静态构造函数从不会被执行一次以上。

一个类的静态构造函数在它的所有派生类的静态构造函数执行之后执行。


这个是指父类有一个静态构造函数,子类也有一个静态构造函数,比如下面的代码

using System; 

namespace StaticConstruct 


public class ObjectChild:ObjectParent 

public ObjectChild() 

Console.WriteLine(
"子类构造函数!"); 
}
 

static ObjectChild() 

Console.WriteLine(
"子类静态构造函数!"); 
}
 
}
 


public class ObjectParent 

public ObjectParent() 

Console.WriteLine(
"父类构造函数!"); 
}
 

static ObjectParent() 

Console.WriteLine(
"父类静态构造函数!"); 
}
 
}
 

class Class1 

[STAThread] 
static void Main(string[] args) 

ObjectChild obj 
= new ObjectChild(); 
Console.ReadLine(); 
}
 
}
 
}
 


执行的结果是:

子类静态构造函数!
父类静态构造函数!
父类构造函数!
子类构造函数!

WPF 构造函数InitializeComponent()和Loaded()

写在Xaml中的控件初始化,发生在InitializeComponent中,所有控件都加载完毕后,会调用Loaded(). 不要滥用Loaded()! Loaded 通常是元素初始化序列中...
  • wudidagou
  • wudidagou
  • 2014年03月12日 15:41
  • 2222

C# 静态构造函数的说明

静态构造函数是C#的一个新特性,其实好像很少用到。不过当我们想初始化一些静态变量的时候就需要用到它了。这个构造函数是属于类的,而不是属于哪里实例的,就是说这个构造函数只会被执行一次。也就是在创建第一个...
  • tieshuxianrezhang
  • tieshuxianrezhang
  • 2017年02月09日 10:58
  • 67

C# 静态代码块(静态构造函数)

本来以为是和java一样写法。结果找了半天也没找到。 原来有 静态构造函数 一样的功能。 class SimpleClass { // Static constructor ...
  • gaotong2055
  • gaotong2055
  • 2013年02月22日 11:17
  • 4134

用静态构造函数初始化静态成员

class A {  public static int X = B.Y;  static A()  {   ++X;  } } class B {  public static ...
  • iamdll
  • iamdll
  • 2011年03月21日 14:11
  • 1049

C# 实例构造函数和静态构造函数

静态构造函数:   #1:静态构造函数主要用于初始化一些静态的变量。静态构造函数只会执行一次,而且是在创建此类的第一个实例 或 引用任何静态成员(包括静态方法)之前,由.NET自动调用。   #2:静...
  • u013644191
  • u013644191
  • 2014年02月13日 15:23
  • 986

【C#】 构造函数

聊聊构造函数
  • Lionel_Coder
  • Lionel_Coder
  • 2016年07月30日 15:56
  • 195

C# 委托 && Virtual, abstract, override && 静态构造函数

1.委托 一般来说我们都是针对数据来做操作,但是有时候需要针对方法来操作,这个时候就用到了委托。 委托将方法当作参数,当委托的返回值是void时,系统会先假定这是个多播委托。所谓多播委托,即多个方...
  • Sephoenix
  • Sephoenix
  • 2012年07月12日 17:19
  • 768

使用静态构造函数初始化静态成员

编写程序的时候难免要使用到静态成员,由于静态成员的访问是脱离类型对象的,所以使用非静态构造函数,私有方法或者一些其他方法都是不合理的。.Net提供了成员初始化语句和静态构造函数来初始化静态成员。 ...
  • pyq228
  • pyq228
  • 2011年10月20日 19:51
  • 402

C#箴言:用静态构造函数初始化静态成员

编写程序的时候难免要使用到静态成员,由于静态成员的访问是脱离类型对象的,所以使用非静态构造函数,私有方法或者一些其他方法都是不合理的。.Net提供了成员初始化语句和静态构造函数来初始化静态成员。   ...
  • yazoli2002
  • yazoli2002
  • 2007年07月26日 14:58
  • 280

用静态构造函数初始化静态成员_

编写程序的时候难免要使用到静态成员,由于静态成员的访问是脱离类型对象的,所以使用非静态构造函数,私有方法或者一些其他方法都是不合理的。.Net提供了成员初始化语句和静态构造函数来初始化静态成员。 ...
  • yangchun1213
  • yangchun1213
  • 2012年03月28日 19:32
  • 546
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:用静态构造函数初始化静态成员_
举报原因:
原因补充:

(最多只允许输入30个字)