Activator.CreateInstance 与 Type.InvokeMember的区别

2024-06-08 05:44

本文主要是介绍Activator.CreateInstance 与 Type.InvokeMember的区别,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 一、使用 Activator.CreateInstance 创建实例
    • 1、使用 Activator.CreateInstance 的优点和缺点
    • 2、使用 Activator.CreateInstance 的代码示例
  • 二、使用 Type.InvokeMember 创建实例
    • 1、使用 Type.InvokeMember 的优点和缺点
    • 2、使用 Type.InvokeMember 的代码示例
  • 三、Activator.CreateInstance 与 Type.InvokeMember 的基准测试
    • 1、反射性能的 BenchmarkDotNet 设置
    • 2、Activator.CreateInstance 与 Type.InvokeMember对比
  • 四、Activator.CreateInstance 与 Type.InvokeMember 总结


一、使用 Activator.CreateInstance 创建实例

在 C# 中,Activator.CreateInstance方法是开发人员通过反射创建实例的最常用方法之一。它允许在运行时创建类的新实例,即使事先不知道它们的具体类型。此方法属于类,System.Activator通常用于需要动态实例化的场景。

的主要目的Activator.CreateInstance是动态创建类的实例 — 因此,如果您在编译时可以访问该类型,那么您可能没有充分的理由使用它!Activator.CreateInstance在要创建的对象类型直到运行时才知道的情况下,例如在加载插件或使用动态加载的程序集时,它特别有用。它消除了对硬编码显式构造函数的需求,并提供了对象创建的灵活性。

1、使用 Activator.CreateInstance 的优点和缺点

Activator.CreateInstance与普通对象实例相比,使用有几个优点:

允许后期绑定对象创建,这在对象类型在运行时可能发生变化的情况下很有用。
可以通过消除使用 switch 语句或 if-else 条件来处理不同对象类型的需求来简化代码库。
提供一种动态且可扩展的对象创建方法。
但是,使用时也需要考虑一些缺点Activator.CreateInstance:

由于运行时解析类型涉及额外的步骤,因此与直接实例化相比,使用反射的性能开销可能更高。
Activator.CreateInstance 通常依赖于公共无参数构造函数的存在。否则,您需要始终知道要传入哪些参数 — 如果您要针对许多不同类型动态执行此操作,则具有挑战性。
由于没有编译时检查签名兼容性,因此在修改目标类型时容易出现错误。

2、使用 Activator.CreateInstance 的代码示例

以下代码示例演示了如何使用Activator.CreateInstance动态创建实例:

// Example 1: Creating an instance of a known type
Type objectType = typeof(MyClass);
object instance = Activator.CreateInstance(objectType);

在示例 1 中,我们使用typeof获取Type代表已知类的对象MyClass。然后,我们使用Activator.CreateInstance创建 的新实例MyClass。

// Example 2: Creating an instance of an unknown type at runtime
string typeName = "MyNamespace.MyClass";
Type unknownType = Type.GetType(typeName);
object dynamicInstance = Activator.CreateInstance(unknownType);

在示例 2 中,我们有一个由字符串表示的未知类型typeName。我们使用根据提供的类型名称Type.GetType获取对象。最后,用于创建动态确定类型的新实例。TypeActivator.CreateInstance

我们再看一个例子,其中我们可以为构造函数传递参数——同样,假设我们知道签名,因为我们无法在编译时通过这种方法证明它:

// Example 3: Creating an instance with constructor parameters:
string typeName = "MyNamespace.MyClass";
Type unknownType = Type.GetType(typeName);
Object dynamicInstance = Activator.CreateInstance(unknownType,new[]{"Hello World!", //  this is the single string parameter!});

二、使用 Type.InvokeMember 创建实例

Type.InvokeMember是 DotNet 中 Reflection 提供的一种可用方法,允许我们动态创建类型的实例。它提供了一种灵活的方法,可以通过利用手头类型的信息在运行时实例化对象。出于这些原因,它在如何利用它来创建对象实例方面非常相似。

1、使用 Type.InvokeMember 的优点和缺点

Type.InvokeMember与普通对象实例相比,使用以下方法具有一些普遍的优点:

允许后期绑定对象创建,这在对象类型在运行时可能发生变化的情况下很有用。
可以通过消除使用 switch 语句或 if-else 条件来处理不同对象类型的需求来简化代码库。
提供一种动态且可扩展的对象创建方法。
等一下……这不就是我们上面看到的那个列表吗Activator.CreateInstance?没错。所以我们就简短地介绍一下这部分。除非我们开始研究性能,否则我们不会看到任何大的差异——也许在某些非常特殊的边缘情况下。但总的来说,两者都提供了非常全面的方法来动态实例化对象,而 InvokeMember 有点冗长,因为它处理的不仅仅是构造函数。

在进行基准测试之前,我们先检查一些代码。

2、使用 Type.InvokeMember 的代码示例

下面是一个示例代码片段,演示了如何Type.InvokeMember动态创建类型实例:

// Example 1: Creating an instance of a known type
Type objectType = typeof(MyClass);
var instance = objectType
.InvokeMember(null,BindingFlags.CreateInstance,null,null,null);

在上面的例子中,我们首先获取代表类“MyClass”的 Type 对象。然后我们使用 Type.InvokeMember 创建该类的实例并将其分配给“instance”变量。这使我们能够动态创建“MyClass”的对象,而无需明确指定类名。

而在编译时不知道类的情况下执行此操作与以前非常相似。但这部分与 InvokeMember 无关:

// Example 2: Creating an instance of an unknown type at runtime
string typeName = "MyNamespace.MyClass";
Type unknownType = Type.GetType(typeName);
var instance = objectType.InvokeMember(null,BindingFlags.CreateInstance,null,null,null);

最后,如果我们需要传递一些构造函数参数,那么我们也可以这样做:

// Example 3: Creating an instance with constructor parameters:
string typeName = "MyNamespace.MyClass";
Type unknownType = Type.GetType(typeName);
var instance = objectType.InvokeMember(null,BindingFlags.CreateInstance,null,null,new[]{"Hello World!",});

三、Activator.CreateInstance 与 Type.InvokeMember 的基准测试

1、反射性能的 BenchmarkDotNet 设置

我认为我们可以在三种情况下使用 BenchmarkDotNet 运行基准测试:

无参数构造函数类
具有单个字符串参数的构造函数
具有单个字符串参数的主构造函数
我想加入主构造函数,因为我知道这个特性会招致很多人的反对——最好还是获取一些数据吧!以下是我们将要实例化的类,供参考:

public class ParameterlessClass
{
}public class ClassicStringParameterClass
{private readonly string _value;public ClassicStringParameterClass(string value){_value = value;}
}public class PrimaryConstructorStringParameterClass(string _value)
{
}

至于基准测试,让我们看看我们将要运行的以下类。请记住,我使用它Activator.CreateInstance作为基线是因为我想比较Activator.CreateInstancevs Type.InvokeMember— 我只包括正常的构造函数路径作为参考。您也可以在 GitHub 上找到所有这些代码:

[ShortRunJob]
public class ParameterlessClassBenchmarks
{private Type? _type;[GlobalSetup]public void GlobalSetup(){_type = typeof(ParameterlessClass);}[Benchmark]public void Constructor(){var instance = new ParameterlessClass();}[Benchmark(Baseline = true)]public void Activator_Create_Instance(){var instance = Activator.CreateInstance(_type!);}[Benchmark]public void Type_Invoke_Member(){var instance = _type!.InvokeMember(null,BindingFlags.CreateInstance,null,null,null);}
}[ShortRunJob]
public class ClassicStringParameterClassBenchmarks
{private Type? _type;[GlobalSetup]public void GlobalSetup(){_type = typeof(ClassicStringParameterClass);}[Benchmark]public void Constructor(){var instance = new ClassicStringParameterClass("Hello World!");}[Benchmark(Baseline = true)]public void Activator_Create_Instance(){var instance = Activator.CreateInstance(_type!,new[]{"Hello World!",});}[Benchmark]public void Type_Invoke_Member(){var instance = _type!.InvokeMember(null,BindingFlags.CreateInstance,null,null,new[]{"Hello World!",});}
}[ShortRunJob]
public class PrimaryConstructorStringParameterClassBenchmarks
{private Type? _type;[GlobalSetup]public void GlobalSetup(){_type = typeof(PrimaryConstructorStringParameterClass);}[Benchmark]public void Constructor(){var instance = new PrimaryConstructorStringParameterClass("Hello World!");}[Benchmark(Baseline = true)]public void Activator_Create_Instance(){var instance = Activator.CreateInstance(_type!,new[]{"Hello World!",});}[Benchmark]public void Type_Invoke_Member(){var instance = _type!.InvokeMember(null,BindingFlags.CreateInstance,null,null,new[]{"Hello World!",});}
}

2、Activator.CreateInstance 与 Type.InvokeMember对比

当我们将这两种反射方法进行对比时,获胜者是…视情况而定。在最常见的一种情况下,我会说有一个非常明显的赢家,但对于其他情况,它们非常接近。但请务必阅读结论,因为这不是故事的结局。

我们要研究的第一个基准是无参数构造函数:
在这里插入图片描述

Activator.CreateInstance 与 Type.InvokeMember - 无参数构造函数的基准
这里明显胜出:Activator.CreateInstance几乎高出一个数量级。如果你的构造函数没有任何参数,那么最好的选择就是这个。

接下来,让我们检查Activator.CreateInstance一下Type.InvokeMember接受单个字符串参数的构造函数:
在这里插入图片描述

Activator.CreateInstance 与 Type.InvokeMember - 带参数的经典构造函数的基准
这两款车基本上势均力敌,尽管Activator.CreateInstance略胜一筹,但差距几乎可以忽略不计。

要查看的最后一个场景是主构造函数,在本例中,主构造函数接受一个字符串参数:
在这里插入图片描述

Activator.CreateInstance 与 Type.InvokeMember - 主构造函数的基准
获胜者:Type.InvokeMember,但只领先一点点。非常有趣的是,这与我们在之前的基准测试中看到的结果相反!

四、Activator.CreateInstance 与 Type.InvokeMember 总结

当谈到Activator.CreateInstancevs的性能结果时Type.InvokeMember,无参数构造函数的情况明显胜出:Activator.CreateInstance。但是当我们开始需要参数或使用带参数的主构造函数时,它开始均衡甚至有利于Type.InvokeMember。

这篇关于Activator.CreateInstance 与 Type.InvokeMember的区别的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1041343

相关文章

native和static native区别

本文基于Hello JNI  如有疑惑,请看之前几篇文章。 native 与 static native java中 public native String helloJni();public native static String helloJniStatic();1212 JNI中 JNIEXPORT jstring JNICALL Java_com_test_g

Android fill_parent、match_parent、wrap_content三者的作用及区别

这三个属性都是用来适应视图的水平或者垂直大小,以视图的内容或尺寸为基础的布局,比精确的指定视图的范围更加方便。 1、fill_parent 设置一个视图的布局为fill_parent将强制性的使视图扩展至它父元素的大小 2、match_parent 和fill_parent一样,从字面上的意思match_parent更贴切一些,于是从2.2开始,两个属性都可以使用,但2.3版本以后的建议使

Collection List Set Map的区别和联系

Collection List Set Map的区别和联系 这些都代表了Java中的集合,这里主要从其元素是否有序,是否可重复来进行区别记忆,以便恰当地使用,当然还存在同步方面的差异,见上一篇相关文章。 有序否 允许元素重复否 Collection 否 是 List 是 是 Set AbstractSet 否

javascript中break与continue的区别

在javascript中,break是结束整个循环,break下面的语句不再执行了 for(let i=1;i<=5;i++){if(i===3){break}document.write(i) } 上面的代码中,当i=1时,执行打印输出语句,当i=2时,执行打印输出语句,当i=3时,遇到break了,整个循环就结束了。 执行结果是12 continue语句是停止当前循环,返回从头开始。

Oracle type (自定义类型的使用)

oracle - type   type定义: oracle中自定义数据类型 oracle中有基本的数据类型,如number,varchar2,date,numeric,float....但有时候我们需要特殊的格式, 如将name定义为(firstname,lastname)的形式,我们想把这个作为一个表的一列看待,这时候就要我们自己定义一个数据类型 格式 :create or repla

maven发布项目到私服-snapshot快照库和release发布库的区别和作用及maven常用命令

maven发布项目到私服-snapshot快照库和release发布库的区别和作用及maven常用命令 在日常的工作中由于各种原因,会出现这样一种情况,某些项目并没有打包至mvnrepository。如果采用原始直接打包放到lib目录的方式进行处理,便对项目的管理带来一些不必要的麻烦。例如版本升级后需要重新打包并,替换原有jar包等等一些额外的工作量和麻烦。为了避免这些不必要的麻烦,通常我们

ActiveMQ—Queue与Topic区别

Queue与Topic区别 转自:http://blog.csdn.net/qq_21033663/article/details/52458305 队列(Queue)和主题(Topic)是JMS支持的两种消息传递模型:         1、点对点(point-to-point,简称PTP)Queue消息传递模型:         通过该消息传递模型,一个应用程序(即消息生产者)可以

深入探讨:ECMAScript与JavaScript的区别

在前端开发的世界中,JavaScript无疑是最受欢迎的编程语言之一。然而,很多开发者在使用JavaScript时,可能并不清楚ECMAScript与JavaScript之间的关系和区别。本文将深入探讨这两者的不同之处,并通过案例帮助大家更好地理解。 一、什么是ECMAScript? ECMAScript(简称ES)是一种脚本语言的标准,由ECMA国际组织制定。它定义了语言的语法、类型、语句、

Lua 脚本在 Redis 中执行时的原子性以及与redis的事务的区别

在 Redis 中,Lua 脚本具有原子性是因为 Redis 保证在执行脚本时,脚本中的所有操作都会被当作一个不可分割的整体。具体来说,Redis 使用单线程的执行模型来处理命令,因此当 Lua 脚本在 Redis 中执行时,不会有其他命令打断脚本的执行过程。脚本中的所有操作都将连续执行,直到脚本执行完成后,Redis 才会继续处理其他客户端的请求。 Lua 脚本在 Redis 中原子性的原因

Caused by: org.hibernate.MappingException: Could not determine type for: org.cgh.ssh.pojo.GoodsType,

MappingException:这个主要是类映射上的异常,Could not determine type for: org.cgh.ssh.pojo.GoodsType,这句话表示GoodsType这个类没有被映射到