如何根据类名创建实例

如何根据类名创建实例,第1张

你所说的属于“反射”的知识点,反射在 Java、.NET 都是被支持的,并不是所有语言都支持,如:C++ 就没有反射的说法。

“反射”其实就是利用程序集的元数据信息。

反射可以有很多方法,编写程序时请先导入 System.Reflection 命名空间,假设你要反射一个 DLL 中的类,并且没有引用它(即未知的类型):

Assembly assembly = Assembly.LoadFile("程序集路径,不能是相对路径")// 加载程序集(EXE 或 DLL)

object obj = assembly.CreateInstance("类的完全限定名(即包括命名空间)")// 创建类的实例

若要反射当前项目中的类可以为:

Assembly assembly = Assembly.GetExecutingAssembly()// 获取当前程序集

object obj = assembly.CreateInstance("类的完全限定名(即包括命名空间)")// 创建类的实例,返回为 object 类型,需要强制类型转换

也可以为:

Type type = Type.GetType("类的完全限定名")

object obj = type.Assembly.CreateInstance(type)

=======================================================

补充:

1)反射创建某个类的实例时,必须保证使用类的完全限定名(命名空间 + 类名)。Type.GetType 方法返回 null 则意味搜索元数据中的相关信息失败(反射失败),请确保反射时使用类的完全限定名。

2)反射功能十分强大,没有什么不能实现的。若实现你所说的“跨程序集”,请使用我给的第一种方法创建类的实例,并反射该实例的字段、属性、方法、事件... 然后动态调用之。

你好!

要成功的反射另一个程序集中的类型及其方法,你必须先加载此程序集

请确定目标程序集的路径,然后使用 Assembly.LoadFrom方法进行加载,最后在载入的程序集上进行反射即可。

以你的解决方案为例,如果你没有更改过各项目的输出路径以及TagClassCustom项目的输出程序集名称,那么应该在PTtemplate.cs中这样写:

var asm = Assembly.LoadFrom(@"..\..\..\TagClassCustom\bin\Debug\TagClassCustom.dll")

var targetType = asm.GetType("TagClassCustom.Class1")

var targetMethod = targetType.GetMethod("MethodName")

如果不在指定程序集上反射而是直接使用Type.GetType方法,将在当前执行的主程序集上进行查找,也即是你的Factory项目。

关于Factory项目引用TagClassCustom项目后依然异常的情况,原因在于项目引用也是基于文件进行引用的,你引用的只是TagClassCustom项目的输出程序集TagClassCustom.dll,所以依然需要使用Assembly.LoadFrom加载程序集后在该程序集上进行反射。

最后还要指出一点,关于MethodInfo.Invoke方法的第一个参数。

当Method是静态的,第一个参数随便写什么都会被无视,通常我们写null;

当Method所在的类型是静态的,第一个参数必须使用null否则异常;

非以上两种情况,第一个参数需传入Method所在的类型的一个实例,一般都是使用Activator.CreateInstance方法创建。

在你的代码中,虽然传入一个字符串Namespace在此例中并不会出错,但是在调用反射的一个非静态类中的非静态方法时则必定异常,敬请注意。


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/yw/8081053.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-04-13
下一篇 2023-04-13

发表评论

登录后才能评论

评论列表(0条)

保存