泛型程序设计详解(一)

前言

泛型是C#和.Net的一个重要概念,泛型不仅是C#编程语言中的一部分,而且与程序集中的IL(Intermediate Language)代码紧密的集成。

在平时编程过程中,常常会出现编写一些差不多的代码块,不同的仅是处理不同的数据类型。比如一个处理int数据的方法,现在新加了string类型的数据。是不是把之前的方法复制一遍,然后修改类型int为string。当然这样的方法是没有错的,那么后面又新增了其他的许多类型怎么办?还是复制修改吗?这样代码看上去很冗余,很复杂。这时候泛型就出现了。下面我们看下为何使用泛型吧。

优点

下面介绍下泛型的优点,尤其是下面几个:

  • l 性能
  • l 类型安全
  • l 二进制代码重用

一、性能

    泛型的一个主要优点就是性能,在泛型集合类和非泛型集合类中,对值类型使用非泛型集合类,在把值类型转换为引用类型和把引用类型转换为值类型的时候,需要进行装箱和拆箱的操作(前面的文章中讲到了拆箱和装箱会造成一定的性能损失),当集合数据量大的时候造成的性能损失也就随之的增大了。

使用非泛型集合时:

 var list = new ArrayList();
            list.Add(100);//装箱  int-object
            int i = (int)list[0];//拆箱  object-int
            foreach (int item in list)
            {
                Console.WriteLine(item);//遍历拆箱输出
            } 

使用泛型集合类时:

var list = new List<int>();
            list.Add(100);//无装箱操作 
            int i =  list[0];//无拆箱拆箱 
            foreach (int item in list)
            {
                Console.WriteLine(item);//无拆箱操作
            } 

减少装箱拆箱操作,节省了性能消耗。这也就是泛型的主要优点了。

二、类型安全

泛型另一个优点就是类型安全,这里我们还是使用非泛型集合类ArrayList()和泛型集合类List<T>来做案例。

非泛型集合类ArrayList():

 var list = new ArrayList();
            list.Add(100);// 添加一个int类型
            list.Add("string");//添加一个string类型
            foreach (int item in list)
            {
                Console.WriteLine(item);//遍历循环输出
 }

这里允许输出和抛出异常:

System.InvalidCastException:“Unable to cast object of type 'System.String' to type 'System.Int32'.” 

无法强制把”string”转换成int类型。

我们再看泛型集合类:

            list.Add(100);// 添加一个int类型 list.Add("string");//添加一个string类型,编译器报错,无法从string转换到int
            foreach (int item in list)
            {
                Console.WriteLine(item);//遍历循环输出 }

在添加”string”类型的时候编译器报错,无法添加。这里也就杜绝了后续的错误。这也就是保证了类型的安全。

三、二进制代码重用

泛型允许更好的重用二进制代码,泛型类型可以定义一次,并且可以再许多不同的类型实例化,相比C++来说,不用每次访问源代码。

例如上面使用的泛型集合类,using System.Collections.Generic; 中的List<T>类,可以用int,string,自定义类去实例化。

泛型类型还可以在一种语言定义,然后再其他任何.Net语言中使用。

泛型类的功能

这里我们可以来了解下创建泛型类了之后,泛型类有哪些功能呢?

  • l 默认值
  • l 约束
  • l 继承
  • l 静态成员

一、默认值

在我们定义了泛型类型之后如何赋值呢?

public class Tclass<T> { public static T Get()
        {
            T a = default; return a;
        }

    }

因为在泛型中初始给值不好给,你说给null吧,null是给引用类型的,你是给0吧,这又是给值类型的,这时候出现了default,当时引用类型调用时就给null,当时值类型时就0。

二、约束

说到泛型类型的约束时,不得不提关键字where,where是用来限制参数的范围的,如果不符合where条件指定的参数范围,编译是不允许通过的。

这里泛型类型的约束主要可以分为以下6中

  • l Where T: class(类型参数必须是引用类型)
  • l Where T:struct(类型参数必须是值类型)
 public class Tclass<T,U>
        where T:class //类型参数为引用类型
        where U:struct  //类型参数为值类型
             {} 
    • l Where T:<接口名称>(类型参数必须是指定的接口或者实现指定的接口)
/// <summary>
    /// 接口 /// </summary>
    interface Itest
    {
    } /// <summary>
    /// 定义一个字典类型 /// </summary>
    /// <typeparam name="TK"></typeparam>
    /// <typeparam name="TV"></typeparam>
    class Dictionary<TK, TV>
      where TK : IComparable, IEnumerable where TV : Itest
    { public void Add(TK key, TV val)
        {
        }
    }
    • l Where T:<基类名>(参数必须是指定的基类或者是派生自指定的基类)
class Ttest { } class Tclass<T> where T:Ttest
    {

    }
    • l Where T:new ()(这是一个构造函数的约束,指定参数类型必须有一个默认构造函数,当与其他约束一起使用时必须放在其最后)
  class EmployeeList<T> where T : Employee, IEmployee, System.IComparable<T>, new()
  {
      // ... }
    • l Where T1:T2(这个约束指定类型T1派生自泛型类型T2,也就是说T1的参数类型要和T2一样)

<pre style="color: rgb(0, 0, 0); font-family: "Courier New"; font-size: 12px; margin: 5px 8px; padding: 5px;">  public class Tclass<T> where T:IComparable { }</pre>

三、继承

泛型类型的继承与普通类的继承相似但不同。

 /// <summary>
    /// 抽象基类,泛型类型 /// </summary>
    /// <typeparam name="T"></typeparam>
    public abstract class Ttest<T> { public abstract T Add(T x, T y);
    } /// <summary>
   /// 继承抽象基类,实现int类型 /// </summary>
   /// <typeparam name="T"></typeparam>
    class Tclass<T> : Ttest<int> { public override int Add(int x, int y) => x + y;  

    }

四、静态成员

泛型类型的静态成员需要特殊的关注,泛型类的静态成员只能在类的一个实例中共享。

 /// <summary>
    /// 泛型类型,静态字段x /// </summary>
    /// <typeparam name="T"></typeparam>
    public class Ttest<T> { public static int x;
    } class Program
    { static void Main(string[] args)
        {
            Ttest<string>.x = 111;
            Ttest<int>.x = 222;
            Console.WriteLine(Ttest<string>.x);
        }

    }

上面事例中最后输出的为111,

总结

这里我们主要是介绍了泛型的优点及泛型类型的功能。在我们日常的编程中会发现很多地方可以使用泛型。提高代码的扩展性及重用性。同时也可以减少对object类型的使用,采用泛型类型的使用来替代。较少对性能的消耗。我们下一节主要是对泛型类型的协变及抗变进行一定的理解。


只要认为是对的就去做,坚持去做,不要在乎别人的看法,哪怕是错,至少你有去做过证明曾经你努力过。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,009评论 5 474
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,808评论 2 378
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 148,891评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,283评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,285评论 5 363
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,409评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,809评论 3 393
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,487评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,680评论 1 295
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,499评论 2 318
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,548评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,268评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,815评论 3 304
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,872评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,102评论 1 258
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,683评论 2 348
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,253评论 2 341

推荐阅读更多精彩内容

  • object 变量可指向任何类的实例,这让你能够创建可对任何数据类型进程处理的类。然而,这种方法存在几个严重的问题...
    CarlDonitz阅读 904评论 0 5
  • C# 泛型(Generics) 泛型概述 泛型是C#编程语言的一部分,它与程序集中的IL(Intermediate...
    OctOcean阅读 2,228评论 0 4
  • 本章将会介绍 泛型所解决的问题泛型函数类型参数命名类型参数泛型类型扩展一个泛型类型类型约束关联类型泛型 Where...
    寒桥阅读 626评论 0 2
  • 泛型(Generics) 泛型代码允许你定义适用于任何类型的,符合你设置的要求的,灵活且可重用的 函数和类型。泛型...
    果啤阅读 661评论 0 0
  • 恍惚、晕眩、眼前一片漆黑,这就是叶文现在的感觉,他甚至感觉不到自己的身体和四肢,就好像他成了一个严重的瘫痪病人一...
    独孤寞如雪阅读 289评论 0 0