C#泛型基础知识点总结
1.0 什么是泛型
泛型是C#2.0和CLR(公共语言运行时)升级的一个新特性,泛型为.NET 框架引入了一个叫 type parameters(类型参数)的概念,type parameters 使得程序在设计的时候,不必设计其具体的参数,其具体的参数可以延迟到需要的时候声明或调用。使用泛型代码运行时避免了类型转换的装箱和拆箱操作。
2.0 泛型的延迟声明:把参数类型的声明推迟到调用,不是语法糖,而是由框架升级提供的功能
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace _20171010Generic
{
/// <summary>
/// 泛型方法相关类
/// </summary>
public class GenericMethod
{
/// <summary>
/// 泛型方法:方法带<>和type parameters(类型参数 T)的
/// </summary>
/// <typeparam name="T"></typeparam>
/// <param name="tParameters"></param>
public static void Show<T>(T tParameters)
{
Console.WriteLine("{0}方法,parameter={1}参数,type={2}类型", typeof(GenericMethod).Name, tParameters, tParameters.GetType().Name);
}
}
}
如代码所示,在声明泛型方法的时候没有指定具体的参数类型,等到需要调用的时候再指定,这就叫做延迟声明。泛型的设计思想(延迟思想,推迟一切可以推迟的)
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace _20171010Generic
{
class Program
{
static void Main(string[] args)
{ int iValue = ;
string sValue = "TestName";
DateTime dtValue = DateTime.Now;
object oValue = new object(); GenericMethod.Show(iValue);
GenericMethod.Show(sValue);
GenericMethod.Show(oValue);
GenericMethod.Show(dtValue);
Console.WriteLine("———————我是华丽的分割线————————");
GenericMethod.Show<int>(iValue);
GenericMethod.Show<string>(sValue);
GenericMethod.Show<object>(oValue);
GenericMethod.Show<DateTime>(dtValue); Console.WriteLine("———————我是华丽的分割线————————");
Console.WriteLine(typeof(List<int>));
Console.WriteLine(typeof(Dictionary<,>));
Console.WriteLine("———————我是华丽的分割线————————");
}
}
}
泛型方法的调用,第一种 GenericMethod.Show(iValue);调用方法不指定类型参数,在编译的时候编译器自动编译推算(语法糖),第二种 GenericMethod.Show<int>(iValue);调用方法指定类型参数,类型参数和参数类型须一致,否则编译不通过。VS2017鼠标移上去会提示可以简化方法名称。编译的时候,类型参数编译为占位符,程序运行的时候,JIT(即时编译(Just In-Time compile)即时编译为真实类型。所以使用泛型性能会比使用object作为参数的方法好,(ps:经过测试)。 Console.WriteLine(typeof(List<int>)); 和Console.WriteLine(typeof(Dictionary<,>));的运行结果中有个~1,和~2就表示类型参数的占位符。
3.0 泛型主要的四种:泛型类, 泛型方法,泛型接口,泛型委托
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace _20171010Generic
{
/// <summary>
/// 动物类
/// </summary>
public class AnimalModel
{
public int Id { get; set; }
public String Name { get; set; }
public virtual void Cry()
{ }
}
public interface IEat
{
void Eat();
}
public interface ISleep
{
void Sleep();
} /// <summary>
/// 狗类
/// </summary>
public class Dog:AnimalModel
{
public override void Cry()
{
Console.WriteLine("旺旺旺。。。。。");
}
}
/// <summary>
/// 猫类
/// </summary>
public class Cat : AnimalModel
{
public override void Cry()
{
Console.WriteLine("喵喵瞄。。。。。。。");
}
} /// <summary>
/// 玫瑰花类
/// </summary>
public class Rose
{
public int Id { get; set; }
public string Name { get; set; }
} }
首先先新建了一个AnimalModel类,里面定义了一个动物类,动物类里有个虚方法Cry,一个狗类,狗类继承了动物类,一个猫类,重写了虚方法Cry。一个IEat接口和ISleep接口,
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace _20171010Generic
{
/// <summary>
/// 泛型类
/// </summary>
/// <typeparam name="T">类型参数</typeparam>
/// <typeparam name="S">类型参数</typeparam>
/// <typeparam name="K">类型参数</typeparam>
public class GenericClass<T, S, K>
{
/// <summary>
/// 无返回值的泛型方法
/// </summary>
/// <typeparam name="T"></typeparam>
public void Show(T t)
{ }
/// <summary>
/// 有返回值的泛型方法
/// </summary>
/// <typeparam name="T"></typeparam>
/// <returns></returns>
public T Get()
{
return default(T);
}
} /// <summary>
/// 泛型接口
/// </summary>
/// <typeparam name="W"></typeparam>
public interface ISleep<W>
{
W Sleep(W t);
} /// <summary>
/// 有返回值的泛型委托
/// </summary>
/// <typeparam name="Y"></typeparam>
/// <returns></returns>
public delegate Y DlgYFun<Y>(); public delegate int DlgIntFun(); /// <summary>
/// 泛型类
/// </summary>
/// <typeparam name="W"></typeparam>
/// <typeparam name="Y"></typeparam>
/// <typeparam name="M"></typeparam>
public class GenericChild<T, S, K>
//: GenericClass<T, S, K>直接继承泛型类
//: GenericClass<T, S, string>//类型参数可直接指定
//: ISleep<string>
: ISleep<T>//实现泛型接口
{
T ISleep<T>.Sleep(T t)
{
return default(T);
}
} /// <summary>
/// 普通类
/// </summary>
public class Child
// :GenericClass<T,S,K>错误的继承,普通类不能直接继承泛型类
//: GenericClass<string, int, double>//必须指定全部确定的类型参数后可继承泛型
//:ISleep<W>错误的实现泛型接口,普通类不能直接实现泛型接口,
: ISleep<string>
{
public string Sleep(string t)
{
Console.WriteLine("实现了sleep泛型接口,返回参数是:{0}", t);
return t;
}
}
}
泛型类就在普通类名字后面加上<>和多个类型参数,需要注意的是 1.普通类不能直接继承泛型类和泛型接口,因为泛型的类型参数不确定,但是泛型类或泛型接口指定类型后可以继承泛型类或实现泛型接口,2.泛型类可以直接继承泛型类,也可以直接实现泛型接口,其子类的类型参数相当于声明了局部参数。
4.0泛型的约束(基类约束,接口约束,引用类型约束,值类型约束,无参构造函数约束)
回到上面写的那个GenericMethod类里的show方法,new 一个cat对象,Cat cat=new Cat(){ Id=1,Name="小黑猫"}; 然后调用genericMentod.show(cat)方法。但是如果想要在show方法里访问Id,或者Name却不行。T是个不明确类型参数,所以无法访问,如图
使用泛型约束解决方法:
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks; namespace _20171010Generic
{
/// <summary>
/// 泛型约束
/// </summary>
public class Constraint
{ public static void Show<T>(T tParameter)
//where T: AnimalModel 基类约束,就可以访问该类的方法或属性
where T:Cat //或者该子类
{
Console.WriteLine("泛型约束show方法--------id={0},name={1}",tParameter.Id,tParameter.Name);
} public static void Show(AnimalModel model)
{
Console.WriteLine("普通show方法--------id={0},name={1}", model.Id, model.Name);
} public static void ShowInterface<T>(T tParameter)
//where T: AnimalModel 基类约束,就可以访问该类的方法或属性
where T : Cat,ISleep,IEat//或者该子类约束,多个接口约束 {
Console.WriteLine("泛型约束ShowInterface方法--------id={0},name={1}", tParameter.Id, tParameter.Name);
tParameter.Sleep();//接口的方法
tParameter.Eat();
}
}
Constraint类里的第一个show方法中在后面带个 where关键字 和 约束类型,泛型方法里就能访问Id和Name,第二个show方法是作为对比,虽然第二个方法也能实现同样的效果,但是相对泛型方法不灵活,泛型方法可以同时约束多个,比如第三个方法约束多个接口,和类,多个约束的关系是&&关系
泛型约束除了基类约束和接口约束几种,还有值类型约束,无参构造约束,引用类型约束等这几种。
基类约束:
1带来权利,可以使用基类里面的属性和方法。
2带来义务,类型参数必须是基类或者其子类。
public static T TestFun<T>()
// where T:class //引用类型约束
// where T:struct //值类型约束
where T : new() //无参构造函数约束
{
T t = new T();
return default(T);
}
5.0协变和逆变
out 协变(covariant) 修饰返回值,in 逆变(contravariant) 修饰传入参数。out和in只能放在接口或者泛型委托的的参数前面,类没有协变和逆变。在.NET Framework里面,IEnumerable<T>转到定义去看,其实就是个带out参数的泛型接口,Action<T>转到定义去看就是个带in参数的泛型委托。还有一个逆变+协变的Func<T>。
像平常一样写代码: AnimalModel animal = new AnimalModel();//实例化一个动物。
Dog dog = new Dog();//实例化一个条单身狗
AnimalModel dog2 = new Dog();//实例化一条单身狗(狗继承了动物父类,父类出现的地方都可以用子类代替,对的,狗一定是个动物),左边父类,右边子类。
// Dog dog3 = new AnimalModel();动物不一定是条单身狗,程序编译不通过
new一条单身狗没问题,new 一群单身狗试试看。
List<Dog> dogList = new List<Dog>();//实例化一群单身狗(编译通过)
List<AnimalModel> animalDog = new List<Dog>();//实例化一群单身狗(语法上不通过)
理论上来说第二种实例化一群狗的方式是没毛病的,一群狗也一定是一群动物,但是程序上是不通过是因为Listt<T>是个泛型 List<Dog>不是继承List<AnimalModel>,没有父子关系,程序只认关系。。。
PS:写到这里我就快写不下去了,狗快被我自己玩坏了。
要使上面那句代码编译通过,可以通过lambda表达式转化 List<AnimalModel> animalDog = new List<Dog>().Select(x => (AnimalModel)x).ToList();把每条狗都转换一遍
使用IEnumerable:IEnumerable<AnimalModel> animalDog= new List<Dog>(); //这就叫协变。IEnumerable<out T>在编译的时候就通过转化了,我个人理解为out 是表示转化后的T返回标识。平常在工作中,有用过out 关键字作为标识的返回参数,会用,但是不其所以然。原理明白后自己也可以定义一个协变的泛型接口。
public interface IMyTest<out T>
{ }
public class Test<T> : IMyTest<T>
{ } IMyTest<Animal> test3 = new Test<Dog>();
逆变就和协变相反。逆变的in 的参数只能作为传入值,不能作为返回值。说白了,也是一种约束。协变和逆变的关键作用就是让编译器在运行时不报错。
public interface IMyTest<inT>
{ }
public class Test<T> : IMyTest<T>
{ } IMyTest<Dog> test3 = new Test<Animal>();
为什么要用泛型:泛型就是为了满足不同类型,相同代码的重用
关于泛型的知识点还有很多,比如还有泛型的缓存,这个就有点难理解了。以上知识点是我平常通过各种途径学习总结的几点。如有不对欢迎指正。欢迎转载和分享,转载分享时请注明原创出处:如此拉风的女人
C#泛型基础知识点总结的更多相关文章
- TypeScript 基础知识点整理
一.TypeScript的特点 1.支持ES6规范 2.强大的IDE支持(集成开发环境) 允许为变量指定类型,减少你在开发阶段犯错误的几率. 语法提示,在IDE编写代码时,它会根据你所处的上下文把你能 ...
- Java基础知识点总结
前言 本文主要是我之前复习Java基础原理过程中写的Java基础知识点总结.Java的知识点其实非常多,并且有些知识点比较难以理解,有时候我们自以为理解了某些内容,其实可能只是停留在表面上,没有理解其 ...
- JAVA基础知识点总结(全集)
1.JAVA简介 1.1java体系结构:j2se,javaweb,j2ee 1.2java特点:平台无关(虚拟机),垃圾回收(使得java更加稳定) 1.3 JDK与JRE,JDK:java开发环境 ...
- Smali相关的基础知识点
通过本篇博客的学习,相信你可以无压力的读懂Smali语言文件,并可以将Smali还原成java!!! 其实Smali语言并不是很难,如果你有一些汇编指令的基础,学习Smali就更加简单了,有兴趣的可以 ...
- Java:泛型基础
泛型 引入泛型 传统编写的限制: 在Java中一般的类和方法,只能使用具体的类型,要么是基本数据类型,要么是自定义类型.如果要编写可以应用于多种类型的代码,这种刻板的限制就会束缚很多! 解决这种限制的 ...
- java泛型基础
泛型是Java SE 1.5的新特性, 泛型的本质是参数化类型, 也就是说所操作的数据类型被指定为一个参数. 这种参数类型可以用在类.接口和方法的创建中, 分别称为泛型类.泛型接口.泛型方法. Ja ...
- fastclick 源码注解及一些基础知识点
在移动端,网页上的点击穿透问题导致了非常糟糕的用户体验.那么该如何解决这个问题呢? 问题产生的原因 移动端浏览器的点击事件存在300ms的延迟执行,这个延迟是由于移动端需要通过在这个时间段用户是否两次 ...
- .NET基础知识点
.NET基础知识点 l .Net平台 .Net FrameWork框架 l .Net FrameWork框架提供了一个稳定的运行环境,:来保障我们.Net平台正常的运转 l 两种交 ...
- JavaScript 开发者经常忽略或误用的七个基础知识点(转)
JavaScript 本身可以算是一门简单的语言,但我们也不断用智慧和灵活的模式来改进它.昨天我们将这些模式应用到了 JavaScript 框架中,今天这些框架又驱动了我们的 Web 应用程序.很多新 ...
随机推荐
- ORACLE分区表、分区索引详解
详见:http://blog.yemou.net/article/query/info/tytfjhfascvhzxcyt160 ORACLE分区表.分区索引ORACLE对于分区表方式其实就是将表分段 ...
- JS查错小工具-三生有幸【推荐】
H5和CSS语言在开发者官网上都有在线查错工具,同样的,更加复杂的JavaScript也需要一个查错工具,(别指望DreamWeaver了,debug功能做的太垃圾,还不如Firefox自带的强..) ...
- MPLS LDP随堂笔记2
前一天排错 Acl 1 匹配所有ospf的数据包 (目的 ospf建立邻居关系 传递路由条目) 2 放行UDP报文 让LDP邻居能互相收发HELLO包 4 放行TCP报文 让LDP邻居能够建立TCP会 ...
- HTML5 贝塞尔绘画 桃心
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/ ...
- 201521123015《Java程序设计》第1周学习总结
1.本周学习总结 知道了JAVA语言的发展历史和目前使用的版本,还有什么是JDK(Java Development Kit).JRE (Java Runtime Environment).JVM(Ja ...
- 201521123010 《Java程序设计》第12周学习总结
1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结多流与文件相关内容. 2. 书面作业 将Student对象(属性:int id, String name,int age,doubl ...
- 201521123070 《JAVA程序设计》第10周学习总结
1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结异常与多线程相关内容. 2. 书面作业 本次PTA作业题集异常.多线程 Q1. finally 题目4-2 1.1 截图你的提交结果 ...
- 再起航,我的学习笔记之JavaScript设计模式22(访问者模式)
访问者模式 概念介绍 访问者模式(Visitor): 针对于对象结构中的元素,定义在不改变该对象的前提下访问结构中元素的新方法 解决低版本IE兼容性 我们来看下面这段代码,这段代码,我们封装了一个绑定 ...
- panda库2
>>> a=pd.Series([1,2],index=['a','b']) >>> a a 1 b 2 dtype: int64 >>> b=p ...
- 如何使用fiddler抓取https请求(PC和移动端)
最近做一个抓取移动端app接口,并执行评论,收藏的接口功能测试.怎么搞/(ㄒoㄒ)/~~ 按照老思路试一试,第一步还是要用fiddler来帮忙获取接口信息! 一.基本的抓取http请求设置: 1.cm ...