MSIL实用指南-创建方法和定义参数

时间:2023-03-09 09:02:36
MSIL实用指南-创建方法和定义参数

本篇讲解实现创建方法、指定参数的名称、实现参数加out和ref修饰符、以及参数加默认值。

创建方法

创建方法用类TypeAttributes的

DefineMethod(string name, MethodAttributes attributes, Type returnType, Type[] parameterTypes)

方法,返回结果是MethodBuilder,就可以创普通方法。

例子

MethodBuilder m2 = typeBuilder.DefineMethod("M2",
MethodAttributes.Public | MethodAttributes.Abstract | MethodAttributes.Virtual,
typeof(string), new Type[] { typeof(int), typeof(string), typeof(object) });

这个写法就和下面的C#程序一样

public abstract void M1();

定义一个抽象方法需要用MethodAttributes的Abstract|Virtual才可以。

定义参数

用MethodBuilder的DefineParameter(int position, ParameterAttributes attributes, string strParamName)

方法

参数说明

position:该参数在参数列表中的位置。为参数编索引,第一个参数从数字 1 开始;数字 0 表示方法的返回值。

attributes: 参数的参数属性。

strParamName: 参数名。名称可以为 null 字符串。

返回结果:

返回一个 ParameterBuilder 对象,该对象表示此方法的参数或此方法的返回值。

1.指定参数的名称

在程序的方法调用中传入第三个参数传入参数名称就可以了。

例如

MethodBuilder m2 = typeBuilder.DefineMethod("M2",
MethodAttributes.Public | MethodAttributes.Abstract | MethodAttributes.Virtual,
typeof(string), new Type[] { typeof(int), typeof(string), typeof(object) }); m2.DefineParameter(, ParameterAttributes.None, "arg0");
m2.DefineParameter(, ParameterAttributes.None, "param1");
m2.DefineParameter(, ParameterAttributes.None, "param2");

这个方法的三个参数名称依次是arg0、param1、param2。

2.指定参数out

我们要实现如下方法,参数的修饰符是out

public abstract string mout(out int arg3 );

需要使用ParameterAttributes.Out就可以了。

具体实现

MethodBuilder m5 = typeBuilder.DefineMethod("mout",
MethodAttributes.Public | MethodAttributes.Abstract | MethodAttributes.Virtual,
typeof(string), new Type[] { typeof(int), typeof(int) }); ParameterBuilder m5p1 = m5.DefineParameter(1, ParameterAttributes.Out, "arg3");

3.实现参数默认值

要实现参数有默认值,比如下面这句

public abstract string mdefault(int arg1 = );

实现这个效果的程序是

MethodBuilder m3 = typeBuilder.DefineMethod("mdefault",
MethodAttributes.Public | MethodAttributes.Abstract | MethodAttributes.Virtual,
typeof(string), new Type[] { typeof(int) }); ParameterBuilder m3p1 = m3.DefineParameter(, ParameterAttributes.HasDefault | ParameterAttributes.Optional, "arg1");
m3p1.SetConstant();

实现定义参数属性为ParameterAttributes.HasDefault | ParameterAttributes.Optional,并得到一个

ParameterBuilder实例,再设置这个实例的默认值。

4.实现指定参数ref

我们要实现如下方法,参数的修饰符是out

public abstract string mref(ref dobule arg2);

首先

MethodBuilder m4 = typeBuilder.DefineMethod("mref",
MethodAttributes.Public | MethodAttributes.Abstract | MethodAttributes.Virtual,
typeof(string), new Type[] { Type.GetType("System.Double&") });

参数的类型不是一般的typeof(double),而要用特殊的Type.GetType("System.Double&")

其次

ParameterBuilder m4p1 = m4.DefineParameter(, ParameterAttributes.Out, "arg2");

要设置参数的属性为ParameterAttributes.Out

完整程序如下

using System;
using System.Reflection;
using System.Reflection.Emit; namespace LX1_ILDemo
{
class Demo07_Method
{
static string binaryName = "Demo07_Method.dll";
static string namespaceName = "LX1_ILDemo";
static string typeName = "EmitMethod"; static AssemblyBuilder assemblyBuilder;
static ModuleBuilder moduleBuilder;
static TypeBuilder typeBuilder; public static void Generate()
{
InitAssembly(); typeBuilder = moduleBuilder.DefineType(namespaceName + "." + typeName,
TypeAttributes.Public | TypeAttributes.Abstract); /* 生成 public static void Main() */
GenerateMethods(); SaveAssembly();
Console.WriteLine("生成成功");
} static void GenerateMethods()
{
/* public abstract void M1(); */
MethodBuilder m1 = typeBuilder.DefineMethod("M1",
MethodAttributes.Public | MethodAttributes.Abstract | MethodAttributes.Virtual,
typeof(void), new Type[] { }); /* public abstract string M2(int arg0,string param1); */
MethodBuilder m2 = typeBuilder.DefineMethod("M2",
MethodAttributes.Public | MethodAttributes.Abstract | MethodAttributes.Virtual,
typeof(string), new Type[] { typeof(int), typeof(string), typeof(object) }); m2.DefineParameter(, ParameterAttributes.None, "arg0");
m2.DefineParameter(, ParameterAttributes.None, "param1");
m2.DefineParameter(, ParameterAttributes.None, "param2"); /* public abstract string mdefault(int arg1 =4 ); */
MethodBuilder m3 = typeBuilder.DefineMethod("mdefault",
MethodAttributes.Public | MethodAttributes.Abstract | MethodAttributes.Virtual,
typeof(string), new Type[] { typeof(int) }); ParameterBuilder m3p1 = m3.DefineParameter(, ParameterAttributes.HasDefault | ParameterAttributes.Optional, "arg1");
m3p1.SetConstant(); /* public abstract string mref(ref dobule arg2); */
MethodBuilder m4 = typeBuilder.DefineMethod("mref",
MethodAttributes.Public | MethodAttributes.Abstract | MethodAttributes.Virtual,
typeof(string), new Type[] { Type.GetType("System.Double&") }); ParameterBuilder m4p1 = m4.DefineParameter(, ParameterAttributes.Out, "arg2"); /* public abstract string mout(out int arg3 ); */
MethodBuilder m5 = typeBuilder.DefineMethod("mout",
MethodAttributes.Public | MethodAttributes.Abstract | MethodAttributes.Virtual,
typeof(string), new Type[] { typeof(int), typeof(int) }); ParameterBuilder m5p1 = m5.DefineParameter(, ParameterAttributes.Out, "arg3");
} static void InitAssembly()
{
AssemblyName assemblyName = new AssemblyName(namespaceName);
assemblyBuilder = AppDomain.CurrentDomain.DefineDynamicAssembly(assemblyName, AssemblyBuilderAccess.RunAndSave);
moduleBuilder = assemblyBuilder.DefineDynamicModule(assemblyName.Name, binaryName);
} static void SaveAssembly()
{
Type t = typeBuilder.CreateType(); //完成Type,这是必须的
assemblyBuilder.Save(binaryName);
}
}
}