C# 泛型(Generic)

泛型(Generic) 允许您延迟编写类或方法中的编程元素的数据类型的规范,直到实际在程序中使用它的时候。换句话说,泛型允许您编写一个可以与任何数据类型一起工作的类或方法。

您可以通过数据类型的替代参数编写类或方法的规范。当编译器遇到类的构造函数或方法的函数调用时,它会生成代码来处理指定的数据类型。下面这个简单的实例将有助于您理解这个概念:

  1. using System;
  2. using System.Collections.Generic;
  3. namespace GenericApplication
  4. {
  5. public class MyGenericArray<T>
  6. {
  7. private T[] array;
  8. public MyGenericArray(int size)
  9. {
  10. array = new T[size + 1];
  11. }
  12. public T getItem(int index)
  13. {
  14. return array[index];
  15. }
  16. public void setItem(int index, T value)
  17. {
  18. array[index] = value;
  19. }
  20. }
  21. class Tester
  22. {
  23. static void Main(string[] args)
  24. {
  25. // 声明一个整型数组
  26. MyGenericArray<int> intArray = new MyGenericArray<int>(5);
  27. // 设置值
  28. for (int c = 0; c < 5; c++)
  29. {
  30. intArray.setItem(c, c*5);
  31. }
  32. // 获取值
  33. for (int c = 0; c < 5; c++)
  34. {
  35. Console.Write(intArray.getItem(c) + " ");
  36. }
  37. Console.WriteLine();
  38. // 声明一个字符数组
  39. MyGenericArray<char> charArray = new MyGenericArray<char>(5);
  40. // 设置值
  41. for (int c = 0; c < 5; c++)
  42. {
  43. charArray.setItem(c, (char)(c+97));
  44. }
  45. // 获取值
  46. for (int c = 0; c < 5; c++)
  47. {
  48. Console.Write(charArray.getItem(c) + " ");
  49. }
  50. Console.WriteLine();
  51. Console.ReadKey();
  52. }
  53. }
  54. }

当上面的代码被编译和执行时,它会产生下列结果:

  1. 0 5 10 15 20
  2. a b c d e

一、泛型(Generic)的特性

使用泛型是一种增强程序功能的技术,具体表现在以下几个方面:

  • 它有助于您最大限度地重用代码、保护类型的安全以及提高性能。

  • 您可以创建泛型集合类。.NET 框架类库在 System.Collections.Generic 命名空间中包含了一些新的泛型集合类。您可以使用这些泛型集合类来替代 System.Collections 中的集合类。

  • 您可以创建自己的泛型接口、泛型类、泛型方法、泛型事件和泛型委托。

  • 您可以对泛型类进行约束以访问特定数据类型的方法。

  • 关于泛型数据类型中使用的类型的信息可在运行时通过使用反射获取。

二、泛型(Generic)方法

在上面的实例中,我们已经使用了泛型类,我们可以通过类型参数声明泛型方法。下面的程序说明了这个概念:

  1. using System;
  2. using System.Collections.Generic;
  3. namespace GenericMethodAppl
  4. {
  5. class Program
  6. {
  7. static void Swap<T>(ref T lhs, ref T rhs)
  8. {
  9. T temp;
  10. temp = lhs;
  11. lhs = rhs;
  12. rhs = temp;
  13. }
  14. static void Main(string[] args)
  15. {
  16. int a, b;
  17. char c, d;
  18. a = 10;
  19. b = 20;
  20. c = 'I';
  21. d = 'V';
  22. // 在交换之前显示值
  23. Console.WriteLine("Int values before calling swap:");
  24. Console.WriteLine("a = {0}, b = {1}", a, b);
  25. Console.WriteLine("Char values before calling swap:");
  26. Console.WriteLine("c = {0}, d = {1}", c, d);
  27. // 调用 swap
  28. Swap<int>(ref a, ref b);
  29. Swap<char>(ref c, ref d);
  30. // 在交换之后显示值
  31. Console.WriteLine("Int values after calling swap:");
  32. Console.WriteLine("a = {0}, b = {1}", a, b);
  33. Console.WriteLine("Char values after calling swap:");
  34. Console.WriteLine("c = {0}, d = {1}", c, d);
  35. Console.ReadKey();
  36. }
  37. }
  38. }

当上面的代码被编译和执行时,它会产生下列结果:

  1. Int values before calling swap:
  2. a = 10, b = 20
  3. Char values before calling swap:
  4. c = I, d = V
  5. Int values after calling swap:
  6. a = 20, b = 10
  7. Char values after calling swap:
  8. c = V, d = I

三、泛型(Generic)委托

您可以通过类型参数定义泛型委托。例如:

  1. delegate T NumberChanger<T>(T n);

下面的实例演示了委托的使用:

  1. using System;
  2. using System.Collections.Generic;
  3. delegate T NumberChanger<T>(T n);
  4. namespace GenericDelegateAppl
  5. {
  6. class TestDelegate
  7. {
  8. static int num = 10;
  9. public static int AddNum(int p)
  10. {
  11. num += p;
  12. return num;
  13. }
  14. public static int MultNum(int q)
  15. {
  16. num *= q;
  17. return num;
  18. }
  19. public static int getNum()
  20. {
  21. return num;
  22. }
  23. static void Main(string[] args)
  24. {
  25. // 创建委托实例
  26. NumberChanger<int> nc1 = new NumberChanger<int>(AddNum);
  27. NumberChanger<int> nc2 = new NumberChanger<int>(MultNum);
  28. // 使用委托对象调用方法
  29. nc1(25);
  30. Console.WriteLine("Value of Num: {0}", getNum());
  31. nc2(5);
  32. Console.WriteLine("Value of Num: {0}", getNum());
  33. Console.ReadKey();
  34. }
  35. }
  36. }

当上面的代码被编译和执行时,它会产生下列结果:

  1. Value of Num: 35
  2. Value of Num: 175

🔚