C#泛型实例详解

本文以实例形式讲述了C#泛型的用法,有助于读者深入理解C#泛型的原理,具体分析如下:

首先需要明白什么时候使用泛型:

当针对不同的数据类型,采用相似的逻辑算法,为了避免重复,可以考虑使用泛型。

一、针对类的泛型

针对不同类型的数组,写一个针对数组的"冒泡排序"。

1.思路

● 针对类的泛型,泛型打在类旁。 
● 由于在"冒泡排序"中需要对元素进行比较,所以泛型要约束成实现IComparable接口。

 1 class Program
 2 {
 3   static void Main(string[] args)
 4   {
 5     SortHelper<int> isorter = new SortHelper<int>();
 6     int[] iarray = {8, 7, 1, 2, 12};
 7     isorter.BubbleSort(iarray);
 8     foreach (int item in iarray)
 9     {
10       Console.Write(item+ ", ");
11     }
12     Console.ReadKey();
13   }
14 }
15
16 public class SortHelper<T> where T : IComparable
17 {
18   public void BubbleSort(T[] array)
19   {
20     int length = array.Length;
21     for (int i = 0; i <= length -2; i++)
22     {
23       for (int j = length - 1; j >= 1; j--)
24       {
25         if (array[j].CompareTo(array[j-1]) < 0)
26         {
27           T temp = array[j];
28           array[j] = array[j - 1];
29           array[j - 1] = temp;
30         }
31       }
32     }
33   }
34 }

运行结果如下图所示:

2.关于泛型约束

where T : IComparable 把T约束为实现IComparable接口 
where T : class 
where T : struct 
where T : IComparable, new() 约束泛型必须有构造函数

3.关于冒泡算法

● 之所以for (int i = 0; i <= length -2; i++),这是边界思维,比如有一个长度为5的数组,如果0号位元素最终调换到4号位,每次调一个位,需要经过4次才能到4号位,即for(int i = 0; i <= 5-2, i++),i依次为0, 1, 2, 4,期间经历了4次。

● 至于for (int j = length - 1; j >= 1; j--)循环,即遍历从最后一个元素开始到索引为1的元素,每次与前一个位置上的元素比较。

4.关于比较

int类型之所以能比较,是因为int类型也实现了IComparable接口。

byte类型也一样实现了IComparable接口。

二、自定义一个类,使之也能实现冒泡算法

冒泡算法涉及到元素比较,所以自定义类必须实现IComparable接口。

 1 class Program
 2 {
 3   static void Main(string[] args)
 4   {
 5     Book[] bookArray = new Book[2];
 6     Book book1 = new Book(100, "书一");
 7     Book book2 = new Book(80, "书二");
 8     bookArray[0] = book1;
 9     bookArray[1] = book2;
10
11     Console.WriteLine("冒泡之前:");
12     foreach (Book b in bookArray)
13     {
14       Console.WriteLine("书名:{0},价格:{1}", b.Title, b.Price);
15     }
16
17     SortHelper<Book> sorter = new SortHelper<Book>();
18     sorter.BubbleSort(bookArray);
19     Console.WriteLine("冒泡之后:");
20     foreach (Book b in bookArray)
21     {
22       Console.WriteLine("书名:{0},价格:{1}", b.Title, b.Price);
23     }
24     Console.ReadKey();
25   }
26 }
27
28 public class SortHelper<T> where T : IComparable
29 {
30   public void BubbleSort(T[] array)
31   {
32     int length = array.Length;
33     for (int i = 0; i <= length -2; i++)
34     {
35       for (int j = length - 1; j >= 1; j--)
36       {
37         if (array[j].CompareTo(array[j-1]) < 0)
38         {
39           T temp = array[j];
40           array[j] = array[j - 1];
41           array[j - 1] = temp;
42         }
43       }
44     }
45   }
46 }
47
48 //自定义类实现IComparable接口
49 public class Book : IComparable
50 {
51   private int price;
52   private string title;
53
54   public Book(){}
55
56   public Book(int price, string title)
57   {
58     this.price = price;
59     this.title = title;
60   }
61
62   public int Price
63   {
64     get { return this.price; }
65   }
66
67   public string Title
68   {
69     get { return this.title; }
70   }
71
72   public int CompareTo(object obj)
73   {
74     Book book = (Book)obj;
75     return this.Price.CompareTo(book.Price);
76   }
77 }

运行结果如下图所示:

三、针对方法的泛型

继续上面的例子,自定义一个类,并定义泛型方法。

 1 //方法泛型
 2 public class MethodSortHelper
 3 {
 4   public void BubbleSort<T>(T[] array) where T : IComparable
 5   {
 6     int length = array.Length;
 7     for (int i = 0; i <= length - 2; i++)
 8     {
 9       for (int j = length - 1; j >= 1; j--)
10       {
11         if (array[j].CompareTo(array[j - 1]) < 0)
12         {
13           T temp = array[j];
14           array[j] = array[j - 1];
15           array[j - 1] = temp;
16         }
17       }
18     }
19   }
20 }

主程序如下:

 1 class Program
 2 {
 3   static void Main(string[] args)
 4   {
 5     Book[] bookArray = new Book[2];
 6     Book book1 = new Book(100, "书一");
 7     Book book2 = new Book(80, "书二");
 8     bookArray[0] = book1;
 9     bookArray[1] = book2;
10
11     Console.WriteLine("冒泡之前:");
12     foreach (Book b in bookArray)
13     {
14       Console.WriteLine("书名:{0},价格:{1}", b.Title, b.Price);
15     }
16
17     MethodSortHelper sorter = new MethodSortHelper();
18     sorter.BubbleSort<Book>(bookArray);
19     Console.WriteLine("冒泡之后:");
20     foreach (Book b in bookArray)
21     {
22       Console.WriteLine("书名:{0},价格:{1}", b.Title, b.Price);
23     }
24     Console.ReadKey();
25   }
26 } 

运行结果如下图所示:

另外,使用泛型方法的时候,除了按以下:

1 MethodSortHelper sorter = new MethodSortHelper(); 2 sorter.BubbleSort<Book>(bookArray);

还可以这样写:

1 MethodSortHelper sorter = new MethodSortHelper(); 2 sorter.BubbleSort(bookArray);

可见,泛型方法可以根据数组实例隐式推断泛型是否满足条件。

四、泛型的其它优点

1.避免隐式装箱和拆箱

以下包含隐式装箱和拆箱:

 1 ArrayList list = new ArrayList();
 2 for(int i = 0; i < 3; i++)
 3 {
 4   list.Add(i); //Add接收的参数类型是引用类型object,这里包含了隐式装箱
 5 }
 6 for(int i = 0; i < 3; i++)
 7 {
 8   int value = (int)list[i]; //引用类型强转成值类型,拆箱
 9   Console.WriteLine(value);
10 }

使用泛型避免隐式装箱和拆箱:

 1 List<int> list = new List<int>();
 2 for(int i = 0; i < 3; i++)
 3 {
 4   list.Add(i);
 5 }
 6 for(int i = 0; i < 3; i++)
 7 {
 8   int value = list[i];
 9   Console.WriteLine(value);
10 }

2.能在编译期间及时发现错误

不使用泛型,在编译期不会报错的一个例子:

ArrayList list = new ArrayList();
int i = 100;
list.Add(i);
string value = (string)list[0];

  使用泛型,在编译期及时发现错误:

List<int> list = new List<int>();
int i = 100;
list.Add(i);
string value = (string)list[0];

五、使用泛型的技巧

1.在当前文件中给泛型取别名

using IntList = List<int>;
IntList list = new IntList();
list.Add(1);

2.在不同文件中使用泛型别名,定义一个类派生于泛型

public class IntList : List<int>{}

  

时间: 2025-01-04 02:26:27

C#泛型实例详解的相关文章

java 泛型实例详解(普通泛型、 通配符、 泛型接口)

java 泛型详解(普通泛型. 通配符. 泛型接口) 2013-02-04 19:49:49| 分类: JAVA | 标签:java |举报|字号 订阅 下载LOFTER客户端 JDK1.5 令我们期待很久,可是当他发布的时候却更换版本号为5.0.这说明Java已经有大幅度的变化.本文将讲解JDK5.0支持的新功能-----Java的泛型. 1.Java泛型 其实Java的泛型就是创建一个用类型作为参数的类.就象我们写类的方法一样,方法是这样的method(String str1,String

C#泛型Dictionary的用法实例详解

本文以实例形式讲述了C#中的泛型Dictionary的用法.具有很好的实用价值.分享给大家供大家参考.具体如下: 泛型最常见的用途是泛型集合,命名空间System.Collections.Generic 中包含了一些基于泛型的集合类,使用泛型集合类可以提供更高的类型安全性,还有更高的性能,避免了非泛型集合的重复的装箱和拆箱. 很多非泛型集合类都有对应的泛型集合类,下面是常用的非泛型集合类以及对应的泛型集合类: 非泛型集合类 泛型集合类 ArrayList List<T> HashTable D

Spring基于事件驱动模型的订阅发布模式代码实例详解

代码下载地址:http://www.zuidaima.com/share/1791499571923968.htm 原文:Spring基于事件驱动模型的订阅发布模式代码实例详解 事件驱动模型简介 事件驱动模型也就是我们常说的观察者,或者发布-订阅模型:理解它的几个关键点: 首先是一种对象间的一对多的关系:最简单的如交通信号灯,信号灯是目标(一方),行人注视着信号灯(多方): 当目标发送改变(发布),观察者(订阅者)就可以接收到改变: 观察者如何处理(如行人如何走,是快走/慢走/不走,目标不会管的

C#处理文本文件TXT实例详解

本文实例讲述了C#处理文本文件TXT的方法.分享给大家供大家参考.具体分析如下: 1. 如何读取文本文件内容: 这里介绍的程序中,是把读取的文本文件,用一个richTextBox组件显示出来.要读取文本文件,必须使用到"StreamReader"类,这个类是由名字空间"System.IO"中定义的.通过"StreamReader"类的"ReadLine()"方法,就可以读取打开数据流当前行 的数据了.下面代码实现的功能就是读取

23、磁盘管理—磁盘阵列(RAID)实例详解

磁盘阵列(RAID)实例详解 raid技术分类 软raid技术 硬raid技术 Raid和lvm的区别 为什么选择用raid RAID详解 RAID-0 RAID-1 RAID-5 Raid-10 Raid的管理 案例:创建一个raid10+冗余盘 磁盘阵列(RAID)实例详解 Raid(磁盘阵列)级别介绍 Raid有"廉价磁盘冗余阵列"的意思,就是利用多块廉价的硬盘组成磁盘组,让数据分部储存在这些硬盘里面,从而达到读取和写入加速的目的:也可以用作数据的冗余,当某块硬盘损毁后,其他硬盘

Cocos2d-x 3.X手游开发实例详解

Cocos2d-x 3.X手游开发实例详解(最新最简Cocos2d-x手机游戏开发学习方法,以热门游戏2048.卡牌为例,完整再现手游的开发过程,实例丰富,代码完备,Cocos2d-x作者之一林顺和泰然网创始人杨雍力荐) 于浩洋 著   ISBN 978-7-121-23998-4 2014年9月出版 定价:59.00元 356页 16开 编辑推荐 以Cocos2d-x V3.0为框架全面讲解手游开发的知识和方法 以热门游戏2048.卡牌为例,完整再现手游的开发过程 Cocos2d-x作者之一林

实例详解Django的 select_related 和 prefetch_related 函数对 QuerySet 查询的优化(一)

在数据库有外键的时候,使用 select_related() 和 prefetch_related() 可以很好的减少数据库请求的次数,从而提高性能.本文通过一个简单的例子详解这两个函数的作用.虽然QuerySet的文档中已经详细说明了,但本文试图从QuerySet触发的SQL语句来分析工作方式,从而进一步了解Django具体的运作方式. 本来打算写成一篇单独的文章的,但是写完select_related()之后发现长度已经有点长了,所以还是写成系列,大概在两到三篇.整个完成之后将会在这里添加上

boost asio异步读写网络聊天程序客户端 实例详解

// // chat_client.cpp // ~~~~~~~~~~~~~~~ // // Copyright (c) 2003-2013 Christopher M. Kohlhoff (chris at kohlhoff dot com) // // Distributed under the Boost Software License, Version 1.0. (See accompanying // file LICENSE_1_0.txt or copy at http://ww

《HTML 5网页开发实例详解》目录

第一篇  从宏观上认识HTML 5 讲述了HTML 5引发的Web革命.HTML 5的整体特性.HTML 5相关概念和框架和开发环境搭建. 第1章 HTML 5引发的Web革命 1.1  你是不是真的了解HTML 5 1.1.1  通过W3C认识HTML 5的发展史 1.1.2  HTML 5.HTML4.XHTML的区别 1.1.3  什么人应该学HTML 5 1.1.4  一个图告诉你如何学习HTML 5 1.2  浏览器之争 1.2.1  说说这些常见的浏览器 1.2.2  浏览器的兼容烦