C Sharp语法深入系列

目录

  • .NET框架(CLR,BCL,FCL,CTS,CLS)
  • C#基础语法(拆箱装箱、值类型与引用类型、String与字符操作、类抽象类接口委托事件、常量属性字段方法特性、抽象继承多态、泛型、关键字)
  • 内存管理与GC垃圾回收
  • 内置数据类型(基元类型、System.Type、System.Object、System.ValueType、)
  • 内置数据结构()
  • 接口分析(IEnumerable、IEnumerable< T >、IComparable、IComparable< in T >、IFormattable、IFormatProvider、IConvertible、ICloneable、IEquatable< T >、IComparer< in T >、IEqualityComparer< in T >、IDisposable、IAsyncResult、ICollection< T >、IList< T >、IQueryable、IQueryProvider、IDictionary< TKey, TValue >、INotifyPropertyChanged、IEditableObject、ISerializable)
  • Lambda、linq、SQL
  • 进程与线程、多线程与异步
  • I/O操作
  • 事件总线

推荐阅读

.NET面试题系列目录

.NET面试题解析

DotNet进阶系列

.NET面试题总结


IL学习

30分钟?不需要,轻松读懂IL

进阶篇:以IL为剑,直指async/await

深入浅出.Net IL

你的C#代码是怎么跑起来的(一)

你的C#代码是怎么跑起来的(二)

c#内存管理概要


.NET框架

C#基础

基类(抽象类)还是接口? 项目实战中如何使用抽象类和接口 —— 大共性放基类,小共性用接口

接口:行为规范,表示能做什么事,成员包含方法、属性、索引器、不含字段和实现的方法

抽象类:字段、属性、实现的方法

1、IS-A对比CAN-DO关系

IS-A代表属于,CAN-DO代表能做某事。
如果派生类型属于基类型,那么就用基类;如果建立的是能做的关系就用接口

2、易用性

对于开发人员,定义从基类派生的类型通常比实现接口的所有方法要容易的多。基类可提供大量功能,所以派生类型可能只需要稍加改动即可。而提供接口的话,派生类型则需要实现所有成员。

3、一致性实现

无论接口协定订立的多好,都无法保证所有人100%的实现它,而如果基类正确的话,以后根据需要稍加修改即可。

4、版本控制

向基类添加一个方法,派生类将继承该方法。而如果向接口添加新成员,则会强迫接口的继承者更改源代码并重新编译。

因为接口是一种特殊的抽象类,所以接口是引用类型。

因为委托是密封类,所以委托是引用类型。

5、类和类成员的访问权限

C# 成员默认访问权限(public、private、protected、internal)

泛型

原理:延时声明,在运行时进行编译。

四种:泛型方法、泛型类、泛型接口、泛型委托

泛型约束:... where T:< ... >

委托与事件

Delegate:表示函数参数(签名)和返回值的类型 函数的指针|引用|类型

可以利用多播委托进行解耦和插件式开发。

默认委托:

  • Func< T > 有返回值 Func< int,String >返回值类型为String,参数为int
  • Action< T > 无返回值 Action< int,int>参数为两个int类型

Linq

延时加载

数据流

C# 流总结

  • FileStream,文件流,继承Stream类
  • MemoryStream,内存流
  • NetworkStream,网络流
  • BufferedStream
  • StreamReader/StreamWriter
  • TextReader/TextWriter

内存管理与GC垃圾回收

引用、指针和句柄的区别

Windows的系统API数目都是十万左右,随着Windows版本的提高而增加。任何一种编程语言都可以调用Windows系统API函数。

由于Windows系统是由C和C++语言写的,所以Windows根据C语言的基本数据类型重新定义了一些新的数据类型(都是大写)。Windows是面向对象的系统设计,所以定义了一些句柄(Handle)的数据类型,用句柄HANDLE来表示一个Windows系统的对象,比如图标句柄(HICON)、程序实体(HINSTANCE)、操作注册表时注册表值得句柄(HKEY)、模块句柄(HMODULE)、窗口句柄(HWND)等。

最常见的一类非托管资源如系统文件句柄(文件流),系统的窗口句柄,网络连接(网络流),数据库连接,画刷,图标,打印机资源等。(在windows系统中的内存管理一般会将当前处于空闲状态的对象的内存释放掉,当需要访问的时候再重新提交分配物理内存,从而导致对象的物理地址是变化的。句柄是一种指向指针的指针,由于这些非托管资源使用句柄进行物理内存中对象的管理,而这些非托管对象是有状态的、持续不中断的对象资源,所以使用句柄进行内存的管理,从而不受.NET中GC的管理,这种间接访问对象的模式增强了系统对引用对象的控制)。操作系统在管理这些句柄(所有系统对象的动态地址,其中包括系统的每张图片,每个IO资源,每个窗口等)。当一个进程被初始化时,系统要为它分配一个句柄表(内核对象)。

什么是垃圾?简单理解就是没有被引用的对象。

这类资源,垃圾回收器在清理的时候会调用Object.Finalize()方法。默认情况下,方法是空的,对于非托管对象,需要在此方法中编写回收非托管资源的代码,以便垃圾回收器正确回收资源。

为了更方便的释放非托管资源,微软为非托管资源的回收专门定义了一个接口:IDisposable,接口中只包含一个Dispose()方法。任何包含非托管资源的类,都应该继承此接口,以确保非托管资源的正确释放。 在一个包含非托管资源的类中,资源释放的标准做法是:

  • 继承IDisposable接口
  • 实现Dispose()方法,在其中释放非托管资源,并将对象本身从垃圾回收器中移除(为了告诉垃圾回收器,此对象的资源我们已经自己释放了,它就没必要再对此对象进行回收了);
  • 实现类析构函数,在其中释放非托管资源。

C#调用C++或者C的DLL库中的函数也属于非托管资源。.NET环境下的DLL、exe叫做程序集,能够在CLR中被直接引用,非托管DLL(Dynamic Link Library)就是平常所的动态链接库等,其中就包括了封装所有Windows API函数的DLL文件。各种非托管DLL中的函数在公共语言运行库中不能直接被调用,而需要经过.Net框架提供的“平台调用”服务后才可以。 平台调用”是.Net框架为Visual Basic .Net、Visual C#等.Net开发语言提供的一种服务,用以在托管代码中引入各种非托管DLL中封装的函数(其中包括Windows API函数)。“平台调用”依赖于元数据在运行时查找导出函数并封装其参数。

托管DLL文件,可以在.NET环境通过 “添加引用” 的方式,直接把托管DLL文件添加到项目中。然后通过 Using DLL命名空间,来调用相应的DLL对象 。

非托管DLL文件,在.NET环境应用时,通过 DllImport 调用。

在托管代码中使用“平台调用”服务调用非托管DLL中封装的函数时,“平台服务”将依次执行以下操作:

1. 查找包含该函数所在的DLL文件。

2. 如果找到,则将该DLL文件 加载到内存中。

3. 查找函数在内存中的地址并将其参数推到堆栈上,并封送所需的数据。

4. 将控制权转移给非托管函数。 这样整个函数调用完成。


C#程序运行原理

同步块索引(synchronization block index):

同步块索引,又被叫做对象头指针,位于对象地址的-4到0字节(以32位机型为例)。

同步块索引在线程同步中用来判断对象是被使用还是闲置。

当有线程要对某个对象操作时,检查其同步块索引的值,若为表示闲置的值,便进行操作。

然后在CLR的同步块数组中增加一块新的同步块,把此块的索引值写入同步块的索引值里,表示其被占用。

线程用完后,便把索引值改回表示闲置的值。

方法表指针:

方法表指针指向类型对象。

类型对象是加载程序集时,在加载堆中创建的。

类型对象存储了静态字段和方法表,被所有该类型的实例共享。

方法表包括所有函数,包括静态函数和非静态函数。


堆栈(Stack)与堆(Heap)

栈是基于线程的,一个线程创建会分配一个默认1M大小的栈,由操作系统管理。

堆是基于进程的,一个进程分配一个堆。


进程与线程

进程:系统进行资源分配和调用的独立单位。

线程:CPU调度和分配的基本单位。

线程种类:

  • 主线程
  • 工作线程

工作线程种类:

  • Thread() 前台线程 | 只有前台线程执行完毕,应用程序才会结束。
  • ThreadPool(线程池) | 后台线程 | 不影响应用程序的终结。 | 后台线程不会影响应用程序的终结,当所有前台线程执行完毕后,后台线程无论是否执行完毕,都会被终结。一般后台线程用来做些无关紧要的任务(比如邮箱每隔一段时间就去检查下邮件,天气应用每隔一段时间去更新天气)。 | ThreadPool适用于并发运行若干个任务且运行时间不长且互不干扰的场景。

内存隔离:CLR给每个线程分配自己的内存栈。

一个线程当阻塞的时候,不占用CPU资源。

Mutex:互斥锁,同一事件只能有一个线程使用它 | 作用:用来控制一个应用程序只能运行一个实例。

Thread.Join(TimeSpan.FromHour(x) | timespan) 直到此线程执行结束,可以在指定时间内返回一个bool值。Task调用Wait(timespan)方法,可以阻塞任务,直至任务完成,效果等同于Thread.Join()。

Thread.Sleep(0) <==> Thread.Yield()

线程同步:lock (obj) { dosomething... }

    1. lock不能锁定空值 ,因为Null是不需要被释放的。
    1. 不能锁定string类型 ,虽然它也是引用类型的。因为字符串类型被CLR“暂留”,这意味着整个程序中任何给定字符串都只有一个实例,具有相同内容的字符串上放置了锁,就将锁定应用程序中该字符串的所有实例。
    1. 值类型不能被lock ,每次装箱后的对象都不一样 ,锁定时会报错 。
    1. 避免锁定public类型 如果该实例可以被公开访问,则 lock(this) 可能会有问题,因为不受控制的代码也可能会锁定该对象。
    1. 推荐使用 private static readonly类型的对象,readonly是为了避免lock的代码块中修改对象,造成对象改变后锁失效。

多线程与异步

异步是最终目的,多线程只是我们实现异步的一种手段。

多线程、并行、异步思维导图

三种异步模式:

  • 1,基于委托的异步模式(APM)(IAsyncResult) 不推荐
  • 2,基于事件的异步模式(EAP) 不推荐
  • 3,基于任务的异步模式(TAP) 推荐

Volatile的原理与使用

该关键字用来修饰字段,在很多语言中都有使用这个关键字,用来实现线程间数据同步,确保该字段任何时刻都呈现最新的值。 多线程同时访问普通的字段时,会在每个线程中创建一个该字段的变量拷贝,每个线程访问自己的memory,然后从自己cache中读取操作该变量。 多线程访问由Volatile修饰的字段时,则直接访问该字段所在memory,而不是从cache中读取,且该字段不受编译器优化,所以更消耗资源。