我怎样才能变成这样的字符串:
"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/