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();
}
}
}