我怎样才能像这样转换字符串:
"call System.Console.WriteLine"
"ldstr \"hello\""
进入带有操作数的Intructions?
答案 0 :(得分:1)
如果你现在如何使用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
,你需要在那里解析string并发出调用指令
伪代码:
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
是一个帮助程序类,它在特定模块中查找类型\。
所以你需要为方法体创建方法和ILProccesor
,然后对于你拥有的每个字符串,你需要将操作数的指令操作码分开,然后在字典中查找所需的操作并传递操作码,操作数为字符串和ILProccesor
。