.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泛型02,泛型的使用

时间: 2024-10-17 14:20:39

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

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

为什么要用泛型集合? 在C# 2.0之前,主要可以通过两种方式实现集合: a.使用ArrayList 直接将对象放入ArrayList,操作直观,但由于集合中的项是Object类型,因此每次使用都必须进行繁琐的类型转换. b.使用自定义集合类 比较常见的做法是从CollectionBase抽象类继承一个自定义类,通过对IList对象进行封装实现强类型集合.这种方式要求为每种集合类型写一个相应的自定义类,工作量较大.泛型集合的出现较好的解决了上述问题,只需一行代码便能创建指定类型的集合. 什么是泛

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

王家林亲授<DT大数据梦工厂>大数据实战视频 Scala 深入浅出实战经典(1-64讲)完整视频.PPT.代码下载:百度云盘:http://pan.baidu.com/s/1c0noOt6 腾讯微云:http://url.cn/TnGbdC 360云盘:http://yunpan.cn/cQ4c2UALDjSKy 访问密码 45e2 技术爱好者尤其是大数据爱好者 可以加DT大数据梦工厂的qq群 DT大数据梦工厂① :462923555 DT大数据梦工厂②:437123764 DT大数据梦工厂③

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

地址   http://blog.csdn.net/lirx_tech/article/details/51570138 1. 设计泛型的初衷: 1) 主要是为了解决Java容器无法记忆元素类型的问题: i. 由于Java设计之初并不知道会往容器中存放什么类型的元素,因此元素类型都设定为Object,这样就什么东西都能放了! ii. 但是这样设计有明显的缺点: a. 取出元素的时候必须进行强制类型转换(尽管集合在运行时里面元素的"运行时类型"不变,即元素的getClass返回的还是最初

什么是泛型?泛型的基本原理与使用优势。

1. 什么是泛型? 泛型将接口的概念进一步延伸,"泛型"的字面意思就是广泛的类型.类.接口和方法代码可以应用于非常广泛的类型,代码与它们能够操作的数据类型不再绑定在一起,同一套代码可以用于多种数据类型,这样不仅可以复用代码,降低耦合性,而且还提高了代码的可读性以及安全性.讲起来优点抽象,我们看个实际的例子. 2. 先来看一个简单的泛型例子 package genericity.demo; /** * @author BastetCat * @data 2019/8/8 21:14 */

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

一.为什么使用泛型? 泛型其实就是一个不确定的类型,可以用在类和方法上,泛型在声明期间没有明确的定义类型,编译完成之后会生成一个占位符,只有在调用者调用时,传入指定的类型,才会用确切的类型将占位符替换掉. 首先我们要明白,泛型是泛型,集合是集合,泛型集合就是带泛型的集合.下面我们来模仿这List集合看一下下面这个例子: 我们的目的是要写一个可以存放任何动物的集合,首先抽象出一个动物类: //动物类 public class Animal { //随便定义出一个属性和方法 public Strin

Java重要技术(19)泛型之泛型的使用方法

1.1. 泛型的用法 参数化类型比如Sample<T>中的类型参数T可以用于构造函数的参数的类型,方法的返回值的类型,画着方法的参数的类型,属性字段的类型等. public class GenericParameterTest { //参数化类型T,可以为String,Integer等引用类型. static class Sample<T> { public Sample(T obj){ this.obj = obj; } //参数化类型作为返回值的类型. public T wor

C++泛型 &amp;&amp; Java泛型实现机制

C++泛型 C++泛型跟虚函数的运行时多态机制不同,泛型支持的静态多态,当类型信息可得的时候,利用编译期多态能够获得最大的效率和灵活性.当具体的类型信息不可得,就必须诉诸运行期多态了,即虚函数支持的动态多态. 对于C++泛型,每个实际类型都已被指明的泛型都会有独立的编码产生,也就是说list<int>和list<string>生成的是不同的代码,编译程序会在此时确保类型安全性.由于知道对象确切的类型,所以编译器进行代码生成的时候就不用运用RTTI,这使得泛型效率跟手动编码一样高.

Java基础 -- 泛型之泛型参数

泛型机制常用的参数有3个: “?”代表任意类型.如果只指定了<?>,而没有extends,则默认是允许任意类. extends关键字声明了类型的上界,表示参数化的类型可能是所指定的类型,或者是此类型的子类. super关键字声明了类型的下界,表示参数化的类型可能是所指定的类型,或者是此类型的父类型,直至Object 前提 Fruit是Apple和Orange的超类. 本章通过java代码逐一分析泛型参数的意义和区别 extends参数: [java] public void extend(Li

Swift泛型和泛型函数

泛型(generic)可以使我们在程序代码中定义一些可变的部分,在运行的时候指定.使用泛型可以最大限度地重用代码.保护类型的安全以及提高性能.在Swift集合类中,已经采用了泛型.一.一个问题的思考怎样定义一个函数来判断两个参数是否相等呢?如果参数是Int类型,则函数定义如下:func isEqualsInt(a:Int, b:Int) -> Bool {    return (a == b)}这个函数参数列表是两个Int类型,它只能比较两个Int类型参数是否相等.如果我们想比较两个Double