C#泛型基础知识点总结
2021-05-06 21:30
标签:c# 5.0 定义 time 鼠标 com font datetime oid
1.0 什么是泛型
泛型是C#2.0和CLR(公共语言运行时)升级的一个新特性,泛型为.NET 框架引入了一个叫 type parameters(类型参数)的概念,type parameters 使得程序在设计的时候,不必设计其具体的参数,其具体的参数可以延迟到需要的时候声明或调用。使用泛型代码运行时避免了类型转换的装箱和拆箱操作。
2.0 泛型的延迟声明
1 1 using System;
2 2 using System.Collections.Generic;
3 3 using System.Linq;
4 4 using System.Text;
5 5 using System.Threading.Tasks;
6 6
7 7 namespace _20171010Generic
8 8 {
9 9 ///
10 10 /// 泛型方法相关类
11 11 ///
12 12 public class GenericMethod
13 13 {
14 14 ///
15 15 /// 泛型方法:方法带和type parameters(类型参数 T)的
16 16 ///
17 17 ///
18 18 ///
19 19 public static void Show(T tParameters)
20 20 {
21 21 Console.WriteLine("{0}方法,parameter={1}参数,type={2}类型", typeof(GenericMethod).Name, tParameters, tParameters.GetType().Name);
22 22 }
23 23 }
24 24 }
如代码所示,在声明泛型方法的时候没有指定具体的参数类型,等到需要调用的时候再指定,这就叫做延迟声明。泛型的设计思想(延迟思想,推迟一切可以推迟的)
1 1 using System; 2 2 using System.Collections.Generic; 3 3 using System.Linq; 4 4 using System.Text; 5 5 using System.Threading.Tasks; 6 6 7 7 namespace _20171010Generic 8 8 { 9 9 class Program 10 10 { 11 11 static void Main(string[] args) 12 12 { 13 13 14 14 int iValue = 123; 15 15 string sValue = "TestName"; 16 16 DateTime dtValue = DateTime.Now; 17 17 object oValue = new object(); 18 18 19 19 GenericMethod.Show(iValue); 20 20 GenericMethod.Show(sValue); 21 21 GenericMethod.Show(oValue); 22 22 GenericMethod.Show(dtValue); 23 23 Console.WriteLine("———————我是华丽的分割线————————"); 24 24 GenericMethod.Showint>(iValue); 25 25 GenericMethod.Showstring>(sValue); 26 26 GenericMethod.Showobject>(oValue); 27 27 GenericMethod.Show(dtValue); 28 28 29 29 Console.WriteLine("———————我是华丽的分割线————————"); 30 30 Console.WriteLine(typeof(Listint>)); 31 31 Console.WriteLine(typeof(Dictionary)); 32 32 Console.WriteLine("———————我是华丽的分割线————————"); 33 33 } 34 34 } 35 35 }
泛型方法的调用,第一种 GenericMethod.Show(iValue);调用方法不指定类型参数,在编译的时候编译器自动编译推算(语法糖),第二种 GenericMethod.Show
3.0 泛型主要的四种:泛型类, 泛型方法,泛型接口,泛型委托
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks; namespace _20171010Generic { ////// 动物类 /// 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(); } ////// 狗类 /// public class Dog:AnimalModel { public override void Cry() { Console.WriteLine("旺旺旺。。。。。"); } } ////// 猫类 /// public class Cat : AnimalModel { public override void Cry() { Console.WriteLine("喵喵瞄。。。。。。。"); } } ////// 玫瑰花类 /// public class Rose { public int Id { get; set; } public string Name { get; set; } } }
首先先新建了一个AnimalModel类,里面定义了一个动物类,动物类里有个虚方法Cry,一个狗类,狗类继承了动物类,一个猫类,重写了虚方法Cry。一个IEat接口和ISleep接口,
1 using System; 2 using System.Collections.Generic; 3 using System.Linq; 4 using System.Text; 5 using System.Threading.Tasks; 6 7 namespace _20171010Generic 8 { 9 ///10 /// 泛型类 11 /// 12 /// 类型参数 13 /// 类型参数 14 /// 类型参数 15 public class GenericClass 16 { 17 /// > 80 { 81 public string Sleep(string t) 82 { 83 Console.WriteLine("实现了sleep泛型接口,返回参数是:{0}", t); 84 return t; 85 } 86 } 87 }18 /// 无返回值的泛型方法 19 /// 20 /// 21 public void Show(T t) 22 { 23 24 } 25 /// 26 /// 有返回值的泛型方法 27 /// 28 /// 29 /// 30 public T Get() 31 { 32 return default(T); 33 } 34 } 35 36 /// 37 /// 泛型接口 38 /// 39 /// 40 public interface ISleep 41 { 42 W Sleep(W t); 43 } 44 45 /// 46 /// 有返回值的泛型委托 47 /// 48 /// 49 /// 50 public delegate Y DlgYFun (); 51 52 public delegate int DlgIntFun(); 53 54 /// 55 /// 泛型类 56 /// 57 /// 58 /// 59 /// 60 public class GenericChild 61 //: GenericClass 直接继承泛型类 62 //: GenericClass//类型参数可直接指定 63 //: ISleep 64 : ISleep //实现泛型接口 65 { 66 T ISleep .Sleep(T t) 67 { 68 return default(T); 69 } 70 } 71 72 /// 73 /// 普通类 74 /// 75 public class Child 76 // :GenericClass 错误的继承,普通类不能直接继承泛型类 77 //: GenericClass//必须指定全部确定的类型参数后可继承泛型 78 //:ISleep 错误的实现泛型接口,普通类不能直接实现泛型接口, 79 : ISleepstring
泛型类就在普通类名字后面加上和多个类型参数,需要注意的是 1.普通类不能直接继承泛型类和泛型接口,因为泛型的类型参数不确定,但是泛型类或泛型接口指定类型后可以继承泛型类或实现泛型接口,2.泛型类可以直接继承泛型类,也可以直接实现泛型接口,其子类的类型参数相当于声明了局部参数。
4.0泛型的约束(基类约束,接口约束,引用类型约束,值类型约束,无参构造函数约束)
回到上面写的那个GenericMethod类里的show方法,new 一个cat对象,Cat cat=new Cat(){ Id=1,Name="小黑猫"}; 然后调用genericMentod.show(cat)方法。但是如果想要在show方法里访问Id,或者Name却不行。T是个不明确类型参数,所以无法访问,如图
使用泛型约束解决方法:
1 using System;
2 using System.Collections.Generic;
3 using System.Linq;
4 using System.Text;
5 using System.Threading.Tasks;
6
7 namespace _20171010Generic
8 {
9 ///
10 /// 泛型约束
11 ///
12 public class Constraint
13 {
14
15 public static void Show(T tParameter)
16 //where T: AnimalModel 基类约束,就可以访问该类的方法或属性
17 where T:Cat //或者该子类
18 {
19 Console.WriteLine("泛型约束show方法--------id={0},name={1}",tParameter.Id,tParameter.Name);
20 }
21
22 public static void Show(AnimalModel model)
23 {
24 Console.WriteLine("普通show方法--------id={0},name={1}", model.Id, model.Name);
25 }
26
27 public static void ShowInterface(T tParameter)
28 //where T: AnimalModel 基类约束,就可以访问该类的方法或属性
29 where T : Cat,ISleep,IEat//或者该子类约束,多个接口约束
30
31 {
32 Console.WriteLine("泛型约束ShowInterface方法--------id={0},name={1}", tParameter.Id, tParameter.Name);
33 tParameter.Sleep();//接口的方法
34 tParameter.Eat();
35 }
36 }
37
Constraint类里的第一个show方法中在后面带个 where关键字 和 约束类型,泛型方法里就能访问Id和Name,第二个show方法是作为对比,虽然第二个方法也能实现同样的效果,但是相对泛型方法不灵活,泛型方法可以同时约束多个,比如第三个方法约束多个接口,和类,多个约束的关系是&&关系
泛型约束除了基类约束和接口约束几种,还有值类型约束,无参构造约束,引用类型约束等这几种。
1 public static T TestFun()
2 // where T:class //引用类型约束
3 // where T:struct //值类型约束
4 where T : new() //无参构造函数约束
5 {
6 T t = new T();
7 return default(T);
8 }
5.0协变和逆变
out 协变(covariant) 修饰返回值,in 逆变(contravariant) 修饰传入参数。out和in只能放在接口或者泛型委托的的参数前面,类没有协变和逆变。在.NET Framework里面,IEnumerable
像平常一样写代码: AnimalModel animal = new AnimalModel();//实例化一个动物。
Dog dog = new Dog();//实例化一个条单身狗
AnimalModel dog2 = new Dog();//实例化一条单身狗(狗继承了动物父类,父类出现的地方都可以用子类代替,对的,狗一定是个动物)
// Dog dog3 = new AnimalModel();动物不一定是条单身狗,程序编译不通过
new一条单身狗没问题,new 一群单身狗试试看。
List
List animalDog = new List
理论上来说第二种实例化一群狗的方式是没毛病的,一群狗也一定是一群动物,但是程序上是不通过是因为Listt
PS:写到这里我就快写不下去了,狗快被我自己玩坏了。
要使上面那句代码编译通过,可以通过lambda表达式转化 ListAnimalModel> animalDog = new List
使用IEnumerable:IEnumerable animalDog= new List
1 public interface IMyTestout T> 2 { 3 4 } 5 public class Test: IMyTest 6 { 7 8 } 9 10 11 IMyTest test3 = new Test ();
逆变就和协变相反。逆变的in 的参数只能作为传入值,不能作为返回值。说白了,也是一种约束。协变和逆变的关键作用就是让编译器在运行时不报错。
1 1 public interface IMyTest2 2 { 3 3 4 4 } 5 5 public class Test : IMyTest 6 6 { 7 7 8 8 } 9 9 10 10 11 11 IMyTest test3 = new Test();
关于泛型的知识点还有很多,比如还有泛型的缓存,这个就有点难理解了。以上知识点是我平常通过各种途径学习总结的几点。如有不对欢迎指正。欢迎转载和分享,转载分享时请注明原创出处:如此拉风的女人
C#泛型基础知识点总结
标签:c# 5.0 定义 time 鼠标 com font datetime oid
原文地址:http://www.cnblogs.com/wwym/p/7646935.html