.net泛型通过类型参数实现可重用、类型安全且高效的代码,避免运行时类型转换。1. 泛型类如Stack<T>和泛型方法如Max<T>支持类型化操作,编译器自动推断类型;2. 运行时CLR为值类型生成专用代码以提升性能,引用类型共享实现以节省内存;3. 泛型约束(where)确保类型合法性,支持基类、接口、构造函数、值/引用类型限制;4. 广泛应用于集合、依赖注入、数据访问、API设计等场景,提升代码质量与执行效率。

.NET中的泛型(Generics)是一种允许在定义类、接口和方法时使用类型参数的机制。它让开发者编写可重用、类型安全且高性能的代码,避免了运行时类型转换和装箱/拆箱操作。泛型不是语法糖,而是在编译时和运行时都发挥作用的语言特性。
泛型的基本语法与用法
泛型最常见的应用场景是集合类和方法。通过在类型或方法名后添加尖括号<T>来声明类型参数。
示例:泛型类
// 定义一个泛型栈 public class Stack<T> { private List<T> items = new List<T>();
public void Push(T item) { items.Add(item); } public T Pop() { if (items.Count == 0) throw new InvalidOperationException(); T item = items[^1]; items.RemoveAt(items.Count - 1); return item; }
}
// 使用 var intStack = new Stack<int>(); intStack.Push(123); int value = intStack.Pop(); // 直接返回int,无需转换
示例:泛型方法
public Static T Max<T>(T a, T b) where T : IComparable<T> { return a.CompareTo(b) >= 0 ? a : b; }
// 调用 var max = Max(5, 10); // 自动推断为int
泛型方法可以在非泛型类中定义,编译器通常能根据参数自动推断类型,减少显式指定的需要。
泛型的底层原理
泛型的核心优势在于类型专用化和运行时性能优化。.NET运行时(CLR)在处理泛型时,会根据不同类型参数生成专用的本地类型信息。
- 对于引用类型(如String、Object),CLR只生成一份泛型类型实例,所有引用类型共享同一份代码,节省内存。
- 对于值类型(如int、DateTime),CLR为每种具体值类型生成独立的专用类型,避免装箱/拆箱,提升性能。
例如:List<int> 和 List<double> 在运行时是两个不同的类型,各自拥有独立的方法表和字段布局,但List<string>和List<object>可能共享部分内部实现。
这种机制保证了类型安全的同时,最大限度地优化了执行效率。
泛型约束(Constraints)的使用
为了在泛型中调用特定成员或构造实例,需要对类型参数施加约束。约束通过where关键字定义。
示例:
public class Factory<T> where T : class, new() { public T Create() => new T(); }
多个约束可以组合使用,确保泛型逻辑在编译期就能验证合法性。
常见使用场景
泛型广泛应用于各种编程场景,以下是一些典型用例:
- 集合类:如List<T>、Dictionary<K,V>、Queue<T>等,提供类型安全的容器。
- 服务定位与依赖注入:通过泛型获取服务实例,如IServiceProvider.GetService<T>()。
- 数据访问层:泛型仓储模式(Repository Pattern)统一处理不同实体的CRUD操作。
- API设计:构建通用结果包装类,如Result<T> 表示操作成功并返回数据,或包含错误信息。
- 工厂与创建模式:利用new()约束动态创建对象实例。
基本上就这些。泛型是.NET中极为重要的特性,掌握其原理和使用方式,能显著提升代码质量与执行效率。理解类型参数如何被处理、何时生成专用代码,有助于写出更高效、更安全的应用程序。