在" .NET泛型01,为什么需要泛型,泛型基本语法"中,了解了泛型的基本概念,本篇偏重于泛型的使用。主要包括:

泛型方法重载需要注意的问题
泛型的类型推断
泛型方法也可以有约束
泛型接口
泛型委托
使用EventHandler<TEventArgs>事件泛型

泛型方法重载需要注意的问题

  1. public class MyArray<T>
  2. {
  3. public T myData;
  4.  
  5. public MyArray()
  6. {
  7. myData = default(T);
  8. }
  9.  
  10. public void ShowInfo()
  11. {
  12. Console.WriteLine(myData.ToString());
  13. }
  14.  
  15. public void ShowInfo(string str)
  16. {
  17. Console.WriteLine(str);
  18. }
  19.  
  20. public void ShowInfo<T>(T data)
  21. {
  22. Console.WriteLine(data.ToString());
  23. }
  24. }

以上,说明:泛型方法可以作为方法的重载。

可以这样调用。

  1. MyArray<Student> myArray = new MyArray<Student>();
  2. myArray.ShowInfo<CollegeStudent>(new CollegeStudent());
  3. myArray.ShowInfo<string>("HelloWorld");

但还有一种情况是:两个语义不明的重载方法,在编译的时候是通过的,但在调用的时候就不会通过。比如以下在编译时没问题:

  1. public class MyArray<T>
  2. {
  3. public void ShowInfo<TA, TB>(TA a, TB b){};
  4. public void ShowInfo<TB, TA>(TA a, TB b){};
  5. }

如果这样调用,就有问题:

  1. MyArray<Student> myArray = new MyArray<Student>();
  2. myArray.showInfo<Student, Student>(new Student(), new Student());

所以,对于泛型重载方法,需要注意语义不明的情况。

泛型的类型推断

编译器可以根据方法参数的类型来推断使用哪个重载方法,优先调用一般重载方法,然后再调用泛型重载方法。

  1. myArray.ShowInfo("hello"); 会调用 ShowInfo(string str)重载方法
  2. myArray.ShowInfo(new CollegeStudent());会调用ShowInfo<T>(T data)重载方法。

泛型方法也可以有约束

我们知道泛型类可以有约束,泛型方法也一样。

  1. public void ShowInfo<T>(T data) where TData : Student
  2. {
  3. Console.WriteLine(data.ToString());
  4. }

泛型接口

.NET集合类就提供了多个泛型接口,比如:IList<T>, ICollection<T>, IComparable<>, IComparer<T>, IEnumerable<T>, IEnumerator<T>, IDictionary<TKey,TValue>,等等。

自定义类的时候,有时候需要让自定义类实现一个指定了具体类型的泛型接口:

  1. class MyClass<T> : IComparable<Int32>, IComparable<String>

泛型委托

  1. public class Generic Delegate
  2. {
  3. //声明泛型委托
  4. public delegate string MyGenericDelegate<T>(T t);
  5.  
  6. public static string GetPoint(Point p)
  7. {
  8. return stirng.Format("地址是{0},{1}", p.X, p.Y);
  9. }
  10.  
  11. public static string GetMsg(string str)
  12. {
  13. return str;
  14. }
  15. }
  16.  
  17. public static void Main()
  18. {
  19. MyGenericDelegate<string> myStrDel = new MyGenericDelegate<string>(GetMsg);
  20. Console.WriteLine(myStrDel("hello"));
  21.  
  22. MyGenericDelegate<Point> myPointDel = new MyGenericDelegate<Point>(GetPoint);
  23. Console.WriteLine(myPointDel(new Point(100, 200)));
  24. }

使用EventHandler<TEventArgs>事件泛型

它的完整定义是:

  1. public delegate void EventHandler<TEventArgs>(object sender, TEventArgs e) where TEventArgs: EventArgs

假设有一个MessageReceiver类,当建立连接时触发OnConnected事件,在接收到信息是触发OnMessageReceived事件。

在创建MessageReceiver类之前,我们先要自定义一个派生于EventArgs,且和MessageReceiver相关的类。

  1. public sealed class MessageReceivedEventArgs : EventArgs
  2. {
  3. public string Message {get;set;}
  4.  
  5. public MessageReceivedEventArgs(string msg)
  6. {
  7. this.Message = msg;
  8. }
  9. }

MessageReceiver类主要包含2个事件,一个是OnConnected事件,另一个是OnMessageReceived事件。

  1. public class MessageReceiver
  2. {
  3. public event EventHandler OnConnected;
  4. public event EventHandler<MessageReceivedEventArgs> OnMessageReceived;
  5. ...
  6.  
  7. public void DoSth()
  8. {
  9. if(OnMessageReceived != null)
  10. {
  11. OnMessageReceived(this, new MessageReceivedEventArgs(msg));
  12. }
  13. }
  14. }

以上,通过if(OnMessageReceived != null)这个判断,能保证:当没有订阅者注册事件的时候,这个事件不被触发。但在多线程场景中,这样做也不是最合理的:

假设线程A作为订阅者注册事件,正准备触发事件的时候,线程B也作为订阅者刚好在此刻注销了事件,即OnMessageReceived变成了null,这就牵累到线程A也无法触发事件。

解决办法是把事件变量赋值给一个局部变量:

  1. public class MessageReceiver
  2. {
  3. public event EventHandler OnConnected;
  4. public event EventHandler<MessageReceivedEventArgs> OnMessageReceived;
  5. ...
  6.  
  7. public void DoSth()
  8. {
  9. var handler = OnMessageReceived;
  10. if(handler != null)
  11. {
  12. handler(this, new MessageReceivedEventArgs(msg));
  13. }
  14. }
  15. }

这样,当线程A作为订阅者注册并准备触发事件的时候,及时线程B在此刻注销注册,让OnMessageReceived为null,由于已经把OnMessageReceived赋值给了局部变量handler,线程A依然能触发事件。

参考资料:
《你必须知道的.NET(第2版)》,作者王涛。

".NET泛型"系列包括:

.NET泛型01,为什么需要泛型,泛型基本语法

.NET泛型02,泛型的使用

.NET泛型03,泛型类型的转换,协变和逆变

.NET泛型04,使用Lazy<T>实现延迟加载

.NET泛型02,泛型的使用的更多相关文章

  1. day29--Java泛型02

    Java泛型02 5.自定义泛型 5.1自定义泛型类 基本语法: class 类名<T,R...>{//-表示可以有多个泛型 成员 } 注意细节: 普通成员可以使用泛型(属性.方法) 使用 ...

  2. C#2.0新增功能02 泛型

    连载目录    [已更新最新开发文章,点击查看详细] C# 语言和公共语言运行时 (CLR) 的 2.0 版本中添加了泛型. 泛型将类型参数的概念引入 .NET Framework,这样就可以设计具有 ...

  3. Scala 深入浅出实战经典 第42讲:scala 泛型类,泛型函数,泛型在spark中的广泛应用

    王家林亲授<DT大数据梦工厂>大数据实战视频 Scala 深入浅出实战经典(1-64讲)完整视频.PPT.代码下载:百度云盘:http://pan.baidu.com/s/1c0noOt6 ...

  4. Java基础---Java---基础加强---内省的简单运用、注解的定义与反射调用、 自定义注解及其应用、泛型及泛型的高级应用、泛型集合的综合

    内省的简单运用: JavaBean是一种特殊的Java类,主要用于传递数据信息,这种java类中的方法主要用于访问私有的字段,且方法名符合某种命名规则. 采用遍历BeanInfo的所有属性方式来查找和 ...

  5. JavaSE学习总结(十六)—— 泛型与泛型应用

    一.泛型概要 泛型(Generic)的本质是类型参数化,通俗的说就是用一个占位符来表示类型,这个类型可以是String,Integer等不确定的类型,表明可接受的类型. 泛型是Java中一个非常重要的 ...

  6. Java泛型:泛型的定义(类、接口、对象)、使用、继承

    地址   http://blog.csdn.net/lirx_tech/article/details/51570138 1. 设计泛型的初衷: 1) 主要是为了解决Java容器无法记忆元素类型的问题 ...

  7. 泛型学习第四天——List泛型终结:什么是List泛型,泛型筛选,泛型排序

    为什么要用泛型集合? 在C# 2.0之前,主要可以通过两种方式实现集合: a.使用ArrayList 直接将对象放入ArrayList,操作直观,但由于集合中的项是Object类型,因此每次使用都必须 ...

  8. C#高级语法之泛型、泛型约束,类型安全、逆变和协变(思想原理)

    一.为什么使用泛型? 泛型其实就是一个不确定的类型,可以用在类和方法上,泛型在声明期间没有明确的定义类型,编译完成之后会生成一个占位符,只有在调用者调用时,传入指定的类型,才会用确切的类型将占位符替换 ...

  9. 深入理解什么是Java泛型?泛型怎么使用?【纯转】

    本篇文章给大家带来的内容是介绍深入理解什么是Java泛型?泛型怎么使用?有一定的参考价值,有需要的朋友可以参考一下,希望对你们有所助. 一.什么是泛型 “泛型” 意味着编写的代码可以被不同类型的对象所 ...

随机推荐

  1. 数据库中INFORMATION_SCHEMA的说明及使用

    第一个查询看看库里有多少个表,表名等select * from INFORMATION_SCHEMA.TABLES information_schema这张数据表保存了MySQL服务器所有数据库的信息 ...

  2. plsql中做计划任务

    第一步: 1.  打开PLSQL后,选择节点jobs,右键新建,弹出界面后再what值中填写需要做计划的存储名加分号结束,如门诊收入存储PH_ClinicIncome(1):其中1代表医疗机构代码 间 ...

  3. Linux学习笔记:ps -ef、ps aux、kill -9

    一.查看进程命令 1.ps命令 Linux中的ps命令是Process Status的缩写. ps命令用来列出系统中当前运行的那些进程. ps命令列出的是当前那些进程的快照,就是执行ps命令的那个时刻 ...

  4. ES按资源类型统计个数

    一.目标:统计各类型资源的个数,输出详细报表 http://10.10.6.225:9200/dsideal_db/t_resource_info/ _mapping {  "propert ...

  5. 【Java】 参数的传递:值传递与引用传递讨论

    内容稍多,可直接看第4点的讨论结果 前言 在涉及到传递参数给方法时,容易出现一些参数传递错误的问题,这就涉及到了参数的传递问题,必须搞清楚:参数是如何传递到方法中的?一般来说,参数的传递可以分为两种: ...

  6. poj1703 Find them, Catch them(带权并查集)

    题目链接 http://poj.org/problem?id=1703 题意 有两个帮派:龙帮和蛇帮,两个帮派共有n个人(编号1~n),输入m组数据,每组数据为D [a][b]或A [a][b],D[ ...

  7. Java工具类之浮点精确计算

    public class Arith { // 默认除法运算精度 private static final int DEF_DIV_SCALE = 10; // 构造器私有,让这个类不能实例化 pri ...

  8. CentOS重装grub修复损坏的系统

    grub损坏一般有两种情况:第一.安装双系统时,后安装的系统把先安装的系统的MBR删除了.第二.误操作将grub文件删除了. 不管怎样都需要进入到救援模式,详细请看CentOS通过光盘启动救援数据 ( ...

  9. 【SQL】180. Consecutive Numbers

    Write a SQL query to find all numbers that appear at least three times consecutively. +----+-----+ | ...

  10. 详解Python中的__init__和__new__(静态方法)

    一.__init__ 方法是什么? 使用Python写过面向对象的代码的同学,可能对 __init__ 方法已经非常熟悉了,__init__ 方法通常用在初始化一个类实例的时候.例如: #-*- co ...