6 CLR静态构造器
CLR保证一个类型构造器在每个AppDomain中只执行一次,而且这种执行是线程安全的。


作用:
就是初始化静态成员
比如有几个静态成员需要初始化
那你把初始化代码放到哪呢?
放到普通构造函数里,那肯定不行。因为静态成员没有创建实例就要可用。
专门建一个static public方法来初始化?这样用起来非常不方便,你需要在“第一次”使用静态成员前先调用这个方法。
如果你在使用静态成员前忘了调用该方法,会导致错误。
如果重复调用,又是冗繁操作。
所以静态构造函数就派上用场了。
它会在你第一次调用静态成员(或创建实例)的时候自动被调用
解析:
静态构造器函数是实现对一个类进行初始化的方法成员,它一般用于对静态数据的初始化。静态构造函数不能有参数,不能有修饰符而且不能被调用,当类被加载时,类的静态构造器自动被调用。
在一个程序的执行过程中,静态构造器最多只执行一次。
静态构造器在类的静态成员初始化之后执行,或者讲编译器会将静态成员初始化语句转化成赋值语句放在静态构造器执行的最开始。
静态构造器在任何类的静态成员被引用之前执行。
静态构造器在任何类的实例变量被分配之前执行。
静态构造器最主要的作用是对类的静态成员进行初始化。
namespace 静态构造器
{
class A
{
public static int i = ;
static A() //静态构造器能不能带参数,不能有修饰符,默认private
{
Console.WriteLine("我是类A的静态构造器");
} public A()
{
Console.WriteLine("我是类A的无参构造器");
}
} //编译之后等价于
//class A
//{
// public static int i ;
// static A() //静态构造器能不能带参数,不能有修饰符,默认private
// {
// i = 100;
// Console.WriteLine("我是类A的静态构造器");
// } // public A()
// {
// Console.WriteLine("我是类A的无参构造器");
// }
//} class Program
{
static void Main(string[] args)
{
Console.WriteLine(A.i);//当类被加载时,类的静态构造函数自动被调用,静态构造器是在a被使用前加载的 // A a = new A(); //静态构造器最多只执行一次 Console.ReadKey();
}
}
}
最近看书,看到静态构造函数的时候,发现其不能从程序中显示调用,而通常静态构造函数初始化类的静态字段,这样的话,为何不直接在静态字段定义的时候直接为其赋值呢?
直接赋值的结果就是它们会被放到静态构造函数中执行.
和一般的构造函数类似的呀,赋值只是构造函数的一个功能。有时候,你需要在静态构造函数中写一些逻辑,或是执比一些代码,比如工厂模式中很多的,要读取配置文件,然后把根据配置的文件初使化一些常用的对象,比如单例模式中的唯一对象,或者spring中的session什么的。反正一句话是,有些逻辑也要放在构造函数中,静态的也是。
先看代码:
1.包含静态字段的调用
- class Program
- {
- static void Main(string[] args)
- {
- Console.WriteLine("EntryPoint");
- Console.WriteLine(StaticClass.staticMem);
- StaticClass.ShowStaticStr("Call Static Method");
- Console.Read();
- }
- class StaticClass
- {
- public static string staticMem = ShowStaticStr ("Static Member");
- public static string ShowStaticStr(string message)
- {
- Console.WriteLine(message);
- return message;
- }
- }
- }
class Program
{
static void Main(string[] args)
{
Console.WriteLine("EntryPoint");
Console.WriteLine(StaticClass.staticMem);
StaticClass.ShowStaticStr("Call Static Method");
Console.Read();
} class StaticClass
{
public static string staticMem = ShowStaticStr ("Static Member");
public static string ShowStaticStr(string message)
{
Console.WriteLine(message);
return message;
}
}
}
结果如下:


添加静态构造函数:
- class Program
- {
- static void Main(string[] args)
- {
- Console.WriteLine("EntryPoint");
- Console.WriteLine(StaticClass.staticMem);
- StaticClass.ShowStaticStr("Call Static Method");
- Console.Read();
- }
- class StaticClass
- {
- public static string staticMem = ShowStaticStr("Static Member");
- public static string ShowStaticStr(string message)
- {
- Console.WriteLine(message);
- return message;
- }
- static StaticClass()
- { }
- }
class Program
{
static void Main(string[] args)
{
Console.WriteLine("EntryPoint");
Console.WriteLine(StaticClass.staticMem);
StaticClass.ShowStaticStr("Call Static Method");
Console.Read();
} class StaticClass
{
public static string staticMem = ShowStaticStr("Static Member");
public static string ShowStaticStr(string message)
{
Console.WriteLine(message);
return message;
} static StaticClass()
{ }
}
结果如下:

2.去掉静态字段调用
- static void Main(string[] args)
- {
- Console.WriteLine("EntryPoint");
- // Console.WriteLine(StaticClass.staticMem);
- StaticClass.ShowStaticStr("Call Static Method");
- Console.Read();
- }
- class StaticClass
- {
- public static string staticMem = ShowStaticStr ("Static Member");
- public static string ShowStaticStr(string message)
- {
- Console.WriteLine(message);
- return message;
- }
- }
static void Main(string[] args)
{
Console.WriteLine("EntryPoint");
// Console.WriteLine(StaticClass.staticMem);
StaticClass.ShowStaticStr("Call Static Method");
Console.Read();
} class StaticClass
{
public static string staticMem = ShowStaticStr ("Static Member");
public static string ShowStaticStr(string message)
{
Console.WriteLine(message);
return message;
}
}
结果如下:

添加静态构造函数:
- class Program
- {
- static void Main(string[] args)
- {
- Console.WriteLine("EntryPoint");
- //Console.WriteLine(StaticClass.staticMem);
- StaticClass.ShowStaticStr("Call Static Method");
- Console.Read();
- }
- class StaticClass
- {
- public static string staticMem = ShowStaticStr("Static Member");
- public static string ShowStaticStr(string message)
- {
- Console.WriteLine(message);
- return message;
- }
- static StaticClass()
- { }
- }
class Program
{
static void Main(string[] args)
{
Console.WriteLine("EntryPoint");
//Console.WriteLine(StaticClass.staticMem);
StaticClass.ShowStaticStr("Call Static Method");
Console.Read();
} class StaticClass
{
public static string staticMem = ShowStaticStr("Static Member");
public static string ShowStaticStr(string message)
{
Console.WriteLine(message);
return message;
} static StaticClass()
{ }
}
结果如下:

显式定义静态构造函数时的IL:

未显式定义静态构造函数的IL:

未显式定义静态构造函数时,IL中多了一个beforefieldinit标志,此标志使得运行库能够在任何时候执行类型构造函数方法,只要该方法(静态构造函数)在第一次访问该类型的静态字段(而不是方法)之前执行即可。换句话说,beforefieldinit 为运行库提供了一个执行主动优化的许可。如果没有 beforefieldinit,运行库就必须在某个精确时间(调用之前)运行类型构造函数,即,恰好在第一次访问该类型的静态或实例字段和方法之前。当存在显式类型构造函数时,编译器不会用 beforefieldinit 标记该类型,精确的计时限制会导致 FxCop 所暗示的性能下降。
6 CLR静态构造器的更多相关文章
- 静态构造器(static constructor)
1.定义: 静态构造函数是实现对一个类进行初始化的方法成员. 它一般用于对静态数据的初始化. 静态构造函数不能有参数,不能有修饰符而且不能被调用,当类被加载时,类的静态构造函数自动被调用. 2.特点: ...
- Clr静态数据Table-Valued函数
前两天Insus.NET实现一个功能<在数据库中提供只读数据>http://www.cnblogs.com/insus/p/4384411.html ,在数据库中为程序提供静态数据.它是在 ...
- 6 CLR实例构造器
引用类型构造器 如果我们没有定义实例构造器,那么编译器会为我们默认产生一个无参构造器. 实例对象初始化过程 为实例分配内存: 初始化附加成员,包括方法表指针和SyncBlockIndex变量(我们已经 ...
- CLR类型设计之方法与构造器
无论学习那门语言都要学习函数体,C#,JAVA,PHP,都会涉及到函数体,而C#的函数体成员并不少,方法和构造器就是函数体成员之一,函数体成员还包括但不限于:方法,属性,构造器,终结器,运算符及索引器 ...
- 《CLR.via.C#第三版》第二部分第12章节 泛型 读书笔记(六)
终于讲到泛型了.当初看到这个书名,最想看的就是作者对泛型,委托,反射这些概念的理解.很多人对泛型的理解停留在泛型集合上,刚开始我也是,随着项目越做越多,对待泛型的认识也越来越深刻. 泛型的概念:泛型是 ...
- c# 变量,对象,静态类型,集合类的线程安全回顾
1.变量的线程安全性与变量的作用域有关. 2.对象 对象是类型的实例 在创建对象时,会单独有内存区域存储对象的属性和方法.所以,一个类型的多个实例,在执行时,只要没有静态变量的参与,应该都是线程安全的 ...
- CLR via C#(05)- 访问限定、数据成员
今天跟大家分享一下关于访问限定和数据成员的知识.主要包括以下两点: Abstract, sealed, virtual, new, override怎么用? Const 和 readonly好像都表示 ...
- CLR via C#学习笔记----知识总概括
第1章 CLR的执行模型 托管模块的各个组成部分:PE32或PE32+头,CLR头,元数据,IL(中间语言)代码. 高级语言通常只公开了CLR的所有功能的一个子集.然而,IL汇编语言允许开发人员访问C ...
- CLR 初步
1. 源代码编译为托管模块 程序在.NET框架下运行,首先要将源代码编译为 托管模块.CLR是一个可以被多种语言所使用的运行时,它的很多特性可以用于所有面向它的开发语言.微软开发了多种语言的编译器,编 ...
随机推荐
- Scala:Next Steps in Scala
Array val greetStrings = new Array[String](3) greetStrings(0) = "Hello" greetStrings(1) = ...
- centos npm 安装后 command not found
ok,
- iOS:NSFileHandle和NSFileManger的使用
一.介绍 利用NSFilehandle类提供的方法,允许更有效地使用文件. 一般而言,处理文件时都要经历以下三个步骤: 1.打开文件,并获取一个NSFileHandle对象,以便在后面的I/O操作中引 ...
- [Unit Testing] Using Mockito Annotations - @Mock, @InjectMocks, @RunWith
Previously we have seen how to do Unit testing with Mockito; import org.junit.Test; import static or ...
- toml
其目标是成为一个小规模的易于使用的语义化配置文件格式.TOML被设计为可以无二义性的转换为一个哈希表(Hash table). # 这是一个TOML文件 title = "TOML Exam ...
- AaronYang WCF教程目录
============================原创,讲究实践===================== 1. 那天有个小孩教我WCF[一][1/3] 基本搭建 阅读 ...
- SNFAutoupdater通用自动升级组件V2.0
1.组件介绍 C/S构的特点是能充分发挥客户端的处理能力,很多工作可以由客户端处理后再提交给服务器,对应的优点就是客户端响应速度快模式客户端以其强大的功能,丰富的表现力受到相当大部分用户的青睐,但是客 ...
- [Big Data - Kafka] Kafka设计解析(五):Kafka Benchmark
性能测试及集群监控工具 Kafka提供了非常多有用的工具,如Kafka设计解析(三)- Kafka High Availability (下)中提到的运维类工具——Partition Reassign ...
- 如何让vue文件中的代码在Sublime Text 3中高亮和智能提示
大家写在Sublime Text 3中编写vue文件时,会发现没有代码智能提示,清一色的黑底白字,不会像html.js一样变成彩色,给我们带来了很大的不便.所以需要安装一款叫作Vue Syntax H ...
- 【系统移植】uboot详细分析
uboot使用 uboot控制台,倒计时 命令: 调试,操作一些硬件 setenv printenv saveenv nand erase nand write tftp zImage h ...