Activator.CreateInstance 是 C# 中用于在运行时动态创建对象的常用方式,适用于类型未知、依赖注入、插件系统或反射场景。它不依赖编译期确定的类型,而是通过
Type对象来实例化类。
基础用法:无参构造函数创建对象
最简单的情况是目标类有 public 无参构造函数:
var type = typeof(List<string>); var list = Activator.CreateInstance(type); // 返回 object,需显式转换 // 或直接指定泛型类型(推荐) var list2 = (List<string>)Activator.CreateInstance(typeof(List<string>));
注意:返回值是
object,如果需要强类型操作,必须强制转换或使用泛型重载(.NET Core 2.0+ / .NET 5+ 支持)。
带参数的构造函数创建对象
当类的构造函数需要参数时,可传入参数数组:
public class Person
{
public Person(string name, int age) { /* ... */ }
}
<p>var person = (Person)Activator.CreateInstance(
typeof(Person),
"张三", 25);</p>
参数顺序和类型必须严格匹配构造函数签名
如果参数类型不匹配(如传入 long但构造函数要
int),会抛出
ArgumentException支持值类型、引用类型、null(前提是构造函数允许)
性能与替代方案
Activator.CreateInstance内部使用反射,比直接 new 慢不少,高频调用时建议缓存或换用更高效方式: 对固定类型:用
Expression.Lambda编译委托(一次性开销,后续极快) .NET 5+ 推荐
Activator.CreateInstance<t>()</t>泛型方法,性能更好且类型安全 依赖注入容器(如 Microsoft.Extensions.DependencyInjection)更适合管理生命周期和依赖关系
常见问题与注意事项
目标类型必须有可访问的构造函数(private 构造函数需配合BindingFlags,但不推荐) 静态类、抽象类、接口无法实例化,会抛出
InvalidOperationException泛型类型需先构造(
typeof(List).MakeGenericType(typeof(int))),再传给
CreateInstance跨程序集时确保类型已加载,否则抛出
FileNotFoundException或
TypeLoadException
基本上就这些。用对场景很实用,但别在热路径里反复调用 —— 不复杂但容易忽略性能代价。
