[C#基础] 泛型
为什么泛型?
在泛型中,最重要的应用便是集合类,因此我们模拟一个简单的集合类
对于上述示例,可以有如下应用
从上可看出,自定义的代码太丑陋了,只能用于string类型。 当然我们可以用object作为参数类型,然而深入分析则会发现有些问题仍然存在,这种方式还会有性能问题,值类型Add到集合时必然存在装箱,而将元素赋值给一个值类型变量时,又存在拆箱,这样的性能损失在操作大量元素的时候格外明显!
正是这个问题的存在,所以泛型登场了。重新实现MyArray类型,体验泛型的好处
自定义的泛型集合类,能够轻松实现对任意类型的管理和操作
泛型基础
泛型类
上述代码已经演示了泛型类,我们发现定义一个泛型类和定义一个非泛型类并没有太大的区别,主要的不同在于,类型参数化。类型定义时,将指定类型形参(通常用T表示),紧随类名,并包含在<>内。
泛型类的声明
上述代码将创建一个值类型对象myArr,并调用构造器完成初始化。int作为T的类型实参在JIT编译时所有T将被替换为int,因此_items内部数组将保存int类型的数组元素。
泛型类型的声明也可以包括多个类型形参,如
约束
约束,指在定义泛型时,对于能够实例化类型参数的类型所做的限制,这种限制能保证类型参数在一定的目标范围。
约束通过where字句来实现,多个约束之间用逗号隔开,通常情况下,在设计泛型类或泛型方法时,如果要实现一个object类不支持的任何方法,则需要对参数类型进行约束。
约束主要包括,构造器约束,值类型约束,引用类型约束,基类约束,接口约束,具体情况为:
- T:new(),表示类型参数必须具有公共无参构造函数,有多个约束存在时,必须把new()放在最后面。
- T:struct,表示类型参数必须是值类型。
- T:class,表示类型参数必须是引用类型。
- T:基类名,表示类型参数必须是基类或及其派生类。但是不能既指定基类约束又指定class
- T:接口名,表示类型参数必须是指定的接口,或实现了该接口的接口,可以同时定义多个接口约束,约束接口也可以是泛型的。
泛型方法
泛型方法可以存在于泛型类,也可以存在于非泛型类中
可以将类型参数作为某个方法的参数,返回值或者局部变量,该类型参数可能并不被整个类所需要,而更明确的用于某个方法,例如
方法重载
由此可见,泛型方法可以以类型参数形成重载。
然而泛型方法重载,存在值得注意的问题:由于类型参数在编译期并不确定,所以重载检查是在实例方法被调用时才发生,而不是在泛型类本身编译时检查,例如下面的泛型类可以安全通过编译
然而在调用时,将会给出调用不明确的编译错误
类型推断
泛型方法的调用,可以用更简洁的方式来实现,例如
myArr.ShowInfo("s");
这种机制由编译器支持,称为:类型推断。表示编译器在调用时根据方法参数来推断类型参数,从而决定使用的泛型方法。因此,这种方式不适用于没有参数的方法,同时编译器会优先选择显示的匹配调用,例如上面所示,将会调用非泛型ShowInfo方法,而非泛型方法ShowInfo<TData>方法,除非以显示方式调用来是实现。
myArr.ShowInfo<string>("s");
泛型接口
泛型接口允许我们编写参数和接口成员返回类型是泛型类型参数的接口
如下代码声明了一个泛型接口,泛型类Sample实现了这个泛型接口,main实例化了两个泛型类,一个int 一个string
- class Program
- {
- static void Main(string[] args)
- {
- var testInt = new Sample<int>();
- var testString = new Sample<string>();
- Console.WriteLine(testInt.ReturnIt());
- Console.WriteLine(testString.ReturnIt("aa"));
- }
- }
- interface IMyIfc<T>
- {
- T ReturnIt(T value);
- }
- class Sample<T> : IMyIfc<T>
- {
- public T ReturnIt(T value)
- {
- throw new NotImplementedException();
- }
- }
使用泛型接口的示例
如下示例演示了泛型接口的两个额外的能力
- 与其他泛型相似,实现不同类型参数的泛型接口是不同的接口
- 可以在非泛型类中实现泛型接口
例如下面的代码与前面的相似,但在这里Sample是个非泛型类,它实现了两个IMyIfc实例,一个int类型,一个string类型
- class Sample : IMyIfc<int>, IMyIfc<string>
- {
- public int ReturnIt(int value) //实现int类型
- {
- throw new NotImplementedException();
- }
- public string ReturnIt(string value) //实现string类型
- {
- throw new NotImplementedException();
- }
- }
泛型接口的实现必须唯一
实现泛型接口时,必须保证类型实参组合不会在类型中产生两个重复的接口
典型的泛型接口
泛型委托
泛型委托支持在返回值和参数上应用类型参数,泛型委托能够有效避免值类型实例传给一个回调方法时的装箱处理。
如下代码实现了一个泛型委托的简单示例
- class Program
- {
- //声明一个泛型委托
- public delegate string MyGenericDelegate<T>(T t);
- //实现两个测试方法
- public static string GetString(string s)
- {
- return s;
- }
- public static string GetInt(int i)
- {
- return i.ToString();
- }
- static void Main(string[] args)
- {
- //实现测试代码
- MyGenericDelegate<string> myStrDel = new MyGenericDelegate<string>(GetString);
- Console.WriteLine(myStrDel("aa"));
- MyGenericDelegate<int> myIntDel=new MyGenericDelegate<int>(GetInt);
- Console.WriteLine(myIntDel());
- }
- }
实用Func泛型委托的示例
- class Sample
- {
- public static string PringStr(string s1, string s2)
- {
- return s1 + s2;
- }
- }
- static void Main(string[] args)
- {
- var myDel = new Func<string, string, string>(Sample.PringStr);
- Console.WriteLine(myDel("",""));
- }
[C#基础] 泛型的更多相关文章
- Java17-java语法基础——泛型
Java18-java语法基础——泛型 一.泛型概念和作用 1.泛型概念: 泛型是JavaSE1.5的新特性,泛型的本质是参数化类型,也就是说,所操作的数据类型被指定为一个参数.这种参数类型可以用在类 ...
- 一天一个Java基础——泛型
这学期的新课——设计模式,由我仰慕已久的老师传授,可惜思维过快,第一节就被老师挑中上去敲代码,自此在心里烙下了阴影,都是Java基础欠下的债 这学期的新课——算法设计与分析,虽老师不爱与同学互动式的讲 ...
- Java 基础 -- 泛型、集合、IO、反射
package com.java.map.test; import java.util.ArrayList; import java.util.Collection; import java.util ...
- [c#基础]泛型集合的自定义类型排序
引用 最近总有种感觉,自己复习的进度总被项目中的问题给耽搁了,项目中遇到的问题,不总结又不行,只能将复习基础方面的东西放后再放后.一直没研究过太深奥的东西,过去一年一直在基础上打转,写代码,反编译,不 ...
- java基础-泛型3
浏览以下内容前,请点击并阅读 声明 8 类型擦除 为实现泛型,java编译器进行如下操作进行类型擦除: 如果类型参数有限制则替换为限制的类型,如果没有则替换为Object类,变成普通的类,接口和方法. ...
- java基础 泛型
泛型的存在,是为了使用不确定的类型. 为什么有泛型? 1. 为了提高安全 2. 提高代码的重用率 (自动 装箱,拆箱功能) 一切好处看代码: package test1; import java.la ...
- java基础-泛型2
浏览以下内容前,请点击并阅读 声明 6 类型推测 java编译器能够检查所有的方法调用和对应的声明来决定类型的实参,即类型推测,类型的推测算法推测满足所有参数的最具体类型,如下例所示: //泛型方法的 ...
- java基础-泛型1
浏览以下内容前,请点击并阅读 声明 泛型的使用能使类型名称作为类或者接口定义中的参数,就像一般的参数一样,使得定义的类型通用性更强. 泛型的优势: 编译具有严格的类型检查 java编译器对于泛型代码的 ...
- Spring基础—— 泛型依赖注入
一.为了更加快捷的开发,为了更少的配置,特别是针对 Web 环境的开发,从 Spring 4.0 之后,Spring 引入了 泛型依赖注入. 二.泛型依赖注入:子类之间的依赖关系由其父类泛型以及父类之 ...
随机推荐
- Javascript DOM 02 在<ul>中创建、删除 <li>
创建DOM元素 createElement(标签名) 创建一个节点 appendChild(节点) 追加一个节点 例子:为ul插入li 插入元素 insertBefore(节点, 原有节点) 在 ...
- 摘抄python __init__
注意1.__init__并不相当于C#中的构造函数,执行它的时候,实例已构造出来了. 1 2 3 4 5 class A(object): def __init__(self,name): ...
- Storyboard中使用UIscrollView添加约束的开发总结
第一次在项目中用storyboard做界面,一般的界面直接添加约束非常爽快 然后有个界面有scrollview,添加了约束还总是出错 刚开始使用了 wCompact,hRegular,滑动出现问题,有 ...
- c++怎样让返回对象的函数不调用拷贝构造函数
我们知道拷贝构造函数有两种“默默”的方式被调用 1. 想函数传入 值参数 2. 函数返回 值类型 今天我们讨论函数返回值类型的情况. 得到结论是 1. 当对象有拷贝构造函数(系统为我们生成.或者我们自 ...
- PHP - 图像处理
第14章 处理图像 学习要点: 1.创建图像 2.简单小案例 在PHP5中,动态图象的处理要比以前容易得多.PHP5在php.ini文件中包含了GD扩展包,只需去掉GD扩展包的相应注释就可以正常使用了 ...
- Python 获取时间戳
Python 获取时间通过 time 模块 如下代码,是通过获取当前的时间,按照格式输出 Python默认获取当前的时间返回的都是时间的元组,下面是元组的,字符串时间的一个转换输出 # -*- cod ...
- Hashtable的使用
Hashtable mylist = new Hashtable(); mylist.Add("1", "100"); ...
- Cppcheck 1.54 C/C++静态代码分析工具
Cppcheck是一个C/C++代码分析工具,只检测那些编译器通常无法检测到的bug类型. 官方上建议让编译器提供尽量多的警告提示:1.使用Visual C++的话,应使用警告等级4 2.使用GC ...
- VC 对话框背景颜色、控件颜色
系统环境:Windows 7软件环境:Visual C++ 2008 SP1本次目的:为对话框设置背景颜色.控件颜色 既然MFC对话框不好开发,那么现在我们来开始美化我们的对话框.为对话框设置背景颜色 ...
- Square:从今天開始抛弃Fragment吧!
原文链接 : Advocating Against Android Fragments 原文作者 : Pierre-Yves Ricau 译文出自 : 开发技术前线 www.devtf.cn 译者 : ...