C#.NET编程的特点

重新看下.NET的特点,细节之处以C#为准。

基本特点

.NET一般用来代指【.NET Framework】,但严格的说.NET是一种设计,【.NET Framework】是其在Windows上的实现API,而【.NET Core】是其跨平台实现的API。

.NET中的【托管代码ManagedCode】是基于各个版本的【.NET API】编写的代码。

【托管代码ManagedCode】在手动编译后,会被编译为【中间语言IL】。

.NET中的【托管代码】具有与具体语言无关的特点,无论是是使用C#还是VB,还是C++,还是F#编写的程序,只要基于【.NET API】,都会被编译为【中间语言IL】。

与【.NET API】对应的runtime被称为【公共语言运行时CLR】-----大概是为了强调多种语言共用这一特点所以叫公共xxxxx。

在程序初次运行的时候,【公共语言运行时CLR】会启用【实时编译JIT--Just In Time】将【中间语言IL】编译为机器码,并且会进行【自动垃圾回收】。

---------------------------------------------------------------------------------------------------------------------------------------------------------

原来的时候JIT是.NET编程的一个特点。(后来,Java好像也有JIT了,只是用得少。but who cares?)

因为Java是先编译为中间代码,后由虚拟机jvm解释执行的。理论上可以一次编译,多处运行,就是会慢一点。

而C是直接编译成机器码运行的。很快,但在不同平台上要多次编译。

而.NET代码是先编译为中间代码,再编译为机器码执行的。这样就是第一次运行慢一点,后面就快了。

-------------------------------------------------------------------------------------------------------------------------------------------------------

总体来说,.NET的特点就是.NET的设计本身。

  • 一个设计的点是,多种语言向IL转化。
  • 另一个设计点是,IL会被CLR进行JIT生成机器码。

不仅跨平台,而且跨语言。

它的前提是在各个不同平台上实现.NET API和CLR。

----------------------------------------------------------------------------------------------------------------------------------------------------------

贴一段.NET的历史。

本身.NET Framework的设计,包括上面语言的设计都是非常良好的。但就是不开源,而且基本上只在Windows上有API和Runtime。

后来Xamarin公司自己基于.NET的设计实现了一套Mono,Mono 是跨平台的实现,它可以运行在 Windows 、 Linux、FreeBSD、Unix、OS X 、 Solaris、Playstation 3、Wii 、 XBox 360等系统上。

大概是因为Java的开源,还有mono的开源所逼的,微软也开源了。

.NET Framework开源。

Xamarin直接被微软买了过来,用VS可以直接新建Xamarin程序了,微软也做了自己的.NET开源跨平台实现------【.NET Core】。

前段时间还听说有人要在浏览器端实现.NET设计,这样就能用C#写前端,统一前后端语言了,目前感觉不是很靠谱的样子。

然后前段时间下载VS的时候我发现VS也是开源的。

github也被微软买了过来,微软还开源了windows计算器。

常见类型系统

指的是.NET中包含5中常见类型:类,结构体,枚举类型,接口,委托。

最基本的类型当然是:值类型,引用类型。

其他常见的东西,像是public,override这些修饰词;但没有通用到常见类型系统的程度。

为什么.NET要提出【常见类型系统Common Type System】呢?

因为.NET要跨语言实现,语言之间需要有一定的共性。

各种语言都要调用同样的【语言无关的组件Language-Independent Components】。

所以各种语言中都要有CTS中的这些东西。

所以常见类型系统是跨语言的关键。

目标框架

我的代码可以把一个【.NET Standard】作为目标框架。

【.NET Standard】是微软文档中的说法,它包括了【.NET APIs】,还包括了【所有可以引用的东西】。要我自己说就是【.NET APIs】嘛。

也就是说【.NET Standard】是.NET的标准设计,而各套【.NET APIs】是具体实现,各套具体实现可能还包含自己特有的内容。

为什么要有目标框架的概念呢?因为各套【.NET APIs】之间存在差异,通过特定的API才能获得特定的功能。

例如,你要把将目标框架设置为【Xamarin.iOS】才能获得IOS10上特有的功能。

以下截图来自微软文档,列出了实现了【.NET Stantard】的一些【APIs】版本。

当然,微软文档认为的目标框架,不只是【APIs】的版本,但那样说这个列表就太长了,不容易把握。

并且,可以通过一些配置文件或者特殊的标记语法,来指定目标框架。

这个目标框架呢,就是跨平台的关键。

托管代码

托管代码托管了什么?托管是指,CLR编译IL代码,并且执行,而非源码本身直接被编译和执行。

编程的本质是操作内存对象,而内存管理/边界安全/类型安全,对程序员是一种负担。托管的意义就在于解放程序员。

能不能调用非托管代码?可以。

C#中可以通过[DllImport()]特性引用C写的函数。

那能不能写非托管代码?可以。

C#中可以可以在unsafe的花括号内写非托管代码,这一小段非托管代码里面,你就自己管理内存了,这段代码不会编译成IL,而会直接编译成二进制,并且可以使用指针。

Visual Studio

编辑器好用,写起来6,也是一个特点吧。

面向对象

字段,属性,方法。

封装,继承,接口。

什么都不缺,并且是真正的完全面向对象的,没有设计上的漏洞,还有自己的设计。

委托和Lambda和事件

委托是方法的类型,委托对象是方法的指针。我可以把方法赋值给委托的实例。

函数不能当参数,但是委托可以。

public delegate string Reverse(string s);

Reverse rev = ReverseString;//把ReverseString方法赋值给了rev对象。接下来rev对象可以直接调用ReverseString方法,也可以当作参数传递。

这样说是非常严谨的;然后看到js里用btn.click(function(){})注册click的响应方法,然后用btn.click()能够直接调用被注册的方法时,总觉得后面缺了一个Invoke()。

-----------------------------------------------------------------------------------------------------------------------------------------------------------

lamda用起来真是666,本来委托类型一行,一个方法至少3行,给委托了实例赋值也得要一行。

而lamda一行就能写出这些意思来。

Func<string,string> rev = (s)=>{return new string(s.Reverse().ToArray());};

Func<string,string>是委托类型。

rev是委托的实例。

等号右边呢是一个匿名方法。

-------------------------------------------------------------------------------------------------------------------------------------------------------------

事件是在委托基础上建立的概念。

一个委托只能将一个方法作为值。

而事件相当于List<Delegate>,并且事件类型重载了-=和+=运算符。

btn.Click+=【某个符合其委托签名的方法】,就相当于listDelegate.Add(【某个符合其委托签名的方法】)。

相应的,-=相当于remove方法。

相比起来,事件有了多次订阅和取消的能力。

泛型T

例如List<int>,当你使用List<int>而不是ArrayList的时候,可以避免取值和存值的拆箱和装箱;并且可以获得VS更加智能的提示。

泛型也是写纯算法的一个前提,可以实现算法重用。

所有具有GetChilds方法或类似方法的对象都可以作为一个T,这就是函数式的看法。

面向对象那一套只能用接口来抽象各种行为,但是不够抽象。

Func<T,List<T>>就能表达T具有GetChilds方法。而且我没有写死方法的名称。

如果用接口写,方法名必须是死的。

这样以后呢,我就可以写出GetAllChilds()方法,像下面这样。

public static List<T> GetAllChilds<T>(T root, Func<T, List<T>> GetChilds)

我可以如下获取所有子路径

我可以如下获取所有的子节点

----------------------------------------------------------------------------------------------------------------------------------------------

可以看出面向对象所讲的重用并不是真正的重用;函数式的重用才是真正的重用。--------算法重用。

面向对象所讲的重用,太依赖于具体的结构。必须是相同名字的才能重用。-----------它是看你叫什么所以你是怎么样的,而函数式的重用关心你做什么你与另一个东西的关系。

函数式的重用实现了数学上追求的一个理念----------不关心事物的定义,只关心事物和事物之间的关系,太美妙了。

面向对象是经验式的,函数式是逻辑推理式的。人同时拥有这两种思维方式,看实际情况哪个顺手就用哪个吧。

-------------------------------------------------------------------------------------------------------------------------------------------------

面向对象中,封装把方法和数据放一起是增加耦合性的。

然后面向对象中两个类之间的过于密切的关系是增加耦合性的。也可以用函数式的两个函数之间的关系来描述。

面向对象中的接口,把部分方法抽离出来,是减少耦合性的。

相比之下函数式的写法没有增加问题本身的复杂程度,就是头疼医头脚疼医脚。

实际的工程中需要怎么做那就是一个经验和权衡的问题了。

LINQ

linq---语言集成查询。

select,where,Aggregate,这些高阶的动词提供了操作数据的简单方法,并且用的这些词跟sql似的,让人很熟悉。

有了它就能愉快的写一些声明式的代码了。

linq最大的好处可能是精简代码,体现代码的意图。

特性Attributes

没什么好说的,给贴个标签,它就有相应的功能了。

声明式的输入校验看着挺合理的,好用就是硬道理。

并行/并发和异步的处理

Task<T>,async,await,Parallel。

【语言级别的异步处理机制】很语义化。

总结

----------------------------------------------------------------------------------------------------------------------------------------------

.NET以.NET Standard作为设计标准,在不同平台上提供了与语言无关的.NET APIs和公共语言运行时。

.NET的常见类型系统【类,结构体,枚举类型,接口,委托】使其能够提供语言无关的组件。

多个版本的目标框架使其实现了跨平台。

----------------------------------------------------------------------------------------------------------------------------------------------

按照.NET的设计,程序员会了一门语言,并且熟悉了一个平台上的框架后就可以编码了,其它的事情都不用管。

并且VS的强大为程序员的编码工作提供了强力支持。

----------------------------------------------------------------------------------------------------------------------------------------------

面向对象的【字段,属性,方法。类,继承,接口】

函数式的【委托,事件,lambda,扩展方法,泛型】

声明式的【LINQ,Attributes】

还有【语言级别的异步编程模型,async,await,Task<T>】

.NET不拘泥于某种编程范式,只要是好用的,都会拿来用。

-------------------------------------------------------------------------------------------------------------------------------------------

最后总结下特点

【跨平台】【多范式】。

程序员只需关注核心的设计和编码工作。

其他问题由【APIs,运行时,VS,语言本身的特性】等帮你解决。

这也许就是.NET的特点。 

 

posted @ 2019-11-16 16:35  法宝  阅读(468)  评论(0编辑  收藏  举报