C#代码生成与注入:使用MSIL Emit技术解析

版权申诉
0 下载量 38 浏览量 更新于2024-06-29 收藏 724KB PDF 举报
"本文主要探讨了如何使用MSIL(Microsoft Intermediate Language)和Emit技术来实现C#的代码生成与注入。文中提到这种方式主要用于在没有源码的情况下对.NET体系的EXE和DLL文件进行修改,以添加新功能或修改原有功能。文章特别提到了在Unity3D中使用LUA进行热更新的场景,并介绍了两种代码注入方案,重点讲解了微软提供的API函数方法,因为它相对易于理解和操作。作者建议对C++反汇编语言有一定了解的读者会更容易上手,但即使不熟悉,也能通过学习MSIL和相关的工具如ILSpy来掌握这一技术。ILSpy是一个用于浏览和反编译.NET程序的开源工具,可以帮助开发者将C#二进制文件转换为MSIL或C#源代码,以便于参考和实现Emit功能。" 在深入探讨之前,先理解MSIL的重要性。MSIL是.NET框架下的低级人类可读语言,所有.NET语言(如C#、VB.NET、F#等)编译后都会转化为MSIL,这是一种基于堆栈的、面向对象的语言。它为跨语言的交互提供了可能,因为所有语言都被编译成统一的中间语言,然后由Common Language Runtime (CLR) 负责解释和执行。 在C#中,System.Reflection.Emit命名空间提供了API来动态生成MSIL,这被称为Emit技术。Emit允许程序员在运行时创建类型、方法和属性,然后动态生成IL代码,从而实现代码注入。例如,可以使用Emit创建一个新的静态方法,该方法在运行时检查并执行预定义的LUA脚本,以实现热更新。 以下是使用Emit实现代码生成的基本步骤: 1. **定义元数据**: 首先,需要定义新类型或方法的元数据,包括类型名称、访问修饰符、基类、接口以及方法签名等。 2. **创建ILGenerator**: 使用TypeBuilder或MethodBuilder对象创建ILGenerator实例,这是用于生成IL代码的对象。 3. **Emit IL指令**: 使用ILGenerator的Emit方法添加IL指令,这些指令对应于C#或其他.NET语言的语句。例如,Emit(OpCodes.Ldstr, "Hello")表示加载字符串"Hello"到堆栈。 4. **完成构建**: 当所有IL指令添加完成后,调用TypeBuilder或MethodBuilder的Finish方法来完成类型或方法的构建。 5. **生成类型**: 最后,使用AssemblyBuilder的Save方法将生成的类型保存到磁盘或内存中。 在实际应用中,例如Unity3D的热更新,可以利用Emit在特定类或方法的入口处插入检查和执行LUA脚本的代码,从而实现对已有功能的热更新,而无需重新编译整个项目。 ILSpy作为一个强大的工具,它不仅可以用来查看和理解已编译的.NET程序集,还可以反编译为接近原始源代码的C#,这对于学习和调试使用Emit生成的代码非常有用。在编写Emit代码时,如果不清楚对应的MSIL指令,可以参考ILSpy反编译的代码,找到相似的功能,然后在自己的Emit代码中复制相应的IL指令。 使用MSIL和Emit实现C#的代码生成与注入是一种灵活且强大的技术,尤其适用于需要动态修改或扩展.NET程序的情况。通过学习和实践,开发者可以掌握这项技术,提高代码的灵活性和适应性。
2466 浏览量
用C#实现生成PDF文档(附源码) 收藏 //write by wenhui.org using System; using System.IO; using System.Text; using System.Collections; namespace PDFGenerator { public class PDFGenerator { static float pageWidth = 594.0f; static float pageDepth = 828.0f; static float pageMargin = 30.0f; static float fontSize = 20.0f; static float leadSize = 10.0f; static StreamWriter pPDF=new StreamWriter("E:\myPDF.pdf"); static MemoryStream mPDF= new MemoryStream(); static void ConvertToByteAndAddtoStream(string strMsg) { Byte[] buffer=null; buffer=ASCIIEncoding.ASCII.GetBytes(strMsg); mPDF.Write(buffer,0,buffer.Length); buffer=null; } static string xRefFormatting(long xValue) { string strMsg =xValue.ToString(); int iLen=strMsg.Length; if (iLen<10) { StringBuilder s=new StringBuilder(); int i=10-iLen; s.Append('0',i); strMsg=s.ToString() + strMsg; } return strMsg; } static void Main(string[] args) { ArrayList xRefs=new ArrayList(); //Byte[] buffer=null; float yPos =0f; long streamStart=0; long streamEnd=0; long streamLen =0; string strPDFMessage=null; //PDF文档头信息 strPDFMessage="%PDF-1.1 "; ConvertToByteAndAddtoStream(strPDFMessage); xRefs.Add(mPDF.Length); strPDFMessage="1 0 obj "; ConvertToByteAndAddtoStream(strPDFMessage); strPDFMessage="<> "; ConvertToByteAndAddtoStream(strPDFMessage); strPDFMessage="stream "; ConvertToByteAndAddtoStream(strPDFMessage); ////////PDF文档描述 streamStart=mPDF.Length; //字体 strPDFMessage="BT /F0 " + fontSize +" Tf "; ConvertToByteAndAddtoStream(strPDFMessage); //PDF文档实体高度 yPos = pageDepth - pageMargin; strPDFMessage=pageMargin + " " + yPos +" Td " ; ConvertToByteAndAddtoStream(strPDFMessage); strPDFMessage= leadSize+" TL " ; ConvertToByteAndAddtoStream(strPDFMessage); //实体内容 strPDFMessage= "(http://www.wenhui.org)Tj " ; ConvertToByteAndAddtoStream(strPDFMessage); strPDFMessage= "ET "; ConvertToByteAndAddtoStream(strPDFMessage); streamEnd=mPDF.Length; streamLen=streamEnd-streamStart; strPDFMessage= "endstream endobj "; ConvertToByteAndAddtoStream(strPDFMessage); //PDF文档的版本信息 xRefs.Add(mPDF.Length); strPDFMessage="2 0 obj "+ streamLen + " endobj "; ConvertToByteAndAddtoStream(strPDFMessage); xRefs.Add(mPDF.Length); strPDFMessage="3 0 obj <> endobj "; ConvertToByteAndAddtoStream(strPDFMessage); xRefs.Add(mPDF.Length); strPDFMessage="4 0 obj <</Type /Pages /Count 1 "; ConvertToByteAndAddtoStream(strPDFMessage); strPDFMessage="/Kids[ 3 0 R ] "; ConvertToByteAndAddtoStream(strPDFMessage); strPDFMessage="/Resources<</ProcSet[/PDF/Text]/Font<> >> "; ConvertToByteAndAddtoStream(strPDFMessage); strPDFMessage="/MediaBox [ 0 0 "+ pageWidth + " " + pageDepth + " ] >> endobj "; ConvertToByteAndAddtoStream(strPDFMessage); xRefs.Add(mPDF.Length); strPDFMessage="5 0 obj <> endobj "; ConvertToByteAndAddtoStream(strPDFMessage); xRefs.Add(mPDF.Length); strPDFMessage="6 0 obj <> endobj "; ConvertToByteAndAddtoStream(strPDFMessage); streamStart=mPDF.Length; strPDFMessage="xref 0 7 0000000000 65535 f "; for(int i=0;i<xRefs.Count;i++) { strPDFMessage+=xRefFormatting((long) xRefs[i])+" 00000 n "; } ConvertToByteAndAddtoStream(strPDFMessage); strPDFMessage="trailer <> "; ConvertToByteAndAddtoStream(strPDFMessage); strPDFMessage="startxref " + streamStart+" %%EOF "; ConvertToByteAndAddtoStream(strPDFMessage); mPDF.WriteTo(pPDF.BaseStream); mPDF.Close(); pPDF.Close(); } } }