zoukankan      html  css  js  c++  java
  • C#高级编程笔记 Day 7, 2016年9月 19日 (泛型)

    1、协变和抗变

    • 泛型接口的协变

      如果泛型类型用 out  关键字标注,泛型接口就是协变的。这也意味着返回类型只能是 T。 接口IIndex 与类型T 是协变的,并从一个制度索引器中返回这个类型。

    •   
      1 public interface IIndex<out T>
      2 {
      3     T this[int index]{ get; }
      4     int Count{ get; }
      5

      如果对接口IIndex 使用了读写索引器,就把泛型类型T 传递给方法,并从方法中检索这个类型。这不能通过协变来实现—泛型类型必须定义为不变的。不使用out 和 in 标注,就可以把类型定义为不变的。

    • 泛型接口的抗变

      如果泛型类型用 in 关键字标注,泛型接口就是抗变的。这样,接口只能把泛型类型 T 用作其方法的输入。

    • 1 public interface IDisplay<in T>
      2 {
      3     void Show(T item);
      4 }

    2、因为可空类型使用得非常频繁,所以 C# 有一种特殊的语法,它用于定义可空类型的变量。定义这类变量时,不适用泛型结构的语法,而是用 “?”运算符。在下面的李子中,变量 x1和x2 都是可空的int 类型的实例:

    •   
      Nullable <int> x1;
      int ?x2; 

      可空类型可以与null 和数字比较,如上所示。这里,x的值与null 比较,如果x 不是 null,它就与小于 0的值比较:

    • 1 int ? x =GetNullableType();
      2 if(x==null)
      3 {
      4     Console.WriteLine("x id null");
      5 }
      6 else if(x < 0)
      7 {
      8     Console.WriteLine("x is smaller than 0");
      9 }

       知道了 Nullable<T> 是如何定义的之后,下面就使用可空类型。可空类型还可以与算术运算符一起使用。变量 x3 是 x1 和x2 的和。 如果这两个可空变量中任何一个的值是 null ,他们的和就是 null

    • 1 int ? x1=GetNullableType();
      2 int ? x2=GetNullableType();
      3 int ? x3=x1+x2;

       !这里调用的 GetNullableThype() 只是一个占位符,它对于任何方法都返回一个可空的 int 。

    3、非可空类型可以转换为可空类型。从非可空类型转换为可空类型时,在不需要强制类型转换的地方可以进行隐式转换。

    • 1 int y1=4;
      2 int ? x1=y1;

       但从可空类型转换为非可空类型可能会失败。如果可空类型的值是null,并且把null 值赋予非可空类型,就会抛出InvalidOperationException 类型的异常。这就是需要类型强制转换运算符进行显示转换的原因:

    • int ? x1=GetNullableType();
      int y1=(int) x1;

       如果不进行显示类型转换,还可以使用合并运算符从可空类型转换为非可空类型,合并运算符的语法是“??”,为转换定义了一个默认值,以防可空类型的值是null。这里,如果 x1 是 null,y1的值就是 0

    • int ? x1 = GetNullableType();
      int y1 = x1 ?? 0

    4、泛型方法:

      除了定义泛型类之外,还可以定义泛型方法。在泛型犯法中,泛型类型用方法声明来定义。泛型方法可以在非泛型类中定义。

    • void Swap<T>(ref T x,ref T y)
      {
          T temp;
          temp =x;
          x=y;
          y= temp;
      }

      把泛型类型赋予方法调用,就可以调用泛型方法:

    • int i=4;
      int j=5;
      Swap<int>(ref i,ref j);
      //或者 Swap(ref i,ref j);

       泛型方法实例,下面的例子使用泛型方法累加集合中的所有元素。为了说明泛型方法的功能,下面使用包含Name 和Balance 属性的Account 类

    •  1 public class Account
       2 {
       3     public string Name{ get; private set;}
       4     public decimal Balance{get; private set;}
       5     
       6     public Account(string name,Decimal balance)
       7     {
       8         this.Name=name;
       9         this.Balance=balance;
      10     }
      11 }

       其中应累加余额的所有账户操作都添加到 List<Account> 类型的账户列表中

    • 1 var accounts=new List<Account>()
      2 {
      3     new Account("Christian",1500),
      4     new Account("Stephanie",2200),
      5     new Account("Angela",1800),
      6     new Account("Matthias",2400)
      7 };

       累加所有Account对象的传统方式是用foreach 语句遍历所有的Account 对象,如下所示,foreach 语句使用 IEnumerable 接口迭代集合的元素,所以 AccumulateSimple() 方法的参数是 IEnumerable类型。foreach语句处理实现 IEunmerable 接口的每个对象。这样,AccumulateSimple() 方法就可以用于所有实现 IEnumerable<Account> 接口的集合类。在这个方法的实现代码中,直接访问Account 对象的 Balance 属性。

    •  1 public static class Algorithm
       2 {
       3     public static decimal AccumulateSimple(IEnumerable<Account> source)
       4     {
       5         decimal sum=0;
       6         foreach(Account a in source)
       7         {
       8             sum+=a.Balance;
       9         }
      10         return sum;
      11     }
      12 }

       调用如下:

    • decimal amount=Algorthm.AccumulateSimple(accounts);

       带约束的泛型方法

        第一个实现代码的问题是,它只能用于Account 对象。使用泛型方法就可以避免这个问题。

        Accumulate() 方法的第二个版本接受实现了 IAccount 接口的任意类型。如前面的泛型类所述,泛型类型可以用where 子句来限制。

      •  1 public static decimal Accmulate<TAccount>(IEnumerable<TAccount> source) where TAccount : IAccount
         2 {
         3     decimal sum=0;
         4 
         5     foreach(TAccount a in source)
         6     {
         7         sum+= a.Balance;
         8     }
         9     return sum;
        10 }

         重构Account 类实现 IAccount 接口

      • public class Account : IAccount
        {
            //...

         IAccount 接口定义了只读属性 Balance 和 Name 

      • 1 public interface IAccount
        2 {
        3     decimal Balance {get; }
        4     string Name{get ;}
        5 }

         新的调用方式:

      • decimal amount =Algorthim.Accmulate<Account>(accounts);
        //或者 decimail amount =Alhorthim.Accmulate(accounts);

        带委托的泛型方法

         泛型类型实现了 IAccount 接口的要求过于严格。下面的示例提示了,如何通过传递一个泛型委托来修改 Accumulate()方法。这个Accumulate() 方法使用两个泛型参数 T1 和 T2。第一个参数T1 用于实现了IEnumerable<T1> 参数的集合,第二个参数使用泛型委托Fun<T1, T2, TResult>。 其中, 第二个和第三个泛型参数都是 T2 类型。需要传递的方法有两个输入参数(T1 和 T2)和一个 T2类型的返回值。

      • 1 public static T2 Accmulate<T1,T2>(IEnumerable<T1> source, Func<T1,T2,T2> action)
        2 {
        3     T2 sum=default(T2);
        4     foreach(T1 item in source)
        5     {
        6         sum=action(item,sum);
        7     }
        8     return sum;
        9 }

         在调用这个方法时,需要指定泛型参数类型,因为编译器不能自动推断出该类型。对于方法的第一个参数,所赋予的accounts集合是IEnumerable<Account> 类型。对于第二个参数,使用一个 Lambda 表达式来定义Account 和 decimal 类型的两个参数,返回一个小数。对于每一项通过Accumulate() 方法调用这个 Lambda 表达式

      • decimal amount=Algorithm.Accumulate<Account,decimal>(accounts,(item,sum)=>sum+=item.Balance);

         泛型方法规范

        泛型方法可以重载,为特定的类型定义规范。这也适用于带泛型参数的方法。Foo()方法定义了两个版本,第一个版本接受一个泛型参数,第二个版本是用于int 参数的专有版本。在编译期间,会使用最佳匹配。如果传递了一个int,就选择带int 参数的方法。对于任何其他参数类型,编译器会选择方法的泛型版本

      •  1 public class MethodOverloads
         2 {
         3     public void Foo<T>(T obj)
         4     {
         5         Console.WriteLine("Foo<T>(T obj),obj type :{0}",obj.GetType().Name);
         6     }
         7     public void Foo(int x)
         8     {
         9         Console.WriteLine("Foo(int x)");
        10     }
        11     public void Bar<T>(T obj)
        12     {
        13         Foo(obj);
        14     }
        15 }

         Foo()方法现在可以通过任意参数类型来调用。下面的示例代码给该方法传递了一个 int 和一个 string

      • 1 static void Main()
        2 {
        3     var test =new MethodOverloads();
        4     test.Foo(33);
        5     test.Foo("abc");
        6 }

         运行该程序,可以从输出中看出选择了最佳匹配的方法:

      • Foo(int x)
        Foo<T>(T obj), obj type: String

         需要注意的是,所调用的方法是在编译期间定义的,而不是运行期间。这很容易举例说明:添加一个调用Foo() 方法的Bar() 泛型方法,并传递泛型参数值:

      • public class MethodOverloads
        {
            //...
            
            public void Bar<T> (T obj)
            {
                Foo(obj);
            }
        }

         Main()方法现在改为调用传递以个 int 值 的Bar()方法:

      • static void Main()
        {
            vat test=new MethodOverloads();
            test.Bar(44);

         从控制台的输出可以看出,Bar()方法选择了泛型Foo()方法,而不是用 int参数重载的Foo()方法。原因是编译器是在编译期间选择Bar() 方法调用的Foo() 方法。由于Bar() 方法定义了一个泛型参数,而且泛型Foo()方法匹配这个类型,所以调用了Foo() 方法。在运行期间给Bar() 方法传递一个 int 值不会改变这一点。

      • Foo<T>(T obj), obj type: Int32

     【小结】:泛型。通过泛型类可以创建独立于类型的类,泛型方法是独立于类型的方法。接口、结构、和委托也可以用泛型的方式创建。泛型引入了一种新的编程方式。我们介绍了如何实现相应的算法(尤其是操作和谓词)以用于不同的类,而且他们呢都是类型安全的。泛型委托可以去除集合中的算法。

  • 相关阅读:
    hadoop shell 命令
    java正则提取括号中的关键词
    java使用ac算法实现关键词高亮
    mysql事务级别和spring中应用
    elasticsearch java工具类
    【记录】研究生周练题目清单
    【记录】研究生已阅文献清单
    论文阅读(11)RoBERTa: A Robustly Optimized BERT Pretraining Approach(2019)
    论文阅读(10)Shallow Convolutional Neural Network for Implicit Discourse Relation Recognition
    论文阅读(9)Towards Cross-Domain PDTB-Style Discourse Parsing(2014)
  • 原文地址:https://www.cnblogs.com/xiyin/p/5884990.html
Copyright © 2011-2022 走看看