C# 中泛型(Generics)‌的核心概念

server/2025/3/19 6:48:10/

在 C# 中,‌泛型(Generics)‌ 是一种强大的编程特性,允许你编写可重用、类型安全的代码,而无需为不同类型重复编写相似的逻辑。泛型的核心思想是‌参数化类型‌,即通过占位符(如 T)表示类型,在编译时确定具体类型。以下是泛型的详细讲解:

‌1. 泛型的基本概念‌

  • 类型参数化‌:用占位符(如 T、TKey、TValue)代替具体类型。
  • 编译时类型安全‌:泛型在编译时检查类型一致性,避免运行时类型错误。
  • 避免装箱拆箱‌:泛型可避免值类型和引用类型之间的转换(如 List 比 ArrayList 更高效)。

‌2. 泛型的实现方式‌

‌(1) 泛型类‌

public class GenericClass<T>
{private T _value;public void SetValue(T value){_value = value;}public T GetValue(){return _value;}
}// 使用示例
var intClass = new GenericClass<int>();
intClass.SetValue(42);
int value = intClass.GetValue();  // 类型安全,直接返回 int

‌(2) 泛型方法‌

public class Utility
{public static T Max<T>(T a, T b) where T : IComparable<T>{return a.CompareTo(b) > 0 ? a : b;}
}// 使用示例
int maxInt = Utility.Max(3, 5);          // 返回 5
string maxStr = Utility.Max("A", "B");   // 返回 "B"

‌(3) 泛型接口‌

public interface IRepository<T>
{void Add(T entity);T GetById(int id);
}public class UserRepository : IRepository<User>
{public void Add(User user) { /* 实现 */ }public User GetById(int id) { /* 实现 */ }
}

‌3. 泛型约束(Constraints)‌

通过 where 关键字限制泛型类型参数的范围,增强类型安全性和功能可用性。

约束类型语法说明
类约束where T : classT 必须是引用类型
结构约束where T : structT 必须是值类型
构造函数约束where T : new()T 必须有无参构造函数
基类约束where T : BaseClassT 必须继承自 BaseClass
接口约束where T : IInterfaceT 必须实现 IInterface
组合约束where T : A, I, new()可组合多个约束

‌示例‌

public class Factory<T> where T : IEntity, new()    // IEntity代表某个实体类
{public T Create(){var entity = new T();entity.Initialize();return entity;}
}

‌4. 泛型的常见应用场景‌

‌(1) 集合类(如 List、Dictionary<TKey, TValue>)‌
泛型集合取代了非泛型集合(如 ArrayList),避免了类型转换和装箱拆箱:

List<int> numbers = new List<int> { 1, 2, 3 };
Dictionary<string, int> ages = new Dictionary<string, int> { { "Alice", 30 } };

‌(2) 泛型委托(如 Action、Func)‌

// 定义泛型委托
public delegate void Logger<T>(T message);// 使用示例
Logger<string> logString = message => Console.WriteLine(message);
Logger<int> logInt = number => Console.WriteLine($"Number: {number}");

‌(3) 泛型单例模式‌

public class Singleton<T> where T : new()
{private static readonly Lazy<T> _instance = new Lazy<T>(() => new T());public static T Instance => _instance.Value;
}// 使用示例
public class AppConfig { /* 配置类 */ }
var config = Singleton<AppConfig>.Instance;

‌5. 泛型的特殊特性‌

‌(1) 默认值(default 关键字)‌

T value = default(T);  // 对于引用类型返回 null,值类型返回 0 等

‌(2) 协变(Covariance)和逆变(Contravariance)‌

  • 协变(out 关键字)‌:允许子类型替换父类型(如 IEnumerable)。
  • ‌逆变(in 关键字)‌:允许父类型替换子类型(如 Action)。

‌示例‌:

IEnumerable<string> strings = new List<string>();
IEnumerable<object> objects = strings;  // 协变允许隐式转换Action<object> logObject = obj => Console.WriteLine(obj);
Action<string> logString = logObject;   // 逆变允许隐式转换

6. 泛型与反射‌

通过反射可以获取泛型类型的元数据:

Type listType = typeof(List<>);
Type intListType = listType.MakeGenericType(typeof(int));
var intList = Activator.CreateInstance(intListType);

‌7. 注意事项‌

  • 避免过度泛型化‌:只在需要类型灵活性的场景使用泛型。
  • ‌命名规范‌:类型参数名应具有描述性(如 TKey、TValue)。
  • ‌性能优化‌:泛型在编译时生成具体类型代码,不会带来运行时性能损失。

‌8. 典型问题与解决‌

‌问题:无法直接比较泛型类型‌

public bool AreEqual<T>(T a, T b)
{// 错误:T 可能未实现比较接口return a == b;  
}// 解决方案:添加 IEquatable<T> 约束
public bool AreEqual<T>(T a, T b) where T : IEquatable<T>
{return a.Equals(b);
}

通过泛型,C# 可以在编译时保证类型安全,同时提高代码的灵活性和复用性。合理使用泛型,可以显著减少重复代码并提升程序的可维护性。


http://www.ppmy.cn/server/176176.html

相关文章

【Linux操作系统——学习笔记二】Linux简单导航命令操作

一、前言 学习Linux&#xff0c;本质上是学习在命令行下熟练使用Linux的各类命令。 命令行&#xff1a;是一种通过输入命令和参数与计算机系统进行交互的方式&#xff0c;可以使用各种字符化命令对系统发出操作指令&#xff0c;打开Linux终端&#xff0c;进入命令行界面。 …

购物车全选功能

在电商平台的购物车页面中&#xff0c;全选功能是提升用户体验的重要特性。我们需要实现以下功能&#xff1a; 点击全选按钮&#xff0c;选中所有商品 商品选择状态变化时&#xff0c;自动更新全选按钮状态 支持动态加载商品列表 代码 $(#allCheck).on(change, function (…

Oracle常见系统函数

一、字符类函数 1&#xff0c;ASCII(c)和CHR(i)字符串和ascii码互转换 SQL> select ascii(Z) ,ascii(H),ascii( A) from dual;ASCII(Z) ASCII(H) ASCII(A) ---------- ---------- ----------90 72 32SQL> select chr(90),chr(72),chr(65) from dual;C…

Conda 环境迁移指南

一、环境打包迁移方法对比 方法适用场景所需时间文件大小网络依赖conda-pack离线迁移、复杂依赖、快速部署快&#xff08;5-15分钟&#xff09;大&#xff08;GB级&#xff09;无YAML导出在线重建、跨平台兼容慢&#xff08;30分钟&#xff09;小&#xff08;KB级&#xff09;…

道路运输安全员考试内容有哪些

道路运输安全员考试包括理论知识和实际操作两个主要部分。具体内容如下&#xff1a; 理论知识&#xff1a; 道路运输法规&#xff1a;需深入研究我国道路交通法规&#xff0c;如《中华人民共和国道路交通安全法》等&#xff0c;以及交通运输管理部门制定的相关安全生产规定。…

VSCode + CMake

参考文献&#xff1a; 如何用 GCC, CMake 和 Make 编译C/C代码Windows 上的 Linux 子系统&#xff1a;WSLWSL&#xff1a;桌面 UI 远程连接 RDP 配置 VScode 文章目录 CMake 配置VSCode 配置launch.jsontask.jsonc_cpp_properties.json CMake 配置 编写如下的 CmakeLists.t…

案例驱动的 IT 团队管理:创新与突破之路:第三章 项目攻坚:从流程优化到敏捷破局-3.1.2需求分级与资源调配策略

&#x1f449; 点击关注不迷路 &#x1f449; 点击关注不迷路 &#x1f449; 点击关注不迷路 文章大纲 3.1.2 需求分级与资源调配策略&#xff1a;构建敏捷破局的双引擎一、需求管理的范式革命&#xff1a;从线性到动态1.1 技术迭代倒逼管理创新1.2 需求分级的战略价值 二、需求…

FreeRTOS移植并实现一个多任务程序

打开CubeMX&#xff0c;点击如图 输入自己的芯片型号&#xff0c;我这里是STM32F103C8T6 选择好后点击start project 依次改为 依次改为 依次改为 在这里把频率改为72MHz 用keil的改成MDK_ARM 勾上 生成代码即可 ——————————————————————————————…