基础
.NET Framework
- 面向对象的平台
- 组成
(1)公共语言运行库(Common Language Runtime,CLR):管理应用的执行(内存管理、代码安全验证、代码执行、垃圾收集等)
(2)基类库(Base Class Libraries,BCL):工具类 - 执行
(1)先编译成通用中间语言(Common Intermediate Language,CIL),由Visual Studio完成;运行时,再由即时编译器(Just-In-Time,JIT)编译成本机代码(native code,目标操作系统能理解的语言)
(2)CIL代码存储在程序集(assembly)中:可执行(.exe)文件和库(.dll)文件 - CLR托管代码
(1)管理内存、处理安全性等,C#只能写托管环境下运行的代码,C++可以编写非托管类型(如访问操作系统的底层功能)
(2)垃圾回收(Garbage collection, GC)
C#图解教程第一章 C#和.NET框架【听好的看看】
Java是先通过编译器编译成Bytecode,然后在运行时通过解释器将Bytecode解释成机器码;
C#是先通过编译器将C#代码编译成CIL,然后通过CLR将IL编译成机器代码。
所以严格来说Java是一种先编译后解释的语言,而C#是一门纯编译语言,且需要编译两次。
C#基础知识
- 特点(安全、稳定、简单)
(1)面向对象(Object-Oriented Programming,OOP),封装、继承(C#仅支持单继承,提供了接口来解决多重继承)、多态
(2)不允许直接操作内存,去掉了指针操作
(3)比 C、C++、Java 提供了更多的数据类型,每个数据类型都是固定大小的
(4)与Web紧密结合,支持绝大多数的Web标准,例如HTML、XML等
(5)提供了方便的功能,如垃圾收集、类型安全、版本控制等
一些关键字
ref
在方法的参数列表中使用 ref 关键字时,它指示参数按引用传递,而非按值传递。
void Method(ref int refArgument)
{
refArgument = refArgument + 44;
}
int number = 1; // 需要初始化,不能是常量 const int number = 1;
Method(ref number);
Console.WriteLine(number); // Output: 45
- 不能用在常量上;
- 不能用在没有初始化的变量上;
- 属性不是变量,是方法,不能传递 ref 参数;
- 通过引用传递参数时,在变量声明前必须要ref关键词(数组这种不用);
// 有问题,int x -- ref int x
ref int F(int x)
{
return ref x;
}
// 没问题
ref int F()
{
int[] a = {1};
return ref a[0];
}
out
- 不需要初始化,但是在方法返回前必须赋值;
- 方法定义和调用方法均必须显式使用 out 关键字;
in
in 关键字会导致按引用传递参数,但确保不修改参数。
按引用传递这些参数可以避免(可能产生的)高昂的复制成本。
- 临时变量允许将编译时常数作为 in 参数;
- 存在从实参类型到形参类型的隐式转换时,临时变量允许使用实参;
- 临时变量允许使用属性或 in 参数的其他表达式(没有看明白。。。);
- in关键字可以隐式使用,如果存在方法重载,in就不能隐式使用了。
ref、in和out三者异同
- 传递到 ref 或 in 形参的实参必须先经过初始化,然后才能传递;out 形参在传递之前,不需要显式初始化该形参的实参。
- in 参数无法通过调用的方法进行修改;out 参数必须由调用的方法进行修改;而 ref 参数是可以修改的。
- 在其他要求签名匹配的情况下(如隐藏或重写),in、ref 和 out 是签名的一部分,相互之间不匹配。
- 不能将 ref、in 和 out 关键字用于以下几种方法:
(1)异步方法,通过使用 async 修饰符定义;
(2)迭代器方法,包括 yield return 或 yield break 语句。 - 扩展方法还限制了对以下这些关键字的使用:
(1)不能对扩展方法的第一个参数使用 out 关键字;
(2)当参数不是结构或是不被约束为结构的泛型类型时,不能对扩展方法的第一个参数使用 ref 关键字;
(3)除非第一个参数是结构,否则不能使用 in 关键字。 即使约束为结构,也不能对任何泛型类型使用 in 关键字。