面向对象编程一个好处就是“代码重用”,极大提高了开发效率。如是,可以派生出一个类,让它继承基类的所有能力,派生类只需要重写虚方法,或添加一些新的方法,就可以定制派生类的行为,使之满足开发人员的需求。
泛型(generic)是CLR和编程语言提供的一种特殊机制,它支持另一种形式的代码重用,即“算法重用”。
 
简单地说,开发人员先定义好一个算法,比如排序、搜索、交换、比较或转换等。但是,定义算法的开发人员并不设定该算法要操作什么数据类型;该算法可以广泛地应用于不同类型的对象。然后,另一个开发人员只要指定了算法要操作的具体数据类型,就可以使用这个现成的算法了。例如,可用一个排序算法来操作Int32和String等类型的对象,或用一个比较算法来操作DateTime和Version等类型的对象。大多数算法都封装在一个类型中,CLR允许创建泛型引用类型和泛型值类型,但不允许创建泛型枚举类型。还允许创建泛型接口和泛型委托。
 
泛型为开发人员提供了一下优势:
#1,源代码保护。
#2,类型安全。编译器和CLR能理解开发人员的意图,并保证只有与指定数据类型兼容的对象才能随同算法使用。
#3,更加清晰的代码。减少了源代码中必须进行的转型次数,代码更容易编写和维护。
#4,更佳的性能。创建泛型算法来操作一种具体的值类型,所以值类型的实例能够以传值方式传递,CLR不再需要执行任何装箱操作。ArrayList来操作值类型(如Int32),会造成大量装箱操作,大量的垃圾回收。
泛型最明显的应用就是集合类。FCL已经定义了几个泛型集合类,其中,大多数类都在System.Collections.Generic 和System.Collections.ObjectModel命名空间中。
 
泛型基础结构
开发类型和封闭类型
我们关注CLR如何为应用程序使用的每个类型创建一个怎样的内部数据结构,这种数据结构称为类型对象(type object)。具有泛型类型参数的类型仍然是类型,CLR同样会为它创建一个内部类型对象。无论引用类型(类)、值类型(结构)、接口类型,还是委托类型,这一点都成立。具有泛型类型参数的类型称为开放类型(open type), CLR禁止构造开发类型的任何实例,类似CLR禁止构造接口类型的实例, 如List<>。
 
代码引用一个泛型类型时,可指定一组泛型类型实参。假如为所有类型实参传递的都是实际数据类型,类型就称为封闭类型(closed type),CLR允许构造封闭类型的实例,如List<String> 。需要注意的是,CLR会在类型对象内部分配类型的静态字段,因此每个封闭类型都有自己的静态字段。例如,List<T>定义了任何静态字段,都不会在一个List<DateTime>和一个List<String>之间共享:每个封闭类型对象都有它自己的静态字段。假如一个泛型类型定义了一个静态构造器,那么针对每个封闭类型,这个构造器都会执行一次。在泛型类型上定义一个静态构造器的目的是保证传递的类型实参满足特定的条件(约束)。如,希望一个泛型类型只用于处理枚举类型,就可以如下定义:
internal sealed class GenericTypeThatRequiresAnEnum<T> {
     static GenericTypeThatRequiresAnEnum() {
          if(!typeof(T).IsEnum) {
               throw new ArgumentException("T must be an enmuerated type");
          }
     }
}
 
CLR提供了一个名为约束的功能,可利用它更好地定义一个泛型类型来支出哪些类型实参是有效的。
 
代码爆炸
 
使用泛型类型参数的一个方法在进行JIT编译时,CLR获取方法的IL,用指定的类型实参进行替换,然后创建恰当的本地代码(这些代码是为操作指定数据类型的方法“量身定制”的)。这样做有一个缺点:CLR要为每一种不同的方法/类型组合生成本地代码。这个现象称为代码爆炸(code explosion)。它可能造成应用程序的工作集显著增大,从而损害性能。
 
CLR其实内建了一些优化措施,能够缓解代码爆炸。假如为一个特定的类型实参调用一个方法,以后再次使用相同的类型实参来调用这个方法。CLR只会为这个方法/类型组合编译一次代码。如果一个程序集使用List<DateTime>, 一个完全不同的程序集(加载到一个AppDomain中)也使用List<DateTime>, CLR只会为List<DateTime>编译一次方法。这样显著缓解了代码爆炸。
CLR还提供了另一个优化措施,它认为所有引用类型实参都是完全相同的,所以代码可以共享。如,CLR为List<String>的方法编译的代码可直接用于List<Stream>的方法,因为String和Stream均为引用类型。事实上,对于任何引用类型,都会使用相同的代码。CLR之所以能执行这个优化,是因为所有引用类型的实参或变量实际只是指向堆上的对象的指针,而对象指针全部是以相同的方式来操作的。
但是,某个类型实参是值类型,CLR就必须专门为这个值类型生成本地代码,因为值类型的大小不定,还可能要用不同的本地CPU指令来操纵这些值。
 
委托和接口的逆变和协变泛型类型实参
 
 

CLR via C#深解笔记六 - 泛型的更多相关文章

  1. CLR via C#深解笔记四 - 方法、参数、属性

    实例构造器和类(引用类型) 构造器(constructor)是允许将类型的实例初始化为良好状态的一种特殊方法.构造器方法在“方法定义元数据表”中始终叫.ctor. 创建一个引用类型的实例时: #1, ...

  2. CLR via C#深解笔记三 - 基元类型、引用类型和值类型 | 类型和成员基础 | 常量和字段

    编程语言的基元类型   某些数据类型如此常用,以至于许多编译器允许代码以简化的语法来操纵它们. System.Int32 a = new System.Int32();  // a = 0 a = 1 ...

  3. CLR via C#深解笔记七 - 自动内存管理(垃圾回收)

    每个应用程序都要使用这样或者那样的资源,比如文件.内存缓冲区.屏幕空间.网络连接.数据库资源等.事实上,在面向对象的环境中,每个类型都代表可供程序使用的一种资源. 要使用这些资源,必须为代表资源的类型 ...

  4. CLR via C#深解笔记五 - 事件

    事件处理实际上是一种具有特殊签名的delegate, 像这个样子:public delegate void EventHandler(object sender, EventArgs e);   类型 ...

  5. CLR via C#深解笔记二 - 类型设计

    类型基础 所有类型都从System.Object派生   CLR要求所有对象都用new 操作符来创建. Employee e = new Employee("Constructor Para ...

  6. CLR via C#深解笔记一 - CLR & C# 基础概念

    写在前言   .Net Framework并不是Win 32 API 和COM上的一个抽象层.   某种程度上,它是自己的操作系统,有自己的内存管理器,自己的安全系统,自己的文件加载器,自己的错误处理 ...

  7. 《CLR via C#》读书笔记 之 泛型

    第十二章 泛型 2014-06-15 初始泛型 12.3 泛型基础结构 12.3.1 开放类型与封闭类型 12.3.2 泛型类型和继承 12.3.3 泛型类型同一性 12.3.4 代码爆炸 12.6 ...

  8. Java泛型学习笔记 - (六)泛型的继承

    在学习继承的时候, 我们已经知道可以将一个子类的对象赋值给其父类的对象, 也就是父类引用指向子类对象, 如: Object obj = new Integer(10); 这其实就是面向对象编程中的is ...

  9. java之jvm学习笔记六-十二(实践写自己的安全管理器)(jar包的代码认证和签名) (实践对jar包的代码签名) (策略文件)(策略和保护域) (访问控制器) (访问控制器的栈校验机制) (jvm基本结构)

    java之jvm学习笔记六(实践写自己的安全管理器) 安全管理器SecurityManager里设计的内容实在是非常的庞大,它的核心方法就是checkPerssiom这个方法里又调用 AccessCo ...

随机推荐

  1. [Java Web] 6、Tomcat服务器的安装及配置以及JSP技术笔记

    目录  1.Web容器简介  2.Tomcat粗介及配置粗讲  3.Tomcat服务器配置 3-1.修改端口号  3-2.配置虚拟目录 3-3.配置首页  4.JSP执行流程  5.JSP粗略了解 1 ...

  2. 点击div外面该div消失

    <head> <meta charset="UTF-8"> <title></title> <script type=&quo ...

  3. AT&T Assembly for Linux and Mac (sys_write)

    Write() in C : (sys_write.c) #include <stdio.h> int main(void) { printf("Hello Landpack\n ...

  4. atitit.GMT UTC Catitit.GMT UTC CST DST CET 星期 月份 节日 时间的不同本质and起源

    atitit.GMT UTC Catitit.GMT UTC CST DST CET 星期 月份 节日 时间的不同本质and起源 1. GMT(Greenwich Mean Time)是格林尼治平时 ...

  5. unity 读取excel表 生成asset资源文件

    做unity 项目也有一段时间了,从unity项目开发和学习中也遇到了很多坑,并且也从中学习到了很多曾经未接触的领域.项目中的很多功能模块,从今天开始把自己的思路和代码奉上给学渣们作为一份学习的资料. ...

  6. JSP内置对象总结

    前几天学习了javaee中的jsp(Java Server Pages),即java服务器页面,其实就是在html里面写java代码. 一.概述前言 在总结九大对象之前,有必要先搞清楚几个概念:请求, ...

  7. ListView、PullToRefreshListView滑动加载可见item

    最近用的了PullToRefreshListView框架,也在listView中加载图片,对于滑动加载可见item,网上找了一些相关文档,但都不太合适,如:http://blog.csdn.net/z ...

  8. 数据采集:完美下载淘宝Ip数据库 简单的程序节省60元人民币而不必购买数据库

    曾经做网站类型的程序时,经常需要收集客户端的访问数据,然后加以分析.这需要一个Ip数据库,数据表中显示Ip所在的省份市区等信息.网络上有流传的Ip纯真数据库,一些公开的Web服务也可以查询Ip地址信息 ...

  9. android: SQLite查询数据

    掌握了查询数据的方法之后,你也就将数据库的 CRUD 操 作全部学完了.不过千万不要因此而放松,因为查询数据也是在 CRUD 中最复杂的一种 操作. 我们都知道 SQL 的全称是 Structured ...

  10. java综合

    java环境变量配置:http://jingyan.baidu.com/article/f96699bb8b38e0894e3c1bef.html 1.导入项目 当下载了包含Eclipse 项目的源代 ...