泛型是什么?

通过上篇的实例  C# 泛型约束 xxx<T> Where T:约束(一),我们对泛型有一定的认识。

所谓泛型,即通过参数化类型来实现在同一份代码上操作多种数据类型,泛型编程是一种编程范式,它利用“参数化类型”将类型抽象化,从而实现更为灵活的复用。

在定义泛型类时,可以对代码能够在实例化类时用于类型参数的类型种类施加限制。如果代码尝试使用某个约束所不允许的类型来实例化类,则会产生编译时错误。这些限制称为约束。约束是使用 where 上下文关键字指定的。

五种类型的约束

下表列出了五种类型的约束:

约束 说明
T:struct 类型参数必须是值类型。可以指定除 Nullable 以外的任何值类型。
T:class 类型参数必须是引用类型,包括任何类、接口、委托或数组类型。
T:new () 类型参数必须具有无参数的公共构造函数。当与其他约束一起使用时,new() 约束必须最后指定。
T:<基类名> 类型参数必须是指定的基类或派生自指定的基类。
T:<接口名称> 类型参数必须是指定的接口或实现指定的接口。可以指定多个接口约束。约束接口也可以是泛型的。
T:U 为 T 提供的类型参数必须是为 U 提供的参数或派生自为 U 提供的参数。这称为裸类型约束.

1. 派生约束

1.常见的

public class MyClass5<T> where T :IComparable { }

2.约束放在类的实际派生之后

public class B { }

public class MyClass6<T> : B where T : IComparable { }

3.可以继承一个基类和多个接口,且基类在接口前面

public class B { }

public class MyClass7<T> where T : B, IComparable, ICloneable { }

2. 构造函数约束

1.常见的

public class MyClass8<T> where T :  new() { }

2.约束组合

可以将构造函数约束和派生约束组合起来,前提是构造函数约束出现在约束列表的最后

public class MyClass8<T> where T : IComparable, new() { }

3. 值约束

1.常见的

public class MyClass9<T> where T : struct { }

2. 接口约束同时使用

与接口约束同时使用,在最前面(不能与基类约束,构造函数约束一起使用)

public class MyClass11<T> where T : struct, IComparable { }

4. 引用约束

1.常见的

public class MyClass10<T> where T : class { }

5. 多个泛型参数

public class MyClass12<T, U> where T : IComparable  where U : class { }

6. 继承和泛型

public class B<T>{ }

1.类型实参

1. 在从泛型基类派生时,可以提供类型实参,而不是基类泛型参数

public class SubClass11 : B<int>{ }

2.子类泛型作为基类泛型的指定类型

2.如果子类是泛型,而非具体的类型实参,则可以使用子类泛型参数作为泛型基类的指定类型

public class SubClass12<R> : B<R>{ }

3.子类重复基类的约束

3.在子类重复基类的约束(在使用子类泛型参数时,必须在子类级别重复在基类级别规定的任何约束)
    public class B<T> where T : ISomeInterface { }
    public class SubClass2<T> : B<T> where T : ISomeInterface { }

4.构造函数约束

public class B<T> where T : new()
{
public T SomeMethod()
{
return new T();
}
}
public class SubClass3<T> : B<T> where T : new() { }

7. 泛型方法

C#2.0泛型机制支持在"方法声名上包含类型参数",这就是泛型方法)

1.泛型类型/非泛型类型

泛型方法既可以包含在泛型类型中,又可以包含在非泛型类型中

public class MyClass5
{
public void MyMethod<T>(T t){ }
}

2.泛型方法的声明与调用

public class MyClass5
{
public void MyMethod<T>(T t)
{
}
} public class App5
{
public void CallMethod()
{
MyClass5 myclass5 = new MyClass5();
myclass5.MyMethod<int>();
}
}

3.泛型方法的重载

//第一组重载
void MyMethod1<T>(T t, int i) { } void MyMethod1<U>(U u, int i) { } //第二组重载
void MyMethod2<T>(int i) { }
void MyMethod2(int i) { } //第三组重载,假设有两个泛型参数
void MyMethod3<T>(T t) where T : A { }
void MyMethod3<T>(T t) where T : B { } //第四组重载 public class MyClass8<T, U>
{
public T MyMothed(T a, U b)
{
return a;
}
public T MyMothed(U a, T b)
{
return b;
}
public int MyMothed(int a, int b)
{
return a + b;
}
}

4.泛型方法的覆写

public class MyBaseClass1
{
public virtual void MyMothed<T>(T t) where T : new() { }
}
public class MySubClass1 : MyBaseClass1
{
//不能重复任何约束
public override void MyMothed<T>(T t) { }
} public class MyBaseClass2
{
public virtual void MyMothed<T>(T t){ }
}
public class MySubClass2 : MyBaseClass2
{
//重新定义泛型参数T
public override void MyMothed<T>(T t){ }
}

8. 虚拟方法

public class BaseClass4<T>
{
public virtual T SomeMethod()
{
return default(T);
}
}
public class SubClass4 : BaseClass4<int> //使用实参继承的时候方法要使用实参的类型
{
public override int SomeMethod()
{
return ;
}
} public class SubClass5<T> : BaseClass4<T> //使用泛型继承时,方法也是泛型
{
public override T SomeMethod()
{
return default(T);
}
}

9. 泛型参数强制转换到Object或约束指定的类型

编译器只允许将泛型参数隐式强制转换到 Object 或约束指定的类型

class MyClass<T> where T : BaseClass, ISomeInterface
{
void SomeMethod(T t)
{
ISomeInterface obj1 = t;
BaseClass obj2 = t;
object obj3 = t;
}
} //变通方法:使用临时的 Object 变量,将泛型参数强制转换到其他任何类型 class MyClass2<T>
{
void SomeMethod(T t)
{
object temp = t;
BaseClass obj = (BaseClass)temp;
}
}

10. 泛型参数强制转换到其他任何接口

编译器允许您将泛型参数显式强制转换到其他任何接口,但不能将其转换到类

class MyClass1<T>
{
void SomeMethod(T t)
{
ISomeInterface obj1 = (ISomeInterface)t;
//BaseClass obj2 = (BaseClass)t; //不能通过编译
}
}

11. 泛型参数强制转换到其他任何类型

使用临时的 Object 变量,将泛型参数强制转换到其他任何类型

class MyClass2<T>
{
void SomeMethod(T t)
{
object temp = t;
BaseClass obj = (BaseClass)temp;
}
}

12. 使用is和as运算符

public class MyClass3<T>
{
public void SomeMethod(T t)
{
if (t is int) { }
if (t is LinkedList<int>) { }
string str = t as string;
if (str != null) { }
LinkedList<int> list = t as LinkedList<int>;
if (list != null) { }
}
}

参考资料

MSDN:http://msdn.microsoft.com/zh-cn/library/d5x73970.aspx

部分内容参考:http://www.cnblogs.com/andrew-blog/archive/2012/03/21/ListT_Where.html

C# 泛型约束 xxx Where T:约束(二)的更多相关文章

  1. 泛型约束:接口约束、new()构造函数约束、组合约束(即多种约束合并)

    泛型 接口约束: using System; namespace ConsoleApp1 { /* * * 接口约束:where T:interfaceName * T是类型形参的名称,interfa ...

  2. c# 泛型<T>类型参数T的约束where

    在定义泛型类时,可以对客户端代码能够在实例化类时用于类型参数的类型种类施加限制.如果客户端代码尝试使用某个约束所不允许的类型来实例化类,则会产生编译时错误.这些限制称为约束.约束是使用 where 上 ...

  3. SQL约束(主键约束、外键约束、自动递增、不允许空值、值唯一、值默认、值限制范围)

    NOT NULL 不允许空值约束 NOT NULL 约束强制列不接受 NULL 值(NULL值就是没有值或缺值).NOT NULL 约束强制字段始终包含值,即不向字段添加值,就无法插入新记录或者更新记 ...

  4. java 约束配置文件和本地约束

    一.寻找spring配置文件约束头(也可直接复制已有的) 1.在本地文件夹解压spring核心包(dist) 例:核心包的约束位置(D:\JavaSources\spring-framework-4. ...

  5. 约束Constraints--主键约束、外键约束、唯一约束、检查约束、默认约束、NOT NULL约束、列约束与表约束、创建约束、删除约束

    约束   Including Constraints 以下内容转自:https://www.cnblogs.com/wcl2017/p/7043939.html和http://blog.csdn.ne ...

  6. Sql Server约束的学习一(主键约束、外键约束、唯一约束)

    一.约束的分类 1.实体约束 实体约束是关于行的,比如某一行出现的值不允许出现在其他行,例如主键约束. 2.域约束 域约束是关于列的,对于所有行,某一列有那些约束,例如检查约束. 3.参照完整性约束 ...

  7. 约束4:唯一约束,Check约束和null

    大家知道,关系型数据库的逻辑运算的结果是三值型的,TRUE,FALSE和UNKNOWN,特别是,NULL值和任何值都不相等,任何值和NULL的比较,返回的逻辑结果都是unknown.而NULL值在唯一 ...

  8. 约束3:default约束

    默认值约束(Default约束)的作用是在执行insert命令时,如果命令没有显式给指定的列赋值,那么把默认约束值插入到该列中:如果在Insert命令中显式为指定的列赋值,那么将该列插入用户显式指定的 ...

  9. MySQL创建表时加入的约束以及外键约束的的意义

    1,创建表时加入的约束 a) 非空约束,not null b) 唯一约束,unique c) 主键约束,primary key d) 外键约束,foreign key 1,非空约束,针对某个字段设置其 ...

随机推荐

  1. java-jmx使用

    先粘一段内容 .程序初哥一般是写死在程序中,到要改变的时候就去修改代码,然后重新编译发布. .程序熟手则配置在文件中(JAVA一般都是properties文件),到要改变的时候只要修改配置文件,但还是 ...

  2. vue-cli中全局组件的注册使用

    一.全局注册 在install函数中全局注册组件,没毛病,老铁. 二.其它组件调用 直接在其他  .vue组件中直接写 <popup ref="popup">,然后就可 ...

  3. C# Quartz的配置

    1. 介绍 Quartz为后台工作者提供了得便利,我们下面介绍一下它的配置.本文配置主要针对服务程序的配置. 但是在做下面配置之前,要安装包 Install-Package Quartz 2. Qua ...

  4. DataGridview启用列重新排序属性的作用

    DataGridview是winform中经常用的控件,今天来了解一下启用列重新排序属性的作用 默认没有选中是不能改变列前后顺序的 启用列重新排序后如下图可以拖动列标题(列标头)来改变列前后顺序

  5. //{{AFX_MSG、//{{AFX_VIRTUAL、//{{AFX_MSG_MAP、//{{AFX_DATA_INIT

    说明:这篇日志我不知道怎么命名好,虽然内容很少,但是讲的关键字很多,如果你有幸打开这篇日志,不妨往下看看 背景:我们使用 VC++6.0 开发MFC应用程序,初学者一定会为那么多行的注释代码感到头痛, ...

  6. bootstrap-datepicker 开始时间-结束时间 thinkphp

    <!DOCTYPE html> <head> <title>开始-结束时间测试</title> </head> <body> & ...

  7. 修改VS类模板自动添加public修饰符和版权注释信息

    在开发过程中,我们经常需要给类或接口添加public修饰符(默认没有)和一些相关的注释信息,这个工作是机械而枯燥的,而这个简单的需求其实是可以通过修改VS自带的类模板来实现的,下面是详细的修改步骤. ...

  8. POJ3304(KB13-C 计算几何)

    Segments Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 15335   Accepted: 4862 Descrip ...

  9. 使用ThinkPHP实现分页功能

    前几篇(上传,缩略图,验证码,自动验证表单)文章介绍的功能实现都是基于ThinkPHP框架封装好的类进行实现的,所以这次自己写一个分页类在框架中使用. 首先在根目录建一个Tools文件夹,在Tools ...

  10. 微信小程序 折叠效果

    <view class='help'> <view class='help_item'> <view class='title' data-index='1' catch ...