我怎样才能变成这样的字符串:

"call        System.Console.WriteLine"
"ldstr       \"hello\""


进入带有操作数的指令?

最佳答案

如果现在您如何使用Mono.Cecil(或Reflection.Emit),那么问题通常是关于将文本解析为代码操作。

您有一些方法可以做到这一点,我可以向您显示一个提示,您可以选择自己的方法。

首先,您需要一些先决条件(如果您的IL文本是有效的IL代码,则这些先决条件已经为您提供)。
例如,您无法猜测Console.WriteLine是什么。 Console是程序集,类型,方法吗?关于WriteLine的相同问题。即使我们知道WriteLine是一种方法,我们也需要选择哪个重载?那泛型呢?因此,您需要设置一个合同,例如定义点是定界符,第一部分是程序集,第二部分是名称空间,依此类推。

例如:

"mscorlib.System.Console.WriteLine(string)"将被翻译为System.Console.WriteLine(string)

签订了严格的合同后,您需要执行一些步骤(对于WriteLine示例):


解析Assembly并得到ModuleDefinition
解决并导入Type
解决并导入MethodReference描述请求的方法
发出呼叫


一种方法是保持操作码的结构及其所需的操作。

例如,我们知道call指令需要发出对静态方法的调用(在大多数情况下),因此您需要将操作数发送至ParseStaticCall,在那里您需要解析字符串并发出调用指令

伪代码:

new Dictionary<string, Tuple<OpCode, Action<string>>>
{
    {
        "call",
        Tuple.Create<OpCode, Action<string>>
          (OpCodes.Call,ParseStaticCall)
    }
};


static void ParseStaticCall(Opcpde opcode,
                            string call,
                            ILProcessor processor)
{
    string assembly, namespaceName, type, method;
    int numOfParameters;

    var moduleDefenition = AssemblyResolver.Resolve(assembly).MainModule;

    var methodReference =
        new ReferenceFinder(moduleDefenition).
            GetMethodReference(typeof (Console),
                md => md.Name == methodName &&
                md.Parameters.Count == numOfParameters);
        processor.Emit(opcode, methodReference);
}


AssemblyResolver是一个帮助程序类,用于通过名称和路径(可以是恒定路径)查找程序集。 ReferenceFinder是一个帮助程序类,可在特定模块中查找类型\方法。

因此,您需要为方法主体创建method和ILProccesor,然后对于每个字符串,都需要将指令操作码与操作数分开,然后在字典中查找所需的操作并将操作码作为字符串传递和ILProccesor

关于c# - Mono.Cecil:从字符串创建指令,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/40326466/

10-16 09:03