热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

使用Cecil修改.Net程序集

Cecil是Mono的一个子项目,用于对程序集进行读写,并且已经用于Mono的调试,Reflector也使用它作为底层库。最近把DbEntry使用Emit生成程序集的方式,改

Cecil 是 Mono 的一个子项目,用于对程序集进行读写,并且已经用于 Mono 的调试,Reflector 也使用它作为底层库。最近把 DbEntry 使用 Emit 生成程序集的方式,改成了使用 Cecil 的方式,就我的感受来说,Cecil 是比较优秀的,有一些地方,比 Emit 使用起来还舒服的多;不过,有一些地方也比较繁琐。

  我使用的是 Git 里的最新版本,如果大家要测试的话,也建议使用 Git 版,所以,需要安装一个 Git 客户端。

  这里,用一个非常简单的例子,说明一下 Cecil 的基本用法。

  首先,我们编写一个测试用的程序集 TestApp.exe :

using System;
namespace TestApp
{
  class Program
  {
    static void Main()
    {
      Console.WriteLine("Main");
    }
    private static void Before()
    {
      Console.WriteLine("Before");
    }
    private static void After()
    {
      Console.WriteLine("After");
    }
  }
}

  然后,编写一个使用 Cecil 进行改写的应用 CecilTest.exe :

using System;
using System.Linq;
using Mono.Cecil;
using Mono.Cecil.Cil;
namespace CecilTest
{
  class Program
  {
    static void Main(string[] args)
    {
      if(args.Length != 1)
      {
        Console.WriteLine("Usage: CecilTest TestApp.exe");
      }
      var m = ModuleDefinition.ReadModule(args[0]);
      var prog = m.Types.First(p => p.Name == "Program");
      var main = prog.Methods.First(p => p.Name == "Main");
      var before = prog.Methods.First(p => p.Name == "Before");
      var after = prog.Methods.First(p => p.Name == "After");
      var il = main.Body.GetILProcessor();
      il.InsertBefore(main.Body.Instructions[0], il.Create(OpCodes.Call, before));
      il.InsertBefore(main.Body.Instructions.Last(), il.Create(OpCodes.Call, after));
      m.Write(args[0] + ".exe");
      Console.WriteLine("Done");
      Console.ReadLine();
    }
  }
}

  编译这两个项目,并且使用 CecilTest.exe 处理一下 TestApp.exe,生成 TestApp.exe.exe,运行 TestApp.exe,运行结果:

  Main

  运行 TestApp.exe.exe,运行结果:

  Before

  Main

  After

  可以看到,我们已经成功的在 Main 函数的前后,分别插入了一次函数调用。

  基本使用方法就是这样,大体和 Emit 类似,只是不止可以使用 Emit 函数,还可以直接修改程序集。当然,还有其它一些细节的不同,比如它的变量定义不是通过 ILProcessor,而是直接操作函数体的 Variables;再比如它没有 DeclareLabel 函数,跳转直接引用函数体的 Instruction 进行。另外,它可以只加载目标程序集,却不加载其依赖项,所以很多东西都分定义和引用两种。

  就目前来说,我认为它的效果是令人满意的。不过它最大的问题,在于泛型处理上。不是说不能做,而是太繁琐,有时候甚至是不可能。在 DbEntry 中,最后被泛型击败,采取了 Reflection+Cecil 的方式,这种方式简单易行,不过问题是,Reflection 需要加载程序集,除了可能出现无法加载依赖项的异常外,也无法简单的回写原文件。我在 Cecil 的 Git 上提交了 issue,不过作者回复,不觉得这是问题,所以我也懒得纠缠了。



推荐阅读
  • golang常用库:配置文件解析库/管理工具viper使用
    golang常用库:配置文件解析库管理工具-viper使用-一、viper简介viper配置管理解析库,是由大神SteveFrancia开发,他在google领导着golang的 ... [详细]
  • 1:有如下一段程序:packagea.b.c;publicclassTest{privatestaticinti0;publicintgetNext(){return ... [详细]
  • 本文介绍了Java并发库中的阻塞队列(BlockingQueue)及其典型应用场景。通过具体实例,展示了如何利用LinkedBlockingQueue实现线程间高效、安全的数据传递,并结合线程池和原子类优化性能。 ... [详细]
  • 本文介绍了如何使用JQuery实现省市二级联动和表单验证。首先,通过change事件监听用户选择的省份,并动态加载对应的城市列表。其次,详细讲解了使用Validation插件进行表单验证的方法,包括内置规则、自定义规则及实时验证功能。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • 扫描线三巨头 hdu1928hdu 1255  hdu 1542 [POJ 1151]
    学习链接:http:blog.csdn.netlwt36articledetails48908031学习扫描线主要学习的是一种扫描的思想,后期可以求解很 ... [详细]
  • 本文介绍了如何使用 Spring Boot DevTools 实现应用程序在开发过程中自动重启。这一特性显著提高了开发效率,特别是在集成开发环境(IDE)中工作时,能够提供快速的反馈循环。默认情况下,DevTools 会监控类路径上的文件变化,并根据需要触发应用重启。 ... [详细]
  • CentOS7源码编译安装MySQL5.6
    2019独角兽企业重金招聘Python工程师标准一、先在cmake官网下个最新的cmake源码包cmake官网:https:www.cmake.org如此时最新 ... [详细]
  • 数据库内核开发入门 | 搭建研发环境的初步指南
    本课程将带你从零开始,逐步掌握数据库内核开发的基础知识和实践技能,重点介绍如何搭建OceanBase的开发环境。 ... [详细]
  • 使用 Azure Service Principal 和 Microsoft Graph API 获取 AAD 用户列表
    本文介绍了一段通用代码示例,该代码不仅能够操作 Azure Active Directory (AAD),还可以通过 Azure Service Principal 的授权访问和管理 Azure 订阅资源。Azure 的架构可以分为两个层级:AAD 和 Subscription。 ... [详细]
  • 本文深入探讨了 Java 中的 Serializable 接口,解释了其实现机制、用途及注意事项,帮助开发者更好地理解和使用序列化功能。 ... [详细]
  • CMake跨平台开发实践
    本文介绍如何使用CMake支持不同平台的代码编译。通过一个简单的示例,我们将展示如何编写CMakeLists.txt以适应Linux和Windows平台,并实现跨平台的函数调用。 ... [详细]
  • 本文介绍了如何在C#中启动一个应用程序,并通过枚举窗口来获取其主窗口句柄。当使用Process类启动程序时,我们通常只能获得进程的句柄,而主窗口句柄可能为0。因此,我们需要使用API函数和回调机制来准确获取主窗口句柄。 ... [详细]
  • 本文详细介绍了如何在 Spring Boot 应用中通过 @PropertySource 注解读取非默认配置文件,包括配置文件的创建、映射类的设计以及确保 Spring 容器能够正确加载这些配置的方法。 ... [详细]
  • 本文详细介绍了Java中org.w3c.dom.Text类的splitText()方法,通过多个代码示例展示了其实际应用。该方法用于将文本节点在指定位置拆分为两个节点,并保持在文档树中。 ... [详细]
author-avatar
手机用户2702936867
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有