# [C#基础知识系列]全面解析C#中静态与非静态
## 一、引言
在C#中,静态和非静态的特征对于我们来说是再熟悉不过了,但是很少看到有一篇文章去好好地总结静态和非静态它们之间的不同,为了帮助大家更好地去理解静态和非静态特征, 所以将在这篇文章中帮大家全面总结下它们之间的不同,包括静态类,静态成员和静态构造函数。希望在大家巩固基础的时候可以拿出来好好复习下的。下面废话不多了,直接进入我们今天的主题。
## 二、为什么需要静态特征
在自定义类或看.NET Framework类库中都可以发现,类中大部分都是具体实例特征(也就是没有static标识的),同时我们也能看到一些具有静态特征的类或成员,例如我们经常使用的Console类以及WriteLine方法就是静态的。然而有些朋友会疑惑,为什么还要有静态特征的呢?干脆都定义为实例的好了? 然后静态特征的存在肯定有它存在的原因的,并不是我们就是要这么定义的,其实我一直认为不管是什么都是源于生活的, 技术的实现也是一样,比如我们开发程序,需要掌握技术外,其实更重要的是业务逻辑这块的,如果你都不知道你开发的东西是怎样的一个流程,即使你技术再牛做出来的东西都是反人类的东西(也就是指不符合用户的用户习惯和之前的一个业务需求),其实静态特征的存在也是源于生活的,对于类好比就是我们现实生活中的人或事物,静态特征和非静态特征就好比生活中人或事物具有的特征, 我们询问人的时候或者电视剧警察查案件的时候,都会听到这样一句话 "那个人有什么特征?"或 “嫌疑犯有什么特征?多高,年龄等” 其实高度、年龄、性别都是一个人的特征,所以这些在语言范畴就需要为其进行定义了,也就是我们定义的实例成员了,然而有些特征需要被所有对象实例所共有的,这些特征在语言范畴就定义为静态特征,具体哪些特征可以定义为静态特征呢? 其实这点一样是源于生活的,所以我们在开发软件的过程中,必不可少的一个流程就是需求分析了,只有在了解客户需求的条件下才能进行之后的所有流程的, 例如一个班级有很多学生,每个学生是一个实体,在语言范畴就可以定义一个类,当我们需要一个学生的时候就可以通过new 关键字创建一个出来(说到这里又让我想到了恶搞泰囧的图片——你有对象吗?没对象,你们程序员可以自己new一个啊?),然而**我们创建出来的学生他们都有一些共有的特征,如同一个班级,学校等, 如果我们把班级、学校这样的特征也定义为实例的话,那么我们不是每次创建对象实例的时候都为这些共有的特征分配一次内存的,这样不仅对内存空间的浪费也是不满足生活常识的,此时我们就可以把班级、学校这样的特征定义为静态特征,这样所有实例都可以共享这两个特征,并且不需要为每个对象实例分配内存**。
## 三、比较静态特征和非静态特征
## 3.1 静态类与非静态类
* 静态类和非静态类在C#中定义基本是一样的,只是静态类定义需要加上static修饰符而已。下面就直接总结下它们之间的区别:
* 静态类只能包含静态成员,否则会抛出编译错误;然而非静态类既可以包含非静态成员也可以包含静态成员
* 静态类是不能实例化,之所以不能实例化,是因为静态类会导致C#编译器将该类同时标记为abstract和sealed,并且编译器不会在类型中生成一个实例的构造函数,从而导致静态类不能实例化,具体原因可以见下图;非静态类可以,并且静态成员的访问只能通过类来进行访问,因为静态成员是属于类的。
上面代码用IL反汇编程序得到的IL代码结构为:
![](https://box.kancloud.cn/2016-01-23_56a2eb2eb9854.png)
## 3.2 静态构造函数与实例构造函数
* 静态构造函数用来初始化类中的静态成员的,包括静态字段和静态属性,并且静态构造函数是不能带有参数、不能有访问修饰符,静态构造函数的调用是由CLR第一次调用类成员之前执行的。
* 下面还是直接总结下静态构造函数与实例构造函数之间的区别:
* 静态构造函数可以与无参的实例构造函数同时存在
* 静态构造函数在CLR加载类时执行,然而实例构造函数在每次实例创建时都会执行
* 静态构造函数只能对静态成员初始化,不能对非静态成员进行初始化操作,然而实例构造函数,既可以初始化实例成员也可以初始化静态成员,但静态只读字段除外
* 静态构造函数只被执行一次,但是CLR也不能确定它什么时候被执行,它的执行方式有两种,precise和before-field-init,这个会在下一篇文章中详细给大家介绍,这里先提出给大家一个思考的空间。而实例构造函数在每次创建对象实例时都会被执行,创建几个就会执行几次
* 一个类只能有一个静态构造函数,却可以有多个实例构造函数
静态字段的初始值在静态构造函数调用之前被指定,构造函数的执行顺序大致如下图所示:
![](https://box.kancloud.cn/2016-01-23_56a2eb2ecb79f.png)
## 3.3 静态字段、属性和实例字段、属性
下面就直接总结下它们之间的区别:
* 静态成员包括静态属性和静态字段,静态字段一般实现为private,静态属性一般实现为public,从而来体现类的封装性
* 静态成员和类相关联,不依赖于对象而存在,只能由类来访问;实例成员与具体类相关联,只能由对象实例访问
* 静态成员不管创建多少实例对象,都在内存中只有一份,实例成员每创建一个实例对象,都会在内存中分配一块内存区域。
## 3.4 静态方法与实例方法
类似于静态字段和属性,静态方法共享代码段,同样以static关键字来标识静态方法,对于他们之间的区别总结为:
* 静态方法只能访问静态成员和方法,但是可以间接通过创建实例对象来访问实例字段、属性和方法;实例方法既可以访问实例成员也可以访问静态成员
* 静态方法由类方法‘实例方法由对象访问
* 静态方法不能引用this关键字,而实例方法可以
* 静态方法不能被标识为virtual、abstract或override,静态方法可以被派生访问,但是不能被派生类重写
* Main方法为静态的,所以Main方法不能直接访问类中的实例字段、属性和方法,否则编译器会报错
* 静态方法一般用于作为通用的工具类来实现
* 在性能上,静态方法和实例方法的差别不大。因为,它们都是在JIT加载类的时候分配内存的,不同的是静态方法是以类为引用,而实例方法是以对象为引用,创建实例时,不会再为静态方法分配内存,所有实例对象共用一个类的方法代码,所以,静态方法和实例方法的调用,区别仅在于静态方法可以直接调用,而实例方法需要当前对象指针指向该方法,在性能上差不并不大。
## 四、小结
到这里,本文章的内容就介绍完了,通过对静态特征和非静态特征的由来来揭开一些都是源于生活的观点,然后再详细分析了静态特征与非静态特征在C#语言中的区别,希望这些总结可以帮助大家在复习基础知识的时候可以有用。同时也是自己的一个复习笔记的。
- C# 基础知识系列
- C# 基础知识系列 专题一:深入解析委托——C#中为什么要引入委托
- C# 基础知识系列 专题二:委托的本质论
- C# 基础知识系列 专题三:如何用委托包装多个方法——委托链
- C# 基础知识系列 专题四:事件揭秘
- C# 基础知识系列 专题五:当点击按钮时触发Click事件背后发生的事情
- C# 基础知识系列 专题六:泛型基础篇——为什么引入泛型
- C# 基础知识系列 专题七: 泛型深入理解(一)
- C# 基础知识系列 专题八: 深入理解泛型(二)
- C# 基础知识系列 专题九: 深入理解泛型可变性
- C#基础知识系列 专题十:全面解析可空类型
- C# 基础知识系列 专题十一:匿名方法解析
- C#基础知识系列 专题十二:迭代器
- C#基础知识 专题十三:全面解析对象集合初始化器、匿名类型和隐式类型
- C# 基础知识系列 专题十四:深入理解Lambda表达式
- C# 基础知识系列 专题十五:全面解析扩展方法
- C# 基础知识系列 专题十六:Linq介绍
- C#基础知识系列 专题十七:深入理解动态类型
- 你必须知道的异步编程 C# 5.0 新特性——Async和Await使异步编程更简单
- 全面解析C#中参数传递
- C#基础知识系列 全面解析C#中静态与非静态
- C# 基础知识系列 C#中易混淆的知识点
- C#进阶系列
- C#进阶系列 专题一:深入解析深拷贝和浅拷贝
- C#进阶系列 专题二:你知道Dictionary查找速度为什么快吗?
- C# 开发技巧系列
- C# 开发技巧系列 使用C#操作Word和Excel程序
- C# 开发技巧系列 使用C#操作幻灯片
- C# 开发技巧系列 如何动态设置屏幕分辨率
- C# 开发技巧系列 C#如何实现图片查看器
- C# 开发技巧 如何防止程序多次运行
- C# 开发技巧 实现属于自己的截图工具
- C# 开发技巧 如何使不符合要求的元素等于离它最近的一个元素
- C# 线程处理系列
- C# 线程处理系列 专题一:线程基础
- C# 线程处理系列 专题二:线程池中的工作者线程
- C# 线程处理系列 专题三:线程池中的I/O线程
- C# 线程处理系列 专题四:线程同步
- C# 线程处理系列 专题五:线程同步——事件构造
- C# 线程处理系列 专题六:线程同步——信号量和互斥体
- C# 多线程处理系列专题七——对多线程的补充
- C#网络编程系列
- C# 网络编程系列 专题一:网络协议简介
- C# 网络编程系列 专题二:HTTP协议详解
- C# 网络编程系列 专题三:自定义Web服务器
- C# 网络编程系列 专题四:自定义Web浏览器
- C# 网络编程系列 专题五:TCP编程
- C# 网络编程系列 专题六:UDP编程
- C# 网络编程系列 专题七:UDP编程补充——UDP广播程序的实现
- C# 网络编程系列 专题八:P2P编程
- C# 网络编程系列 专题九:实现类似QQ的即时通信程序
- C# 网络编程系列 专题十:实现简单的邮件收发器
- C# 网络编程系列 专题十一:实现一个基于FTP协议的程序——文件上传下载器
- C# 网络编程系列 专题十二:实现一个简单的FTP服务器
- C# 互操作性入门系列
- C# 互操作性入门系列(一):C#中互操作性介绍
- C# 互操作性入门系列(二):使用平台调用调用Win32 函数
- C# 互操作性入门系列(三):平台调用中的数据封送处理
- C# 互操作性入门系列(四):在C# 中调用COM组件
- CLR
- 谈谈: String 和StringBuilder区别和选择
- 谈谈:程序集加载和反射
- 利用反射获得委托和事件以及创建委托实例和添加事件处理程序
- 谈谈:.Net中的序列化和反序列化
- C#设计模式
- UML类图符号 各种关系说明以及举例
- C#设计模式(1)——单例模式
- C#设计模式(2)——简单工厂模式
- C#设计模式(3)——工厂方法模式
- C#设计模式(4)——抽象工厂模式
- C#设计模式(5)——建造者模式(Builder Pattern)
- C#设计模式(6)——原型模式(Prototype Pattern)
- C#设计模式(7)——适配器模式(Adapter Pattern)
- C#设计模式(8)——桥接模式(Bridge Pattern)
- C#设计模式(9)——装饰者模式(Decorator Pattern)
- C#设计模式(10)——组合模式(Composite Pattern)
- C#设计模式(11)——外观模式(Facade Pattern)
- C#设计模式(12)——享元模式(Flyweight Pattern)
- C#设计模式(13)——代理模式(Proxy Pattern)
- C#设计模式(14)——模板方法模式(Template Method)
- C#设计模式(15)——命令模式(Command Pattern)
- C#设计模式(16)——迭代器模式(Iterator Pattern)
- C#设计模式(17)——观察者模式(Observer Pattern)
- C#设计模式(18)——中介者模式(Mediator Pattern)
- C#设计模式(19)——状态者模式(State Pattern)
- C#设计模式(20)——策略者模式(Stragety Pattern)
- C#设计模式(21)——责任链模式
- C#设计模式(22)——访问者模式(Vistor Pattern)
- C#设计模式(23)——备忘录模式(Memento Pattern)
- C#设计模式总结
- WPF快速入门系列
- WPF快速入门系列(1)——WPF布局概览
- WPF快速入门系列(2)——深入解析依赖属性
- WPF快速入门系列(3)——深入解析WPF事件机制
- WPF快速入门系列(4)——深入解析WPF绑定
- WPF快速入门系列(5)——深入解析WPF命令
- WPF快速入门系列(6)——WPF资源和样式
- WPF快速入门系列(7)——深入解析WPF模板
- WPF快速入门系列(8)——MVVM快速入门
- WPF快速入门系列(9)——WPF任务管理工具实现
- ASP.NET 开发
- ASP.NET 开发必备知识点(1):如何让Asp.net网站运行在自定义的Web服务器上
- ASP.NET 开发必备知识点(2):那些年追过的ASP.NET权限管理
- ASP.NET中实现回调
- 跟我一起学WCF
- 跟我一起学WCF(1)——MSMQ消息队列
- 跟我一起学WCF(2)——利用.NET Remoting技术开发分布式应用
- 跟我一起学WCF(3)——利用Web Services开发分布式应用
- 跟我一起学WCF(3)——利用Web Services开发分布式应用
- 跟我一起学WCF(4)——第一个WCF程序
- 跟我一起学WCF(5)——深入解析服务契约 上篇
- 跟我一起学WCF(6)——深入解析服务契约 下篇
- 跟我一起学WCF(7)——WCF数据契约与序列化详解
- 跟我一起学WCF(8)——WCF中Session、实例管理详解
- 跟我一起学WCF(9)——WCF回调操作的实现
- 跟我一起学WCF(10)——WCF中事务处理
- 跟我一起学WCF(11)——WCF中队列服务详解
- 跟我一起学WCF(12)——WCF中Rest服务入门
- 跟我一起学WCF(13)——WCF系列总结
- .NET领域驱动设计实战系列
- .NET领域驱动设计实战系列 专题一:前期准备之EF CodeFirst
- .NET领域驱动设计实战系列 专题二:结合领域驱动设计的面向服务架构来搭建网上书店
- .NET领域驱动设计实战系列 专题三:前期准备之规约模式(Specification Pattern)
- .NET领域驱动设计实战系列 专题四:前期准备之工作单元模式(Unit Of Work)
- .NET领域驱动设计实战系列 专题五:网上书店规约模式、工作单元模式的引入以及购物车的实现
- .NET领域驱动设计实战系列 专题六:DDD实践案例:网上书店订单功能的实现
- .NET领域驱动设计实战系列 专题七:DDD实践案例:引入事件驱动与中间件机制来实现后台管理功能
- .NET领域驱动设计实战系列 专题八:DDD案例:网上书店分布式消息队列和分布式缓存的实现
- .NET领域驱动设计实战系列 专题九:DDD案例:网上书店AOP和站点地图的实现
- .NET领域驱动设计实战系列 专题十:DDD扩展内容:全面剖析CQRS模式实现
- .NET领域驱动设计实战系列 专题十一:.NET 领域驱动设计实战系列总结