利用反射动态调用类成员

利用反射动态调用类成员

 

如果想复用别人写的.NET程序但手上只有一个编译过的EXE,那么方法之一是用Reflection(反射)。下面是一些例子,初次上手的人可以参考。

先假设我们要复用的第三方应用程序EXE是由下面的代码编译成的:

using System;
namespace MyNamespace
{
 public class MyApp
 {
  public MyNestedObject myNestedObject=null;
  public class MyNestedObject{
   public string name;
  }
  public enum FourSeasonsEnum{
   spring, summer, autumn, winter
  }
  public MyApp(){
  }
  public MyApp(MyNamespace.MyForm form){
  }
  public MyNestedObject Foo1(FourSeasonsEnum season){
   return this.myNestedObject;
  }
  public string Foo2(){
   return "";
  }
  static void Main(){
  }
 }
 public class MyForm
 {
  public MyForm(){
  }
 }
}

以下是一些直接引用时常见的调用如何用Reflection来改写:

1. 用不含参数的构造函数生成对象

直接引用时,代码是

MyNamespace.MyApp app=new MyNamespace.MyApp();

用Reflection来调用的话就需要这么写(记得using System.Reflection)

Assembly assem=Assembly.LoadFile(@"C:\Home\Workspace\MyNamespace\myapp.exe");
Type MyAppType=assem.GetType("MyNamespace.MyApp");
ConstructorInfo MyAppType_Constructor=MyAppType.GetConstructor(new Type[]{});
object app=MyAppType_Constructor.Invoke(new object[]{});

2. 用含参数的构造函数生成对象

直接引用时,代码是

MyNamespace.MyApp app=new MyNamespace.MyApp(new MyNamespace.MyForm());

用Reflection来调用的话就需要这么写

Assembly assem=Assembly.LoadFile(@"C:\Home\Workspace\MyNamespace\myapp.exe"); 
Type MyAppType=assem.GetType("MyNamespace.MyApp"); 
Type MyFormType=assem.GetType("MyNamespace.MyForm");
ConstructorInfo MyAppType_Constructor=MyAppType.GetConstructor(new Type[]{MyFormType});
ConstructorInfo MyFormType_Constructor=MyFormType.GetConstructor(new Type[]{});
object form=MyFormType_Constructor.Invoke(new object[]{});
object app=MyAppType_Constructor.Invoke(new object[]{form});

3. 调用对象的方法

直接引用时,代码是

MyNamespace.MyApp app=new MyNamespace.MyApp();
string str=app.Foo2();

用Reflection来调用的话就需要这么写

Assembly assem=Assembly.LoadFile(@"C:\Home\Workspace\MyNamespace\myapp.exe");
Type MyAppType=assem.GetType("MyNamespace.MyApp");
ConstructorInfo MyAppType_Constructor=MyAppType.GetConstructor(new Type[]{});
object app=MyAppType_Constructor.Invoke(new object[]{});
object str=MyAppType.GetMethod("Foo2").Invoke(app,new object[]{});

4. Set/Get成员变量

直接引用时,代码是

MyNamespace.MyApp app=new MyNamespace.MyApp();
MyNamespace.MyApp.MyNestedObject obj=app.myNestedObject;
MyNamespace.MyApp.MyNestedObject obj2=new MyNamespace.MyApp.MyNestedObject();
app.myNestedObject =obj2;

用Reflection来调用的话就需要这么写(注意,这里的MyNestedObject类是nested type,名字要用”MyNamespace.MyApp+MyNestedObject”而不是”MyNamespace.MyApp.MyNestedObject”)

Assembly assem=Assembly.LoadFile(@"C:\Home\Workspace\MyNamespace\myapp.exe");
Type MyAppType=assem.GetType("MyNamespace.MyApp");
ConstructorInfo MyAppType_Constructor=MyAppType.GetConstructor(new Type[]{});
object app=MyAppType_Constructor.Invoke(new object[]{});
Type MyNestedObjectType=assem.GetType("MyNamespace.MyApp+MyNestedObject");
FieldInfo MyNestedObjField=MyAppType.GetField("myNestedObject");
object obj=MyNestedObjField.GetValue(app);
    
ConstructorInfo MyNestedObjectType_Constructor=MyNestedObjectType.GetConstructor(new Type[]{});
object obj2=MyNestedObjectType_Constructor.Invoke(new object[]{});
MyNestedObjField.SetValue(app,obj2);

5. 使用枚举类型

直接引用时,代码是

MyNamespace.MyApp app=new MyNamespace.MyApp();
MyNamespace.MyApp.MyNestedObject obj=app.Foo1(MyNamespace.MyApp.FourSeasonsEnum.spring);

用Reflection来调用的话就需要这么写

Assembly assem=Assembly.LoadFile(@"C:\Home\Workspace\MyNamespace\myapp.exe");
Type MyAppType=assem.GetType("MyNamespace.MyApp");
ConstructorInfo MyAppType_Constructor=MyAppType.GetConstructor(new Type[]{});
object app=MyAppType_Constructor.Invoke(new object[]{});
Type FourSeasonsEnumType=assem.GetType("MyNamespace.MyApp+FourSeasonsEnum");
Array FourSeasonsEnumValues=Enum.GetValues(FourSeasonsEnumType);
object SpringValue=FourSeasonsEnumValues.GetValue(0);
object result=MyAppType.GetMethod("Foo1").Invoke(app,new object[]{SpringValue});

最后,只要有可能,应该尽量不用Reflection,因为相比起直接调用,Reflection的性能相当差

转载于:https://www.cnblogs.com/liuzhiwei/archive/2005/12/05/291385.html

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/111157.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • (7)JMeter元件详解之 Interleave Controller 交替控制器

    (7)JMeter元件详解之 Interleave Controller 交替控制器

    2021年7月14日
    99
  • CentOS7关闭selinux[通俗易懂]

    1、为什么要关闭selinux初学者配置linux服务器时不成功,却没有头绪,那是因为在linux操作系统中默认开启了防火墙,SELinux也处于启动状态,一般状态为enforing。致使很多服务端口默认是关闭的。所以好多服务初学者明明配置文件正确,等验证时有时连ping也ping不通。建议初学者在未学到SELlinux与iptables之前,配置服务器把这两项都关掉。2、查看selinux状…

    2022年4月18日
    52
  • fastclick插件

    fastclick插件fastclick插件可以解决移动端点击延迟300ms的bug首先引入插件fastclick在项目中导入插件document.addEventListener(‘DOMContentLoaded’,function(){FastClick.attach(document.body);},false);…

    2022年6月19日
    75
  • php递归函数详解_用php递归函数实现阶乘计算

    php递归函数详解_用php递归函数实现阶乘计算本节内容:PHP递归算法。PHP递归算法代码:复制代码代码示例://定义PI一分的角度的值define(“PII”,M_PI/180);//新建图像资源,并定义其背景为白色,前景色为黑色$im=imagecreate(670,500);$white=imagecolorallocate($im,0xFF,0xFF,0xFF);$g=imagecolorallocate($im,0x00,0x0…

    2022年8月11日
    3
  • Javascript高级程序设计第四版详细测评[通俗易懂]

    Javascript高级程序设计第四版详细测评[通俗易懂]简介哈喽大家好,我是大圣,上次做了一个js的书籍测评,评价还不错,在做css和node之前,再做几本书的详细推荐吧,国庆今天又在家好好读了几本具有代表性的javascript高级程…

    2022年8月20日
    3
  • 0187eaia data access error_文档错误码700015

    0187eaia data access error_文档错误码700015AnalyticDB错误码-DDL,ACL相关范围说明18000~18599DDLCREATE语句用户错误18600~18799DDLALTER语句用户错误18800~18899DDLDROP语句用户错误18900~18999ACL操作相关用户错误19000~19599D…

    2022年9月17日
    0

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号