CLR via c#读书笔记五:方法
注:书本第8章:方法
实例构造器和类(引用类型)
构造器方法在“方法定义元数据表”中始终叫做.ctor(constructor的简称)。
构造引用类型的对象,在调用类型的实例构造器之前,为对象分配的内存总是先被归零。
如果类没有显示定义任何构造器,c#编译器将定义一个默认(无参)构造器。如果类的修饰符为abstrat,那么编译器生成的默认构造器的可访问性就为protected;否则,构造器会被赋予public可访问性。静态类编译器根本不会生成默认构造器。
类的实例构造器在访问从基类继承的任何字段之前,必须先调用基类的构造器。
Obect的MemberwiseClone方法可以在不调用实例构造器的前提下创建类型的实例。
“内联”初始化,编译器会转换成调用构造器方法中的代码来执行。这样会容易使代码膨胀。
实例构造器和结构(值类型)
值类型其实并不需要定义构造器,c#编译器根本不会为值类型内联(嵌入)默认的无参构造器。
考虑到性能,CLR不会为包含在引用类型中的每个值类型字段都主动调用构造器。但是,值类型的字段会被初始化为0或null。
CLR确实允许为值类型定义构造器,但是必须显示调用才会执行。如下例子:
internal struct Point{
public Int32 m_x,m_y;
public Point(Int32 x,Int32 y){
m_x=x;
m_y=y;
}
}
internal sealed class Rectangle{
public Point m_topLeft,m_bottomRight;
public Rectangle(){
//在C#中,向一值类型应用关键字new,
//可以调用构造器来初始化值类型的字段
m_topLeft=new Point(1,2);
m_bottomRight=new Point(100,200);
}
}
C#编译器不允许值类型定义无参构造器;但是CLR允许。所以C#不能在值类型中内联初始化字段。但是可以为静态字段进行内联初始化。
在值类型的构造器中,this代表值类型本身的一个实例,用new创建的值类型的一个实例可以赋给this。而在引用类型的构造器中,this被认为是只读的,所以不能对它进行赋值。
类型构造器(静态构造器、类构造器)
类型构造器方法总是叫.cctor。
类型默认没有定义类型构造器。如果定义,也只能定义一个。此外,类型构造器永远没有参数。c#自动将类构造器标记为private。不允许出现修饰符。
类型构造器只会被执行一次。
值类型可以定义类型构造器(但如前所述,不能定义无参实例构造器),但永远不会被执行。
类型只有在AppDomain卸载时才会卸载。AppDomain卸载时,用于标识类型的对象(类型对象)将成为“不可达”的对象(不存在对它的引用),垃圾回收器会回收类型对象的内存。
CLR不支持静态Finalize方法。可以在System.AppDomain类型的DomainUnload事件登记一个回调方法。
操作符重载方法
CLR规范要求操作符重载方法必须是public和static方法。要求操作符重载方法至少有一个参数的类型与当前定义这个方法的类型相同,以便C#编译器能在合理的时间内找到要绑定的操作符方法。
public sealed class Complex{
public static Complex operator+(Complex c1,Complex c2){
}
}
编译器为名为op_Addition的方法生成元数据方法定义项;这个方法还设置了specialname标志,表明这是一个“特殊”方法。
转换操作符方法
将对象从一个类型转换为另一种类型的方法。CLR规范要求转换操作符重载方法必须是public和static方法。
一般的非基元类型,在执行强行转型时,CLR只是检查对象的类型和目标类型(或者从目标类型派生的其他类型)是不是相同。
public sealed class Rational{
//由一个Int32构造一个Rational
public Rational (Int32 num){ ... }
//由一个Single构造一个Rational
public Rational(Single num){ ... }
//将一个Rational转换成一个Int32
public Single ToString(){ ... } //由一个Int32隐式构造并返回一个Rational
public static implicit operator Rational(Int32 num){
return new Rational(num);
}
//由一个Single隐式构造并返回一个Rational
public static implicit operator Rational(Single num){
return new Rational(num);
}
//由一个Rational显式返回一个Int32
public static explicit operator Int32(Rational r){
return r.ToInt32();
}
//由一个Rational显示返回一个Single
public static explicit operator Single(Rational r){
return r.ToSingle();
}
}
在C#中,impicit关键字告诉编译器为了生成代码来调用方法,不需要在源代码中进行显示转型,expicit只有发现了显示转型,才调用方法。
只有在转换不损失精度或数据级的前提下(比如将个Int32转换成Rational)才能定义隐式转换操作符。如果有,应该定义一个显式转换操作符。
使用强制类型转换表达式时,C#生成代码调用显式转换操作符方法。使用as或is操作符时,则永远不会调用这些方法。
扩展方法
它允许定义一个静态方式,并用实例方法的语法来调用。要将方法变成扩展方法,只需在第一个参数前加this关键字:
public static class StringBuilderExtensions{
public static Int32 IndexOf(this StringBuilder sb,Char value){//假定StringBuilder中没有IndexOf方法
for(Int32 index=0;index<sb.Length;index++)
if(sb[index]==value)return index;
return -1;
}
}
Int32 index=sb.IndexOf('X');
C#只支持扩展方法,不支持扩展事件、属性、操作符等。
分部方法
//工具生成的代码,存储在某个源代码文件中
internal sealed partial class Base{
private String m_name;
//这是分部方法的声明
partial void OnNameChanging(string value);
public String Name{
get{ return m_name; }
set{
OnNameChanging(value.ToUpper());//通知类要进行更改了
m_name=value;//更改字段
}
}
} //开发人员生成的代码,存储在另一个源代码文件中:
internal sealed partial class Base{
//这是分部方法的实现,会在m_name更改前调用
partial void OnNameChanging(String value){
if(String.IsNullOrEmpty(value))
throw new ArgumentNullException("value");
}
}
分部方法只能在分部类或结构中声明;
分部方法的返回类型始终是void,任何参数都不能用out修饰符来标记。
分部方法总是被视为private方法,但是c#编译器禁止在分部方法声明之前添加private关键字。
CLR via c#读书笔记五:方法的更多相关文章
- clr via c#读书笔记五:常量和字段
1.常量是值从不变化的符号.只能定义编译器识别的基元类型的常量.如:Boolean,Char,Byte,SByte,Int16,UInt16,Int32,UInt32,Int64,Single,Dou ...
- [Clr via C#读书笔记]Cp8方法
Cp8方法 构造器 作用就是初始化所有成员字段:.ctor:派生类和基类都有自己的构造函数.默认有一个无参数的构造函数,值字段初始化为0,引用字段初始化为null:可以有多个构造器: 值类型的初始化其 ...
- CLR via C# 读书笔记---常量、字段、方法和参数
常量 常量是值从不变化的符号.定义常量符号时,它的值必须能在编译时确定.确定后,编译器将唱两只保存在程序集元数据中.使用const关键字声明常量.由于常量值从不变化,所以常量总是被视为类型定义的一部分 ...
- 《CLR.via.C#第三版》第二部分第10,11章节读书笔记(五)
第10章是对 属性 做阐述 属性本质上是方法 匿名类型的使用: }; Console.WriteLine("Name={0},Year={1}",o1.Name,o1.Year); ...
- CLR via C# 读书笔记-26.线程基础
前言 这俩个月没怎么写文章做记录分享,一直在忙项目上线的事情,但是学习这件事情,停下来就感觉难受,clr线程这章也是反复看了好多遍,书读百遍其义自见,今天我们来聊下线程基础 1.进程是什么,以及线程起 ...
- CLR via C# 读书笔记-27.计算限制的异步操作(上篇)
前言 学习这件事情是一个习惯,不能停...另外这篇已经看过两个月过去,但觉得有些事情不总结跟没做没啥区别,遂记下此文 1.CLR线程池基础 2.ThreadPool的简单使用练习 3.执行上下文 4. ...
- CLR via C# 读书笔记-21.托管堆和垃圾回收
前言 近段时间工作需要用到了这块知识,遂加急补了一下基础,CLR中这一章节反复看了好多遍,得知一二,便记录下来,给自己做一个学习记录,也希望不对地方能够得到补充指点. 1,.托管代码和非托管代码的区别 ...
- Effective Java 读书笔记之六 方法
一.检查参数的有效性 1.考虑参数有哪些限制,把限制写到文档中,在方法的开头处通过显式地检查来实施这些限制. 二.必要时进行保护性拷贝 1.如果类具有从客户端得到或者返回的可变组件,类就必须考虑保护性 ...
- Clr Via C#读书笔记---I/O限制的异步操作
widows如何执行I/O操作 构造调用一个FileStream对象打开一个磁盘文件-----FileStream.Read方法从文件中读取数据(此时线程从托管代码转为本地/用户模式代码)- ...
随机推荐
- Html 列表实现展开和收起
HTML中,点击列表元素,在其下展开更多的小选项.不点的时候是收起来的.就是实现路由器左边的菜单那样的功能.怎么实现,知道的指点一下,谢谢了!! 最常见的方法是通过Javascript控制某标签的CS ...
- UVa 1638 - Pole Arrangement(dp)
链接: https://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem& ...
- BZOJ2440:[中山市选2011]完全平方数(莫比乌斯函数)
Description 小 X 自幼就很喜欢数.但奇怪的是,他十分讨厌完全平方数.他觉得这些数看起来很令人难受.由此,他也讨厌所有是完全平方数的正整数倍的数.然而这丝毫不影响他对其他数的热爱. 这天是 ...
- @RequestParam和@PathVariable的区别及其应用场景
@RequestParam和@PathVariable这两者之间区别不大,主要是请求的URL不一样 用@RequestParam请求接口时,URL是:http://www.test.com/user/ ...
- mybatis全局配置文件
一.properties:引入外部配置文件 1.resource :引入类路径下的全局配置文件,例如:<properties resource="conf/dbconfig.prope ...
- 图形解析理解 css3 之倾斜属性skew()
1.作用 改变元素在页面中的形状2.语法 属性:transform 函数: 1.skew(xdeg) 向横向倾斜指定度数 x取值为正,X轴不动,y轴逆时针倾斜一定角度 x取值为负,X轴不动,y轴顺时针 ...
- unittest单元测试框架之测试环境的初始化与还原(fixture)(五)
1.方法一:针对每条测试用例进行初始化与还原 import unittest from UnittestDemo.mathfunc import * class TestMathFunc(unitte ...
- Element表单验证规则
一.简单的逻辑验证使用方法: 方法步骤: 1.在html中给el-form增加 :rules="rules" 2.html中在el-form-item 中增加属性 prop=&qu ...
- #leetcode刷题之路37-解数独
编写一个程序,通过已填充的空格来解决数独问题.一个数独的解法需遵循如下规则:数字 1-9 在每一行只能出现一次.数字 1-9 在每一列只能出现一次.数字 1-9 在每一个以粗实线分隔的 3x3 宫内只 ...
- CF1066EBinary Numbers AND Sum(前缀和,二进制)
题目大意 现在,给你两个位数为 n 和 m 的两个二进制数a,b,现在,我们要进行如下操作: 计算a&b 答案累加上一个操作的值 bbb右移一位,最后一位直接舍弃 现在,请你算出最终的答案,并 ...