.NET泛型02,泛型的使用
在" .NET泛型01,为什么需要泛型,泛型基本语法"中,了解了泛型的基本概念,本篇偏重于泛型的使用。主要包括:
■ 泛型方法重载需要注意的问题
■ 泛型的类型推断
■ 泛型方法也可以有约束
■ 泛型接口
■ 泛型委托
■ 使用EventHandler<TEventArgs>事件泛型
泛型方法重载需要注意的问题
- public class MyArray<T>
- {
- public T myData;
- public MyArray()
- {
- myData = default(T);
- }
- public void ShowInfo()
- {
- Console.WriteLine(myData.ToString());
- }
- public void ShowInfo(string str)
- {
- Console.WriteLine(str);
- }
- public void ShowInfo<T>(T data)
- {
- Console.WriteLine(data.ToString());
- }
- }
以上,说明:泛型方法可以作为方法的重载。
可以这样调用。
- MyArray<Student> myArray = new MyArray<Student>();
- myArray.ShowInfo<CollegeStudent>(new CollegeStudent());
- myArray.ShowInfo<string>("HelloWorld");
但还有一种情况是:两个语义不明的重载方法,在编译的时候是通过的,但在调用的时候就不会通过。比如以下在编译时没问题:
- public class MyArray<T>
- {
- public void ShowInfo<TA, TB>(TA a, TB b){};
- public void ShowInfo<TB, TA>(TA a, TB b){};
- }
如果这样调用,就有问题:
- MyArray<Student> myArray = new MyArray<Student>();
- myArray.showInfo<Student, Student>(new Student(), new Student());
所以,对于泛型重载方法,需要注意语义不明的情况。
泛型的类型推断
编译器可以根据方法参数的类型来推断使用哪个重载方法,优先调用一般重载方法,然后再调用泛型重载方法。
- myArray.ShowInfo("hello"); 会调用 ShowInfo(string str)重载方法
- myArray.ShowInfo(new CollegeStudent());会调用ShowInfo<T>(T data)重载方法。
泛型方法也可以有约束
我们知道泛型类可以有约束,泛型方法也一样。
- public void ShowInfo<T>(T data) where TData : Student
- {
- Console.WriteLine(data.ToString());
- }
泛型接口
.NET集合类就提供了多个泛型接口,比如:IList<T>, ICollection<T>, IComparable<>, IComparer<T>, IEnumerable<T>, IEnumerator<T>, IDictionary<TKey,TValue>,等等。
自定义类的时候,有时候需要让自定义类实现一个指定了具体类型的泛型接口:
- class MyClass<T> : IComparable<Int32>, IComparable<String>
泛型委托
- public class Generic Delegate
- {
- //声明泛型委托
- public delegate string MyGenericDelegate<T>(T t);
- public static string GetPoint(Point p)
- {
- return stirng.Format("地址是{0},{1}", p.X, p.Y);
- }
- public static string GetMsg(string str)
- {
- return str;
- }
- }
- public static void Main()
- {
- MyGenericDelegate<string> myStrDel = new MyGenericDelegate<string>(GetMsg);
- Console.WriteLine(myStrDel("hello"));
- MyGenericDelegate<Point> myPointDel = new MyGenericDelegate<Point>(GetPoint);
- Console.WriteLine(myPointDel(new Point(100, 200)));
- }
使用EventHandler<TEventArgs>事件泛型
它的完整定义是:
- public delegate void EventHandler<TEventArgs>(object sender, TEventArgs e) where TEventArgs: EventArgs
假设有一个MessageReceiver类,当建立连接时触发OnConnected事件,在接收到信息是触发OnMessageReceived事件。
在创建MessageReceiver类之前,我们先要自定义一个派生于EventArgs,且和MessageReceiver相关的类。
- public sealed class MessageReceivedEventArgs : EventArgs
- {
- public string Message {get;set;}
- public MessageReceivedEventArgs(string msg)
- {
- this.Message = msg;
- }
- }
MessageReceiver类主要包含2个事件,一个是OnConnected事件,另一个是OnMessageReceived事件。
- public class MessageReceiver
- {
- public event EventHandler OnConnected;
- public event EventHandler<MessageReceivedEventArgs> OnMessageReceived;
- ...
- public void DoSth()
- {
- if(OnMessageReceived != null)
- {
- OnMessageReceived(this, new MessageReceivedEventArgs(msg));
- }
- }
- }
以上,通过if(OnMessageReceived != null)这个判断,能保证:当没有订阅者注册事件的时候,这个事件不被触发。但在多线程场景中,这样做也不是最合理的:
假设线程A作为订阅者注册事件,正准备触发事件的时候,线程B也作为订阅者刚好在此刻注销了事件,即OnMessageReceived变成了null,这就牵累到线程A也无法触发事件。
解决办法是把事件变量赋值给一个局部变量:
- public class MessageReceiver
- {
- public event EventHandler OnConnected;
- public event EventHandler<MessageReceivedEventArgs> OnMessageReceived;
- ...
- public void DoSth()
- {
- var handler = OnMessageReceived;
- if(handler != null)
- {
- handler(this, new MessageReceivedEventArgs(msg));
- }
- }
- }
这样,当线程A作为订阅者注册并准备触发事件的时候,及时线程B在此刻注销注册,让OnMessageReceived为null,由于已经把OnMessageReceived赋值给了局部变量handler,线程A依然能触发事件。
参考资料:
《你必须知道的.NET(第2版)》,作者王涛。
".NET泛型"系列包括:
.NET泛型01,为什么需要泛型,泛型基本语法
.NET泛型02,泛型的使用
.NET泛型03,泛型类型的转换,协变和逆变
.NET泛型04,使用Lazy<T>实现延迟加载
.NET泛型02,泛型的使用的更多相关文章
- day29--Java泛型02
Java泛型02 5.自定义泛型 5.1自定义泛型类 基本语法: class 类名<T,R...>{//-表示可以有多个泛型 成员 } 注意细节: 普通成员可以使用泛型(属性.方法) 使用 ...
- C#2.0新增功能02 泛型
连载目录 [已更新最新开发文章,点击查看详细] C# 语言和公共语言运行时 (CLR) 的 2.0 版本中添加了泛型. 泛型将类型参数的概念引入 .NET Framework,这样就可以设计具有 ...
- Scala 深入浅出实战经典 第42讲:scala 泛型类,泛型函数,泛型在spark中的广泛应用
王家林亲授<DT大数据梦工厂>大数据实战视频 Scala 深入浅出实战经典(1-64讲)完整视频.PPT.代码下载:百度云盘:http://pan.baidu.com/s/1c0noOt6 ...
- Java基础---Java---基础加强---内省的简单运用、注解的定义与反射调用、 自定义注解及其应用、泛型及泛型的高级应用、泛型集合的综合
内省的简单运用: JavaBean是一种特殊的Java类,主要用于传递数据信息,这种java类中的方法主要用于访问私有的字段,且方法名符合某种命名规则. 采用遍历BeanInfo的所有属性方式来查找和 ...
- JavaSE学习总结(十六)—— 泛型与泛型应用
一.泛型概要 泛型(Generic)的本质是类型参数化,通俗的说就是用一个占位符来表示类型,这个类型可以是String,Integer等不确定的类型,表明可接受的类型. 泛型是Java中一个非常重要的 ...
- Java泛型:泛型的定义(类、接口、对象)、使用、继承
地址 http://blog.csdn.net/lirx_tech/article/details/51570138 1. 设计泛型的初衷: 1) 主要是为了解决Java容器无法记忆元素类型的问题 ...
- 泛型学习第四天——List泛型终结:什么是List泛型,泛型筛选,泛型排序
为什么要用泛型集合? 在C# 2.0之前,主要可以通过两种方式实现集合: a.使用ArrayList 直接将对象放入ArrayList,操作直观,但由于集合中的项是Object类型,因此每次使用都必须 ...
- C#高级语法之泛型、泛型约束,类型安全、逆变和协变(思想原理)
一.为什么使用泛型? 泛型其实就是一个不确定的类型,可以用在类和方法上,泛型在声明期间没有明确的定义类型,编译完成之后会生成一个占位符,只有在调用者调用时,传入指定的类型,才会用确切的类型将占位符替换 ...
- 深入理解什么是Java泛型?泛型怎么使用?【纯转】
本篇文章给大家带来的内容是介绍深入理解什么是Java泛型?泛型怎么使用?有一定的参考价值,有需要的朋友可以参考一下,希望对你们有所助. 一.什么是泛型 “泛型” 意味着编写的代码可以被不同类型的对象所 ...
随机推荐
- 数据库中INFORMATION_SCHEMA的说明及使用
第一个查询看看库里有多少个表,表名等select * from INFORMATION_SCHEMA.TABLES information_schema这张数据表保存了MySQL服务器所有数据库的信息 ...
- plsql中做计划任务
第一步: 1. 打开PLSQL后,选择节点jobs,右键新建,弹出界面后再what值中填写需要做计划的存储名加分号结束,如门诊收入存储PH_ClinicIncome(1):其中1代表医疗机构代码 间 ...
- Linux学习笔记:ps -ef、ps aux、kill -9
一.查看进程命令 1.ps命令 Linux中的ps命令是Process Status的缩写. ps命令用来列出系统中当前运行的那些进程. ps命令列出的是当前那些进程的快照,就是执行ps命令的那个时刻 ...
- ES按资源类型统计个数
一.目标:统计各类型资源的个数,输出详细报表 http://10.10.6.225:9200/dsideal_db/t_resource_info/ _mapping { "propert ...
- 【Java】 参数的传递:值传递与引用传递讨论
内容稍多,可直接看第4点的讨论结果 前言 在涉及到传递参数给方法时,容易出现一些参数传递错误的问题,这就涉及到了参数的传递问题,必须搞清楚:参数是如何传递到方法中的?一般来说,参数的传递可以分为两种: ...
- poj1703 Find them, Catch them(带权并查集)
题目链接 http://poj.org/problem?id=1703 题意 有两个帮派:龙帮和蛇帮,两个帮派共有n个人(编号1~n),输入m组数据,每组数据为D [a][b]或A [a][b],D[ ...
- Java工具类之浮点精确计算
public class Arith { // 默认除法运算精度 private static final int DEF_DIV_SCALE = 10; // 构造器私有,让这个类不能实例化 pri ...
- CentOS重装grub修复损坏的系统
grub损坏一般有两种情况:第一.安装双系统时,后安装的系统把先安装的系统的MBR删除了.第二.误操作将grub文件删除了. 不管怎样都需要进入到救援模式,详细请看CentOS通过光盘启动救援数据 ( ...
- 【SQL】180. Consecutive Numbers
Write a SQL query to find all numbers that appear at least three times consecutively. +----+-----+ | ...
- 详解Python中的__init__和__new__(静态方法)
一.__init__ 方法是什么? 使用Python写过面向对象的代码的同学,可能对 __init__ 方法已经非常熟悉了,__init__ 方法通常用在初始化一个类实例的时候.例如: #-*- co ...