博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
《CLR via C#》读书笔记 之 方法
阅读量:6195 次
发布时间:2019-06-21

本文共 1300 字,大约阅读时间需要 4 分钟。

第八章 方法

2013-02-27

8.1 实例构造器和类(引用类型)


创建引用类型的实例的过程

(1)       为实例的数据字段分配内存(实例字段包括本身及其基类的实例字段)

(2)       然后初始化对象的附加字段(类型对象指针和同步块索引)

(3)       调用类型的实例构造器来设置对象的初始状态。

基类的构造器总是在类的实例构造器之前调用,这是为了使代码“可验证”(当类的实例构造器访问从基类继承来的字段)。因此,不要再构造器中调用虚方法(比如这个虚方法要在子类中重写,可是重写方法所用字段还没初始化)。

C#提供简单语法,允许在构造引用类型实例时,对类型中定义的字段进行“内联”初始化

1     class SomeType2     {3         private int m_x = 5;4     }

我们通过SomeType构造器的IL代码发现,字段初始化的代码会在幕后移入到构造器中,构造器IL代码的执行顺序为:字段初始化=》基类构造器=》构造器本身的代码,因此,尽量把初始化任务放在构造器中。

 

和其他方法不同,实例构造器永远不能被继承。

C#编译器将定义一个默认(无参)构造器,下表显示类修饰符与默认构造器的关系。

表1类修饰符与默认构造器的关系

类的修饰符

默认构造器

无abstract

默认构造器可访问性为public

有abstract

默认构造器可访问性为protected

static(静态类在元数据中是抽象密封类)

编译器根本不会在类中定义生成一个默认构造器

8.2 实例构造器和结构(值类型)


 

值类型(struct)构造器的工作方式与引用类型(class)的构造器截然不同。C#编译器根本不会为值类型生成无参构造器。即使为值类型提供了无参构造器,许多编译器也永远不会生成代码调用它。C#编译器干脆不允许值类型定义无参构造器,也不允许值类型中内联方式初始化实例字段。

注意:严格的说,只有当值类型的字段嵌套到引用类型中时,才保证会被初始化为0或null。但是,基于栈的值类型字段不保证为0或null。

        为了代码的“可验证性”(verifiablity),任何基于栈类型的子类型都必须在读取前写入(赋值)。如果允许代码先读再写,就会造成安全漏洞,所以C#和其他生成“可验证”代码的编译器可以保证对它们进行“置零”。

8.3 类型构造器


 

类型默认没有定义类型构造器,如果定义,也只能定义一个。此外,类型构造器永远没有参数。

类型构造器必须标记为static,而且,总是私有的,C#会自动把它们标记为private。

任何一个类型定义了类型构造器,JIT编译器都会检查针对当前AppDomain,是否已经执行了这个类型构造器来决定是否要调用它。

注意:由于CLR保证一个类型构造器在每个AppDomain1中只能执行一次,而且是线程安全的,因此,非常适合在类型构造器中初始化类型需要的任何单实例(Singleton)对象。

类型构造器不应调用其基类型的类型构造器。这种调用没有必要,因为类型并没有从其基类型继承静态字段。

8.6 扩展方法


 

对无法修改的类扩充其方法,要小心使用。

转载地址:http://ujuca.baihongyu.com/

你可能感兴趣的文章
C语言笔记
查看>>
ajax实例1
查看>>
js学习篇--数组按升序降序排列
查看>>
MyBatis笔记——Mapper动态代理
查看>>
ElasticSearch入门及核心概念介绍
查看>>
VC++2010开发数字图像系统1
查看>>
表达式求值
查看>>
使用PPRevealSideViewController创建抽屉式导航
查看>>
算法踩坑小记
查看>>
网关地址设置
查看>>
一个动画 Label (走马观花)
查看>>
requests 中文乱码
查看>>
VMware下ubuntu与Windows实现文件共享的方法(zhuan)
查看>>
再见了Server对象,拥抱IHostingEnvironment服务对象(.net core)
查看>>
Javassist初体验
查看>>
LaTex in Markdown
查看>>
eclipse中使用svn
查看>>
bootstrap 初始学习篇
查看>>
Reverse Linked List II
查看>>
python发送邮件
查看>>