编写高质量代码改善C#程序的157个建议——建议42:使用泛型参数兼容泛型接口的不可变性

建议42:使用泛型参数兼容泛型接口的不可变性

让返回值类型返回比声明的类型派生程度更大的类型,就是“协变”。如:

        public Employee GetAEmployee(string name)
        {
            Console.WriteLine("我是雇员:"+name);
            return new Programmer() { Name = name };//Programmer是Employee的子类
        }

Programmer是Employee的子类,所以Programmer对象也是Employee对象。方法GetAEmployee返回一个Programmer的对象,也就是相当于返回一个Employee对象。

由于协变是一种如此自然的应用,我们很可能写出如下代码:

    class Program
    {
        static void Main(string[] args)
        {
            ISalary<Programmer> s = new BaseSalaryCounter<Programmer>();
            PrintSalary(s);
        }

        static void PrintSalary(ISalary<Employee> s)
        {
            s.Pay();
        } } interface ISalary<T> { void Pay(); } class BaseSalaryCounter<T> : ISalary<T> { public void Pay() { Console.WriteLine("Pay base salary"); } } class Employee { public string Name { get; set; } } class Programmer : Employee { } class Manager : Employee { }

在PrintSalary这个方法中,方法接收的类型是ISalary<Employee>。于是,我们想当然的认为ISalary<Programmer>必然也可以被PrintSalary方法接收的。事实却不然,代码编译会通不过:

无法从“MyTest.ISalary<MyTest.Programmer>”转换为“MyTest.ISalary<MyTest.Employee>”

编译器对于接口和委托类型参数的检查是非常严格的,除非用关键字out特别声明,不然这段代码只会编译失败。要让PrintSalary完成需求,我们可以使用泛型类型参数:

        static void PrintSalary<T>(ISalary<T> s)
        {
            s.Pay();
        }

注意:建议开头指出“协变”是针对返回值而言的,但是所举的这个例子并没有体现“返回值”这个概念。实际上,只有泛型类型参数在一个接口声明中不被用来作为方法的输入参数,我们就姑且把它看成是“返回值”类型的。所以,本建议中这种模式是满足“协变”定义的。但是,只要将T作为输入参数,就不满足“协变”定义了。

转自:《编写高质量代码改善C#程序的157个建议》陆敏技

原文地址:https://www.cnblogs.com/jesselzj/p/4733120.html