MSIL实用指南-生成索引器
MSIL实用指南-生成索引器
索引器是一种特殊的属性,它有参数的,也有get和set方法,属性名称一般是"Item",并且方法名称一般名称是"get_Item"和"set_Item"。
下面我们来生成如下的带有索引器的类

using System; namespace LX1_ILDemo { public class IndexerDemo { private string[,] _CustomerNames; public string this[int, int] { get { return this._CustomerNames[num, num2]; } set { this._CustomerNames[num, num2] = value; } } } }
第一步,生成一个字段
生成字段用TypeBuilder.DefineField方法。
实例代码:
FieldBuilder CustomerNamesBldr = typeBuilder.DefineField
("_CustomerNames",typeof(string[,]),FieldAttributes.Private);
第二步,创建一个名称为Item的PropertyBuilder对象
创建PropertyBuilder对象需要用生成字段用方法
PropertyBuilder custNamePropBldr = typeBuilder.DefineProperty ("Item", PropertyAttributes.HasDefault,typeof(string),new Type[] { typeof(int), typeof(int) });
注意定义这个PropertyBuilder对象是有参数new Type[] { typeof(int), typeof(int) }的。
第三步,生成get和set方法
方法名为"get_Item"和"set_Item"。
MethodAttributes getSetAttr = MethodAttributes.Public | MethodAttributes.SpecialName | MethodAttributes.HideBySig; // 定义get方法 MethodBuilder custNameGetPropMthdBldr = typeBuilder.DefineMethod("get_Item", getSetAttr, typeof(string) , new Type[] { typeof(int), typeof(int) }); ILGenerator ilGetGenerator = custNameGetPropMthdBldr.GetILGenerator(); LocalBuilder localBuilderv1 = ilGetGenerator.DeclareLocal(typeof(string)); ilGetGenerator.Emit(OpCodes.Ldarg_0); ilGetGenerator.Emit(OpCodes.Ldfld, CustomerNamesBldr); ilGetGenerator.Emit(OpCodes.Ldarg_1); ilGetGenerator.Emit(OpCodes.Ldarg_2); ilGetGenerator.Emit(OpCodes.Call ,typeof(string[,]).GetMethod("Get",new Type[] { typeof(int), typeof(int) })); ilGetGenerator.Emit(OpCodes.Stloc_0); ilGetGenerator.Emit(OpCodes.Ldloc_0); ilGetGenerator.Emit(OpCodes.Ret); // 定义set方法 MethodBuilder custNameSetPropMthdBldr = typeBuilder.DefineMethod("set_Item", getSetAttr, null, new Type[] { typeof(int), typeof(int), typeof(string) }); ILGenerator ilSetGenerator = custNameSetPropMthdBldr.GetILGenerator(); ilSetGenerator.Emit(OpCodes.Ldarg_0); ilSetGenerator.Emit(OpCodes.Ldfld, CustomerNamesBldr); ilSetGenerator.Emit(OpCodes.Ldarg_1); ilSetGenerator.Emit(OpCodes.Ldarg_2); ilSetGenerator.Emit(OpCodes.Ldarg_3); ilSetGenerator.Emit(OpCodes.Call, typeof(string[,]).GetMethod("Set", new Type[] { typeof(int), typeof(int), typeof(string) })); ilSetGenerator.Emit(OpCodes.Ret); custNamePropBldr.SetGetMethod(custNameGetPropMthdBldr); custNamePropBldr.SetSetMethod(custNameSetPropMthdBldr);
第四步,给类加DefaultMemberAttribute特性
用反射查找DefaultMemberAttribute类型,传入参数值为"Item",创建一个DefaultMemberAttribute实例,
源码如下
Type myType = typeof(DefaultMemberAttribute); ConstructorInfo infoConstructor = myType.GetConstructor(new Type[] { typeof(string) }); CustomAttributeBuilder attributeBuilder = new CustomAttributeBuilder(infoConstructor, new object[] {"Item" }); typeBuilder.SetCustomAttribute(attributeBuilder);
完整程序如下:

using System; using System.Reflection; using System.Reflection.Emit; namespace LX1_ILDemo { class Demo12_Indexer { static string binaryName = "Demo12_Indexer.dll"; static string namespaceName = "LX1_ILDemo"; static string typeName = "IndexerDemo"; static AssemblyBuilder assemblyBuilder; static ModuleBuilder moduleBuilder; static TypeBuilder typeBuilder; private static void Generate_Indexer() { FieldBuilder CustomerNamesBldr = typeBuilder.DefineField ("_CustomerNames",typeof(string[,]),FieldAttributes.Private); PropertyBuilder custNamePropBldr = typeBuilder.DefineProperty ("Item", PropertyAttributes.HasDefault,typeof(string),new Type[] { typeof(int), typeof(int) }); MethodAttributes getSetAttr = MethodAttributes.Public | MethodAttributes.SpecialName | MethodAttributes.HideBySig; // 定义get方法 MethodBuilder custNameGetPropMthdBldr = typeBuilder.DefineMethod("get_Item", getSetAttr, typeof(string) , new Type[] { typeof(int), typeof(int) }); ILGenerator ilGetGenerator = custNameGetPropMthdBldr.GetILGenerator(); LocalBuilder localBuilderv1 = ilGetGenerator.DeclareLocal(typeof(string)); ilGetGenerator.Emit(OpCodes.Ldarg_0); ilGetGenerator.Emit(OpCodes.Ldfld, CustomerNamesBldr); ilGetGenerator.Emit(OpCodes.Ldarg_1); ilGetGenerator.Emit(OpCodes.Ldarg_2); ilGetGenerator.Emit(OpCodes.Call ,typeof(string[,]).GetMethod("Get",new Type[] { typeof(int), typeof(int) })); ilGetGenerator.Emit(OpCodes.Stloc_0); ilGetGenerator.Emit(OpCodes.Ldloc_0); ilGetGenerator.Emit(OpCodes.Ret); // 定义set方法 MethodBuilder custNameSetPropMthdBldr = typeBuilder.DefineMethod("set_Item", getSetAttr, null, new Type[] { typeof(int), typeof(int), typeof(string) }); ILGenerator ilSetGenerator = custNameSetPropMthdBldr.GetILGenerator(); ilSetGenerator.Emit(OpCodes.Ldarg_0); ilSetGenerator.Emit(OpCodes.Ldfld, CustomerNamesBldr); ilSetGenerator.Emit(OpCodes.Ldarg_1); ilSetGenerator.Emit(OpCodes.Ldarg_2); ilSetGenerator.Emit(OpCodes.Ldarg_3); ilSetGenerator.Emit(OpCodes.Call, typeof(string[,]).GetMethod("Set", new Type[] { typeof(int), typeof(int), typeof(string) })); ilSetGenerator.Emit(OpCodes.Ret); custNamePropBldr.SetGetMethod(custNameGetPropMthdBldr); custNamePropBldr.SetSetMethod(custNameSetPropMthdBldr); SetTypeAttr(); } private static void SetTypeAttr() { Type myType = typeof(DefaultMemberAttribute); ConstructorInfo infoConstructor = myType.GetConstructor(new Type[] { typeof(string) }); CustomAttributeBuilder attributeBuilder = new CustomAttributeBuilder(infoConstructor, new object[] {"Item" }); typeBuilder.SetCustomAttribute(attributeBuilder); } public static void Generate() { InitAssembly(); typeBuilder = moduleBuilder.DefineType( namespaceName+"."+ typeName, TypeAttributes.Public); Generate_Indexer(); SaveAssembly(); Console.WriteLine("生成成功"); } 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); } } }
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· 分享 3 个 .NET 开源的文件压缩处理库,助力快速实现文件压缩解压功能!
· Ollama——大语言模型本地部署的极速利器
· DeepSeek如何颠覆传统软件测试?测试工程师会被淘汰吗?