反射发送实战(-)InvokeMember

原创 2004年08月24日 09:58:00

反射是.net中的高级功能之一,利用反射可以实现许多以前看来匪夷所思的功能,下面是我看了《Programming C#》(O'Reilly)之后对于反射的一点实践,本想直接做个应用程序来说明问题,但苦于工作繁忙并考虑到以简单为主,故先对反射发送(reflection emit)的使用做一些介绍。文章最后再给出一个实例。

 

下面的程序在运行时生成了一个Test.cs文件,并调用csc编译成Test.dll文件,然后利用Type.InvokeMember()方法调用其中的SayHello()方法,然后和原始方法对比一下性能。

using System;
using System.Diagnostics;
using System.IO;
using System.Reflection;

namespace InvokeMember
{
   
///
   
/// Class1 的摘要说明。
   
///
    class Class1 { ///
       
/// 应用程序的主入口点。
       
///
        [STAThread]
       
static void Main(string[] args)
        {
           
//循环次数
            const int iterations = 100;
           
//计算所用时间
            DateTime startTime = DateTime.Now;
           
for(int i = 0;i< iterations;i++) { //对照方法
                Console.WriteLine("Hello,World");
            }
            TimeSpan elasped
= DateTime.Now - startTime;
            Console.WriteLine(
"Looping Elapsed milliseconds:" + elasped.TotalMilliseconds + "for {0} iterations",iterations);
           
           
//使用反射发送
            ReflectionTest t = new ReflectionTest();
           
//计算所用时间
            startTime = DateTime.Now;
           
for(int i = 0;i < iterations;i++)
            {
                t.DoOperation();
            }
           
            elasped
= DateTime.Now - startTime;

            Console.WriteLine(
"Looping Elapsed milliseconds:" + elasped.TotalMilliseconds + "for {0} iterations",iterations);
            Console.ReadLine();
        }
    }

   
///
   
/// Reflection 的摘要说明。
   
///
    public class ReflectionTest
    {
       
//保存动态生成并编译的类的type对象
        Type theType = null;
       
//保存动态生成类的实例
        object theClass = null;

       
///
       
/// 供Client调用的方法
       
///
        public void DoOperation()
        {
           
//未初始化
            if(theType == null)
            {
               
//初始化
                GenerateCode();
            }
           
//调用方法时的参数数组(此处为空)
            object[] arguments = new object[0];
           
//调用动态生成类的方法
            theType.InvokeMember("SayHello",//要调用的方法名
                BindingFlags.Default|BindingFlags.InvokeMethod,//Binding标志,具体参看msdn
                null,//使用默认Binding对象
                theClass,//在theClass实例上调用此方法
                arguments//调用方法时的参数数组
                );
        }

       
///
       
/// 运行时生成代码
       
///
        private void GenerateCode()
        {
           
//文件名
            string fileName = "Test";
           
//打开文件,如果不存在,则创建
            Stream s = File.Open(fileName + ".cs",FileMode.Create);
           
//创建一个StreamWriter来写入数据
            StreamWriter wrtr = new StreamWriter(s);
           
//写入动态创建类的源代码
            wrtr.WriteLine("// 动态创建Test类");

           
//类名
            string className = "TestClass";
            wrtr.WriteLine(
"using System;");
            wrtr.WriteLine(
"class {0}",className);
            wrtr.WriteLine(
"{");

            wrtr.WriteLine(
"/tpublic void SayHello()");
            wrtr.WriteLine(
"/t{");

            wrtr.WriteLine(
"/t/tConsole.WriteLine(/"Hello,World/");");
            wrtr.WriteLine(
"/t}");
            wrtr.WriteLine(
"}");

           
//关闭StreamWriter和文件
            wrtr.Close();
            s.Close();

           
//启动进程编译源文件
           
//指定参数
            ProcessStartInfo psi = new ProcessStartInfo();
           
//启动cmd.exe
            psi.FileName = "cmd.exe";
           
//cmd.exe的参数,/c-close,完成后关闭;后为参数,指定cmd.exe使用csc来编译刚才生成的源文件
            string compileString = "/c C://WINNT//Microsoft.NET//Framework//v1.1.4322//csc.exe /optimize+ /target:library {0}.cs";
            psi.Arguments
= String.Format(compileString,fileName);
           
//运行时的风格-最小化
            psi.WindowStyle = ProcessWindowStyle.Minimized;
       
           
//启动进程
            Process proc = Process.Start(psi);
           
//指定当前在此进程退出前等待
            proc.WaitForExit();

           
//从编译好的dll文件load一个Assembly
            Assembly a = Assembly.LoadFrom(fileName + ".dll");

           
//创建类的实例
            theClass = a.CreateInstance(className);
           
//取得此类实例的类型
            theType = a.GetType(className);
           
//删除源文件
           
//File.Delete(flieName + ".cs");
        }
    }
}

程序运行结果:

Hello,World
Hello,World

.

.

.

Looping Elapsed milliseconds:93.75 for 100 iterations

Hello,World
Hello,World

.

.

.

Looping Elapsed milliseconds:2875 for 100 iterations

性能上不占优势,主要是因为要进行写文件和编译的工作,但是后面的方法会对性能进行优化,到最后一个方法时性能会有大幅提高,但是最后一种方法的实用性不如前两种。

反射发送实战(-)InvokeMember

反射是.net中的高级功能之一,利用反射可以实现许多以前看来匪夷所思的功能,下面是我看了《Programming C#》(OReilly)之后对于反射的一点实践,本想直接做个应用程序来说明问题,但苦于...
  • qq_net
  • qq_net
  • 2004-09-21 12:40:00
  • 758

c# Type.InvokeMember用法

函数原型:  public object InvokeMember(string, BindingFlags, Binder, object, object[]); string:你所要调用的函数...
  • fuck487
  • fuck487
  • 2015-07-09 16:46:50
  • 4894

反射带有out参数的方法

public int GetFlag(out int Result);使用下面的方法来反射:   System.Type myType = typeof(TestReflection);   //反射...
  • JerryChou
  • JerryChou
  • 2010-09-22 22:16:00
  • 1906

C# 反射详解:定义、创建对象、调用实例方法及静态方法

C# 反射详解 定义 动态创建对象及调用实例方法 1、反射的定义及功能介绍:审查元数据并收集关于它的类型信息的能力。元数据(编译以后的最基本数据单元)就是一大堆的表,当编译程序集或者模块时,编译器会...
  • u014019959
  • u014019959
  • 2014-05-18 16:31:11
  • 3037

Type InvokeMember()用法简介

举例: Type tDate = typeof(System.DateTime); Object result = tDate.InvokeMember("Now", ...
  • az44yao
  • az44yao
  • 2016-04-22 07:23:25
  • 561

反射发送实战(-)InvokeMember [转载] 应为:动态调用才对。

声明:以前找(InvokeMember))动态调用实例的时候,找了这样一个实例。对文章的标题很是不解,后来查了相关资料,可以证实,这不是一个反射发送的例子,而是一个动态调用的例子,只是用到了动态联编技...
  • c8640
  • c8640
  • 2007-06-13 18:20:00
  • 450

InvokeMember 使用

函数原型:       public   object   InvokeMember(string,   BindingFlags,   Binder,   object,   object[]); ...
  • gooer
  • gooer
  • 2008-09-14 10:31:00
  • 9013

java反射实战

《Java反射实战》—— Ira R.Forman 、Nate Forman第一章 一些基本知识 反射基础 类基础 使用方法我们经常碰到一些问题,而这些问题可以使用反射来简单并优雅的解决掉。如果没有反...
  • rjcs888
  • rjcs888
  • 2016-12-01 22:57:39
  • 926

.NET中如何在调用COM时得到返回参数值System.Type.InvokeMember的使用

在调用COM组件接口的时候,我们一般使用Type.InvokeMember()的方法来调用。Type.InvokeMember()在调用的时候需要接受一个Object类型的数组来传递接口的参数信息。对...
  • suiyunonghen
  • suiyunonghen
  • 2009-12-03 12:43:00
  • 2536

调用Type.InvokeMember()时出现MissingMethodException

调用Type.InvokeMember()时出现MissingMethodException经常是由于缺少必要的BindingFlags引起的。 在关于Type.InvokeMember   Me...
  • yefengmeander
  • yefengmeander
  • 2010-12-28 13:51:00
  • 668
收藏助手
不良信息举报
您举报文章:反射发送实战(-)InvokeMember
举报原因:
原因补充:

(最多只允许输入30个字)