1、泛型类的功能:在创建泛型类时,还需要一些其他C#关键字。例如,不能把null赋予泛型类型。此时,可以使用default 关键字。如果泛型类型不需要Object类的功能,但需要调用泛型类上的某特定方法,就可以定义约束。

 具体讨论以下:

  • 默认值
  • 约束
  • 继承
  • 静态成员

例:下面是一个使用 泛型文档管理器示例。DocumentManager

  •  using System;
    using System.Collections.Generic;
    namespace Jandar.Test.YinXi{
    public class DocumentManager<T>{
    private readonly Queue<T> documentQueue=new Queue<T>();
    public void AddDocument(T doc){
    lock(this){
    documentQueue.Enqueue(doc);
    }
    }
    public bool IsDocumentAvailable{
    get{
    return documentQueue.Count>;
    }
    }
    }

除了IEnumerable 接口,还有一个泛型版本 IEnumerable<T>。IEnumerable<T> 派生自IEnumerable 添加了IEnumerator<T> 的GetEnumerator() 方法,LinkeedList<T>实现泛型接口 IEnumerable<T>

  【默认值】

    给DocumentManager<T>类添加一个 GetDocument() 方法。在这个方法中,应把 类型T 指定为null。但是,不能把null 赋予泛型类型。原因是泛型类型也可以实例化为值类型,而null 只能用于引用类型。为了解决这个问题,可以使用 default 关键字。通过default 关键字,将null 赋予引用类型,将 0 赋予值类型。

  •   

     public T GetDocument()
    {
    T doc =default(T);
    lock(this)
    {
    doc=documentQueue.Dequeue();
    }
    return doc;
    }

    default 关键字根据上下文可以有多种含义。 switch 语句使用 default 定义默认情况。在泛型中,根据泛型类型是引用类型还是值类型,泛型default 用于将泛型类型初始化为 null 或 0;

  【约束】

  • Document 类实现带有 Title 和Content属性的 IDocument 接口。

     public interface IDocument
    {
    string Title {get; set;}
    string Content{get; set;}
    } public class Document: IDocument
    {
    public Document()
    { }
    public Document(string title,string content)
    {
    this.Title=title;
    this.Content=content;
    }
    public string Title{ get; set;}
    public string Content{getl;set;}
    }
  • 要使用DocumentManager<T> 类显示文档,可以将类型 T 强制转换为 IDocument 接口,以显示标题 title 或者 内容 content
  •  public void DisplayAllDocuments()
    {
    foreach(T doc in documentQueue)
    {
    Console.WriteLine(((IDocument)doc).Title);
    }
    }
  • 问题是如果类型T没有实现IDocument 接口,这个类型强制转换就会导致一个运行异常。最好给DocumentManager<TDocument>类定义一个约束: TDocument 类型必须实现 IDocument 接口,
  •  public class DocumentManager<TDocument> where TDocument : IDocument
    {
    ...
    }
    约束 说明

    where T : struct

    对于结构约束,类型 T 必须是值类型

    where T : class

    类约束指定类型 T 必须是引用类型

    where T : IFoo

    指定类型 T 必须实现接口 IFoo

    where T : Foo

    指定类型 T 必须派生自基类 Foo

    where T : new()

    这是一个构造函数约束,指定类型 T 必须有一个默认构造函数

    where T1 : T2

    这个约束也可以指定,类型T1 派生自 泛型类型T2。 该约束也成为裸类型约束

    只能为默认构造函数定义构造函数约束,不能为其他构造函数定义构造函数约束

    使用泛型类型还可以合并多个约束。 where T :IFoo,new() 约束和 MyClass<T>声明指定,类型T 必须实现IFoo 接口,且必须有一个默认构造函数。

    •       

      public class MyClass<T> where T :IFoo, new()
      {
      ...
      }

   【继承】

   可以定义一个抽象的泛型基类,它在派生类中用一个具体的类型实现。

    •       

       public abstract class Calc<T>
      {
      public abstract T Add(T x, T y);
      public abstract T Sub(T x, T y);
      } public class IntCalc :Calc<int>
      {
      public override int Add(int x ,int y)
      {
      return x+y;
      }
      public override int Sub(int x, int y)
      {
      return x-y;
      }
      }

    【静态成员】

    泛型类的静态成员需要特别关注。泛型类的静态成员只能在类的一个实例中共享。

  •     

     public class StaticDemo<T>
    {
    public static int x;
    } //由于同时对一个string类型和一个int 类型使用了 StaticDemo<T> 类。所以存在两组静态字段。
    StaticDemo<string>.x=;
    StaticDemo<int>.x=;
    Console.WriteLine(StaticDemo<string>.x);//

2、泛型接口

  【协变和抗变】

    协变和抗变指对参数和返回值类型进行转换。

    在.NET 中,参数类型是协变的。假定有 Shape 和 Rectangle 类,Rectangle 派生自Shape基类。声明 Display()方法是为了接受Shape 类型的对象作为其参数:    

public void Display(Shape o){ }

    方法的返回类型是抗变的。当方法返回一个Shape 时,不能把它赋予 Rectangle, 因为Shape不一定总是Rectangle。反过来是可行的:如果一个方法像 GetRectangle() 方法那样返回一个Rectangle,

public Rectangle GetRectangle();

就可以把结果赋予某个Shape

Shape s=GetRectangle();

在 .NET Framework4 版本之前,这种行为方式不适用于泛型。 在C# 4 中,扩展后的语言支持泛型接口和泛型委托的协变和抗变。

  泛型接口的协变

  如果泛型类型用 out 关键字标注,泛型接口就是协变的。这也意味着返回类型只能是 T 。 接口 IIndex 与 类型 T 是协变的,并从一个只读索引器中返回这个类型。

public interface IIndex<out T>
{
T this[int index] { get;}
int Count{get;}
}

C#高级编程笔记 Day 6, 2016年9月 14日 (泛型)的更多相关文章

  1. 2016年12月14日 星期三 --出埃及记 Exodus 21:9

    2016年12月14日 星期三 --出埃及记 Exodus 21:9 If he selects her for his son, he must grant her the rights of a ...

  2. 2016年11月14日 星期一 --出埃及记 Exodus 20:5

    2016年11月14日 星期一 --出埃及记 Exodus 20:5 You shall not bow down to them or worship them; for I, the LORD y ...

  3. 2016年10月14日 星期五 --出埃及记 Exodus 18:25

    2016年10月14日 星期五 --出埃及记 Exodus 18:25 He chose capable men from all Israel and made them leaders of th ...

  4. 关于苹果开发证书失效的解决方式(2016年2月14日Failed to locate or generate matching signing assets)

    前言: 从2月14日開始,上传程序的同学可能会遇到提示上传失败的提示. 而且打开自己的钥匙串,发现所有的证书所有都显示此证书签发者无效. Failed to locate or generate ma ...

  5. 2016年11月14日--SQL创建数据库、表-查、插、删、改

    --创建数据库(create database 数据库名)create database hq20161114go --使用选择数据库(use 数据库名)use hq20161114go --创建学生 ...

  6. Android高级编程笔记(四)深入探讨Activity(转)

    在应用程序中至少包含一个用来处理应用程序的主UI功能的主界面屏幕.这个主界面一般由多个Fragment组成,并由一组次要Activity支持.要在屏幕之间切换,就必须要启动一个新的Activity.一 ...

  7. 我的Python成长之路---第八天---Python基础(24)---2016年3月5日(晴)

    多线程编程 什么是多线程,线程是操作系统能够进行运算调度的最小单位.他包含在进程之中,是进程中的实际运作单位.线程是进程中一个单顺序的空值六,一个进程可以并发多个线程,每个线程可以并行处理不同的任务. ...

  8. 我的Python成长之路---第七天---Python基础(22)---2016年2月27日(晴)

    socket网络编程 socket通常也称作"套接字",用于描述IP地址和端口,是一个通信链的句柄,应用程序通常通过"套接字"向网络发出请求或者应答网络请求. ...

  9. 2016年2月16日开始,每天一篇,记录学习心得,【基本技能篇】>>开篇《如何阅读一本书——心得》

    如何阅读一本书——心得 ——2016年2月12日 要达到阅读的所有目的,就必须在阅读不同书籍的时候,运用适当的不同速度.读的太快或太慢,都一无所获. 四个阅读层次:①基础阅读,具有基本阅读的能力,包括 ...

随机推荐

  1. YII2如何修改默认控制器/方法

    在网上找了非常多的方法,但是都不好使最后自己综合网上所有自己琢磨出来的,见笑了 首先Yii2中在/vendor/yiisoft/yii2/web/Application.php的第28行 public ...

  2. vs2015 安装之后安装MSSM 2016 导致 vs启动报错 System.ArgumentException 已添加了具有相同键的项,ActivityLog.xml

    如题,先是装了vs2015,开发什么的都没有问题,后来安装了SqlServer2016 MSSM,出大问题了,vs2015打开就报错,具体错误如上,还想还有个ActivityLog.xml 这个文件的 ...

  3. 梦想成真,喜获微软MVP奖项,微软MVP FAQ?

    之前一直很钦佩那些MVP获奖者,想着自己有一天也能拿到该多好,就在10月1日邮箱收到了微软的邮件,当选了2016年10月份的MVP.今天主要分享一下获奖的喜悦也分享一下如何获得MVP奖项. 什么是微软 ...

  4. SQL Server 常用关键字

    SQL 建库 建表 --1.创建一个数据库 create database School; --删除数据库 drop database School; --创建数据库的时候指定一些选项. create ...

  5. 【Chrome】新建Chrome插件,新建,事件行为,本地存储

    源:walkingq 1,新建:静态网站+manifest.json:[注:mainifest_version是chrome 18以后新增key] manifest.json 截图 2,事件行为: 3 ...

  6. [转]struts2处理.do后缀的请求

    原文地址:http://skyuck.iteye.com/blog/545988 默认情况下,struts2是无法处理以.do为后缀的请求url的(默认情况下是.action或者不填,可以参见org. ...

  7. cocopod 中添加第三方框架,包含静态库文件,使用svn添加上传

    step one: 进入静态库文件的目录 cd 路径: step two:使用命令添加 svn add 静态库名字; 然后更新一下代码就OK

  8. UML类图详解

    下面是类图的实例: (注:飞翔接口那里应为空心三角形) UML中类图实例 接口:空心圆+直线(唐老鸭类实现了‘讲人话’):依赖:虚线+箭头(动物和空气的关系):关联:实线+箭头(企鹅需要知道气候才迁移 ...

  9. Python3 基本数据类型

    Python中的变量不需要声明,每个变量使用前必须赋值,变量赋值后才会被创建,在Python中变量就是变量,它没有类型.我们所说的"类型"是变量所指的内存中对象的类型. 等号(=) ...

  10. 数据结构图文解析之:树的简介及二叉排序树C++模板实现.

    0. 数据结构图文解析系列 数据结构系列文章 数据结构图文解析之:数组.单链表.双链表介绍及C++模板实现 数据结构图文解析之:栈的简介及C++模板实现 数据结构图文解析之:队列详解与C++模板实现 ...