.NET基础之自定义泛型
在.NET中泛型使用非常频繁,在控制台应用程序中,默认的引入了System.Collection.Generics名称空间,其中就提供了我们经常使用的泛型:List<T>和Dictionary<T>,相信用过它们的都知道它们的强大。还有一种我们经常使用的简单的泛型:System.Nullable<T>,即可空类型。我们可以:
System.Nullable<int> nullableInt;
声明一个可空的int类型,由于C#语法对这个做了简化通常我们都不这样写,而是这样写:
int? nullableInt
下面重点介绍一下如何自定义泛型。
定义泛型类
创建泛型类是需要在类定义中用尖括号语法:
class MyGenericClass<T>
{
...
}
T可以是任意的标示符,只要遵守命名规则即可。
可以把类型用在类成员的返回类型,方法参数类型等,例如:
class MyGenericClass<T1, T2, T3>
{
private T1 t1Object; public MyGenericClass(T1 item)
{
t1Object = item;
} public T1 T1Object
{
get
{
return t1Object;
}
}
}
注意如果不能假定提供了什么类型。下面的代码不能执行:
class MyGenericClass<T1, T2, T3>
{
private T1 t1Object; public MyGenericClass()
{
t1Object = new T1();
}
}
因为我们不知道T1是否有公有的默认构造函数。
default关键字
如果我们定义了一个泛型的字段,我们想在构造函数中初始化它,但是我们不知道它的引用类型还是值类型,那么default就派上用处了:
public MyGenericClass()
{
t1Object = default(T1);
}
如果是值类型就赋值0,引用类型就赋值null。
约束类型
在定义泛型的时候我们可以对类型进行约束,通过where关键字实现:
class MyGenericClass<T1> where T : constraint1,constraint
{
...
}
constraint定义了约束,多个约束用逗号隔开,如果有多个类型:
class MyGenericClass<T1, T2> where T1 : constraint1 where T2 : constraint
{
...
}
下面给出一些可用的约束
约束 说明
where T:struct 使用结构约束,类型T必须是值类型
where T:calss 类约束指定,类型T必须是引用类型
where T:interface 指定类型T必须实现是接口或者实现了接口
where T:base-class 指定类型T必须是基类或者派生于基类
where T:new() 指定类型T必须有一个默认构造函数
下面结合以上知识给个实例:(PS不要看到代码多 其实很简单的 耐心看下去)
先定义四个类Animal、Cow 、Chicken和SuperCow
#region Animal 虚基类 有一个name属性 Feed方法和一个虚方法MakeANoise
//虚基类 有一个name属性 Feed方法和一个虚方法MakeANoise
public abstract class Animal
{
protected string name; public string Name
{
get
{
return name;
}
set
{
name = value;
}
} public Animal()
{
name = "The animal with no name";
} public Animal(string newName)
{
name = newName;
} public void Feed()
{
Console.WriteLine("{0} has been fed.", name);
} public abstract void MakeANoise();
}
#endregion
//Cow Animal的子类,实现虚方法
public class Cow:Animal
{
public Cow(string name) :
base(name)
{
}
public override void MakeANoise()
{
Console.WriteLine("{0} says 'moo!'", name);
}
}
//Chicken类,Animal子类
public class Chicken:Animal
{
public Chicken(string name)
: base(name)
{ }
public override void MakeANoise()
{
Console.WriteLine("{0} says 'cluck'", name);
}
}
//Cow的子类,有一个自己的方法Fly
class SuperCow : Cow
{
public SuperCow(string name) : base(name)
{
} public void Fly()
{
Console.WriteLine("{0} is flying!", name);
} public override void MakeANoise()
{
Console.WriteLine("{0} says 'I am supercow!'", name);
} }
类准备好了之后,我们可以开始定义我们的泛型了:
//继承了迭代器接口,这样方便使用Foreach 约束它的类型为Animal及其子类
public class Farm<T>:IEnumerable<T> where T : Animal
{
private List<T> animals = new List<T>(); public List<T> Animals
{
get
{
return animals;
}
}
//迭代器
public IEnumerator<T> GetEnumerator()
{
return animals.GetEnumerator();
} IEnumerator IEnumerable.GetEnumerator()
{
return animals.GetEnumerator();
} //执行所有animal的MakeANoise()
public void MakeNoises()
{
foreach (T animal in animals)
{
animal.MakeANoise();
}
}
//执行所有animal的Feed()
public void FeedTheAnimals()
{
foreach (T animal in animals)
{
animal.Feed();
}
}
//获得animals中的cow
public Farm<Cow> GetCows()
{
Farm<Cow> cowFarm = new Farm<Cow>();
foreach (T animal in animals)
{
if (animal is Cow)
{
cowFarm.Animals.Add(animal as Cow);
}
}
return cowFarm;
}
}
泛型定义好了,我们用写代码来调用它:
class Program
{
static void Main(string[] args)
{
Farm<Animal> farm = new Farm<Animal>();
farm.Animals.Add(new Cow("Jack"));
farm.Animals.Add(new Chicken("Vera"));
farm.Animals.Add(new Chicken("Sally"));
farm.Animals.Add(new SuperCow("Kevin"));
farm.MakeNoises(); Farm<Cow> dairyFarm = farm.GetCows();
dairyFarm.FeedTheAnimals(); foreach (Cow cow in dairyFarm)
{
if (cow is SuperCow)
{
(cow as SuperCow).Fly();
}
}
Console.ReadKey();
}
}
结果:
这样一个泛型就OK了。
——Stay hungry!Stay foolish!
.NET基础之自定义泛型的更多相关文章
- JAVA基础_自定义泛型
泛型的来源 在Java中,泛型借鉴了C++的模版函数,从而引入了泛型. C++泛型 int add(int x,int y){ return x + y; } float add(float x.fl ...
- Java基础教程:泛型基础
Java基础教程:泛型基础 引入泛型 传统编写的限制: 在Java中一般的类和方法,只能使用具体的类型,要么是基本数据类型,要么是自定义类型.如果要编写可以应用于多种类型的代码,这种刻板的限制就会束缚 ...
- Net is as typeof 运行运算符详解 net 自定义泛型那点事
Net is as typeof 运行运算符详解 概述 在了解运行运算符的前提我们需要了解什么是RTTI ,在任何一门面向对象的语言中,都有RTTI这个概念(即 运行时). RTTI(Run-Ti ...
- java JDK8 学习笔记——第18章 自定义泛型、枚举与注释
第十八章 自定义泛型.枚举与注释 18.1 自定义泛型 泛型定义: (1)仅定义在方法上的泛型语法 (2)用来限制泛型可用类型的extends与super关键字(3)?类型通配字符的使用 18.1.1 ...
- net 自定义泛型那点事
泛型概述 泛型是程序设计语言的一种特性.允许程序员在强类型程序设计语言中编写代码时定义一些可变部分,那些部分在使用前必须作出指明.各种程序设计语言和其编译器.运行环境对泛型的支持均不一样.将类型参数化 ...
- Swagger UI改造 增加 Token验证、显示控制器注释、自定义泛型缓存应用、
/// <summary> /// Swagger 增加 Token 选项和控制器描述 /// </summary> public class CustomOperationF ...
- Flask基础(14)-->自定义过滤器
Flask基础(13)-->自定义过滤器 什么是过滤器? 过滤器的本质就是函数.有时候我们不仅仅只是需要输出变量的值,我们还需要修改变量的显示,甚至格式化.运算等等,而在模板中是不能直接调用 P ...
- Day 8:方法上自定义泛型、类上、接口上、泛型的上下限
泛型 泛型是jdk1.5使用的新特性 泛型的好处: 1. 将运行时的异常提前至了编译时 2. 避免了无谓的强制类型转换 泛型在集合中的常见应用: ArrayList<Strin ...
- Day11_57_自定义泛型
自定义泛型 package com.shige.Generic; //自定义泛型 public class CustomizeGeneric { public static void main(Str ...
随机推荐
- 关于 error: LNK1123: failure during conversion to COFF: file invalid or corrupt 错误的解决方案【Qt】【 VS2010】
近日因为换装硬盘重装了系统,于是不得不重新安装VS2010 Qt 这些个开发工具.安装过程都没什么问题,安装完了顺手点了个例子测试下好没好用,于是就出现了标题中的错误提示.之前处理过一次,时间久了就忘 ...
- git 在windows上 生成ssh公钥
今天上传代码到服务器时,报如下错误: 上网搜了一下,应该是ssh过期了.我们就来生成新的ssh公钥吧. 1. 打开git bash 2. 输入命令: ssh-keygen -t rsa ...
- [转]Oracle hang分析
hanganalyze是ORACLE的一款性能诊断工具,这个款工具是从oracle 8.0.6开始可用,在oracle数据库出现严重的性能问题的时候它可以帮助你定位问题所在. 1.首先说说hangan ...
- 机器学习之SVM(支持向量机)
支持向量机(SVM)是当前非常流行的监督学习方法,其核心主要有两个: 构造一个极大边距分离器--与样例点具有最大可能距离的决策边界: 将在原输入空间中线性不可分的样例映射到高维空间中,从而进行线性分离 ...
- [改善Java代码]不同的列表选择不同的遍历方法
一.场景: 我们来看一个场景,统计一个省的各科高考科目考试的平均分. 当然使用数据库中的一个SQL语句就能求出平均值,不过这个不再我们的考虑之列,这里只考虑使用纯Java的方式来解决.(由于我的机器配 ...
- HTML的style属性
HTML的style属性 HTML的style属性提供了一种改变HTML样式的通用方法.style是在HTML4版本中引用的,它是一种首选的改变HTML元素样式的方法.可以使用style直接的将样式添 ...
- Dynamic\Static\IsKinematic
1.Dynamic: 有Collider和RigidBody的GameObject, Unity视之为Dynamic. 适用于经常变换移动的对象. 2.Static: 只含有Collider的Game ...
- IIS 发布网站到外网
前段时间做了一个项目在局域网中测试后要发布到外网上,一时间不知怎么搞,以为直接在IIS中修改发布时的IP就可以了,但是不可行,经过摸索终于成功发布到外网,下面是具体步骤. 前期准备:公网IP,掩码,网 ...
- HTML+CSS3 纯代码实现转盘效果
<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <meta http ...
- 关于System.Collections空间
System.Collections命名空间包含可使用的集合类和相关的接口,提供了集合的基本功能. 该命名空间下的.NET非泛型集合类如下所示: — System.Collections.ArrayL ...