且构网

分享程序员开发的那些事...
且构网 - 分享程序员编程开发的那些事

编译时MSIL注入--实践Mono Cecil(1)

更新时间:2021-12-22 20:00:06

紧接上两篇浅谈.NET编译时注入(C#-->IL)浅谈VS编译自定义编译任务—MSBuild Task(csproject),在第一篇中我们简单研究了c#语法糖和PostSharp的MSIl注入,紧接第二篇中我们介绍了自定义MSBuild编译任务(记得有位老兄发链接用 MSBuild自动发布Silverlight xap ,我想说的我做的是自定义编译任务,不是什么发布,MSBuild本就是一个发布工具)。之所以在此前介绍编译Task是因为我讲介绍的就是利用MSBuild和MSILInject制作静态注入式AOP,想成熟的产品PostSharp,当然我也不会去重造***,但需要明白起原理和自动化注入时机。废话不多说,今天将请出我们的MSIL注入的好东西:Mono.Cecil.官方网站http://www.mono-project.com/Cecil,他是一个强大的MSIL 注入工具,在我们的Reflector插件Reflexil(动态修改程序集插件,很好用,我已经尝试多次)就有他的身影出现。还有如大名鼎鼎的SharpDevelopLINQPadJa.NET等等(可以参见https://github.com/jbevain/cecil/wiki/Users)。

      在本节我们需要看看这个Mono.Cecil,先来一个简单的认识。

我来在我们的方法执行前后加入我们的输出信息:

原来代码:


  1. class Program   
  2.    {   
  3.        static void Main(string[] args)   
  4.        {   
  5.            Console.WriteLine("破浪Blog:http://www.cnblogs.com/whitewolf/");   
  6.        }   
  7.    } 

任务:

1我将在方法执行前后添加一个Console.WriteLine("Method start…");

2方法最后添加Console.WriteLine("Method finish…“);

具体Mono.Cecil Code:


  1. using System;   
  2. using System.Collections.Generic;   
  3. using System.Linq;   
  4. using System.Text;   
  5. using Mono.Cecil;   
  6. using Mono.Cecil.Cil;   
  7.  
  8. namespace BlogSample   
  9. {   
  10.     class Program   
  11.     {   
  12.         static void Main(string[] args)   
  13.         {   
  14.             AssemblyDefinition assembiy = AssemblyFactory.GetAssembly(args[0]);   
  15.             foreach (Mono.Cecil.TypeDefinition item in assembiy.MainModule.Types)   
  16.             {   
  17.                 foreach (MethodDefinition method in item.Methods)   
  18.                 {   
  19.                     if (method.Name.Equals("Main"))   
  20.                     {   
  21.  
  22.                         var ins = method.Body.Instructions[0];   
  23.                         var worker = method.Body.CilWorker;   
  24.                         worker.InsertBefore(ins, worker.Create(OpCodes.Ldstr, "Method start…"));   
  25.                         worker.InsertBefore(ins, worker.Create(OpCodes.Call,   
  26.                             assembiy.MainModule.Import(typeof(Console).GetMethod("WriteLine", new Type[] { typeof(string) }))));   
  27.                         ins = method.Body.Instructions[method.Body.Instructions.Count - 1];   
  28.  
  29.                         worker.InsertBefore(ins, worker.Create(OpCodes.Ldstr, "Method finish…"));   
  30.                         worker.InsertBefore(ins, worker.Create(OpCodes.Call,   
  31.                             assembiy.MainModule.Import(typeof(Console).GetMethod("WriteLine", new Type[] { typeof(string) }))));   
  32.                         break;   
  33.                     }   
  34.                 }   
  35.  
  36.             }   
  37.  
  38.             AssemblyFactory.SaveAssembly(assembiy, "IL_" + args[0]);   
  39.             Console.Read();   
  40.         }   
  41.     }   

DOS运行结果:

编译时MSIL注入--实践Mono Cecil(1)

我们在来看看反编译后的MSIL

编译时MSIL注入--实践Mono Cecil(1)

C#:

编译时MSIL注入--实践Mono Cecil(1)

在这最后我们可以想一下,如果我们利用Mono.Cecil可以干些什么事情,能做的当然很多,我首先想尝试的了与上一节浅谈VS编译自定义编译任务—MSBuild Task(csproject)结合PostSharp一样的静态注入AOP框架。还能做什么的就要靠大家发挥大家聪明的才智。

 




 本文转自 破狼 51CTO博客,原文链接:http://blog.51cto.com/whitewolfblog/835126,如需转载请自行联系原作者