gpt4 book ai didi

c# - Mono.Cecil:创建局部变量并更改返回语句

转载 作者:行者123 更新时间:2023-11-30 16:02:24 27 4
gpt4 key购买 nike

我正在尝试重写一个属性的 get 方法:

 get
{
return dataString;
}

到:

 get
{
string temp = dataString;
PropertyLogging.Get("DataString", ref temp);
return temp;
}

到目前为止,我已经尝试了不同的方法:

//the static method I#m trying to insert
var getMethod = att.AttributeType.Resolve().Methods.FirstOrDefault(x => x.Name == _getMethodName);
if (getMethod != null)
{
// ilProcessor.InsertBefore(returnInstruction, ilProcessor.Create(OpCodes.Starg, 1));

ilProcessor.InsertBefore(returnInstruction, ilProcessor.CreateLoadInstruction(property.Name));

// ilProcessor.InsertBefore(returnInstruction, ilProcessor.Create(OpCodes.Ldloc, 0));
ilProcessor.InsertBefore(returnInstruction, ilProcessor.Create(OpCodes.Ldloca_S, 0));

ilProcessor.InsertBefore(returnInstruction, ilProcessor.Create(OpCodes.Call, currentMethod.Module.ImportReference(getMethod)));

}

但它总是以这样的代码结尾(用 ilspy 反编译):

get
{
string text = this.dataString;
string arg_17_0 = text;
string text2;
PropertyLogging.Get("DataString", ref text2);
return arg_17_0;
}

我目前拥有的 IL 代码是:

IL_0000:  nop         
IL_0001: ldarg.0
IL_0002: ldfld UserQuery.dataString
IL_0007: stloc.0 // text
IL_0008: ldloc.0 // text
IL_0009: stloc.1 // arg_17_0
IL_000A: ldnull
IL_000B: stloc.2 // text2
IL_000C: ldstr "DataString"
IL_0011: ldloca.s 02 // text2
IL_0013: call UserQuery+PropertyLogging.Get
IL_0018: nop
IL_0019: ldloc.1 // arg_17_0
IL_001A: stloc.3
IL_001B: br.s IL_001D
IL_001D: ldloc.3
IL_001E: ret

但我需要的是:

IL_0000:  nop         
IL_0001: ldarg.0
IL_0002: ldfld UserQuery.dataString
IL_0007: stloc.0 // temp
IL_0008: ldstr "DataString"
IL_000D: ldloca.s 00 // temp
IL_000F: call UserQuery+PropertyLogging.Get
IL_0014: nop
IL_0015: ldloc.0 // temp
IL_0016: stloc.1
IL_0017: br.s IL_0019
IL_0019: ldloc.1
IL_001A: ret

我也试过在方法主体中创建一个新变量,但我不知道如何使用它。

有谁知道如何正确重写这个get方法?

提前致谢

最佳答案

之前的C#代码:

public string DataString
{
get { return _dataString; }
}

之前的 IL 代码:

.method public hidebysig specialname 
instance string get_DataString () cil managed
{
// Method begins at RVA 0x2050
// Code size 12 (0xc)
.maxstack 1
.locals init (
[0] string
)

IL_0000: nop
IL_0001: ldarg.0
IL_0002: ldfld string ClassLibrary1.Class1::_dataString
IL_0007: stloc.0
IL_0008: br.s IL_000a
IL_000a: ldloc.0
IL_000b: ret
} // end of method Class1::get_DataString

重写:

public void InterceptPropertyGetter()
{
// get the module where the property exist
var module = ModuleDefinition.ReadModule(@"C:\temp\ClassLibrary1.dll");

// get the property get method
TypeDefinition myType = module.Types.First(type => type.Name == "Class1");
var property =
myType.Properties.First(prop => prop.Name == "DataString").GetMethod;

// get the _dataString field
FieldDefinition dataStringDef =
myType.Fields.First(field => field.Name == "_dataString");
FieldReference dataStringRef = module.Import(dataStringDef);

// get the PropertyLogging static method
MethodDefinition propertyLoggingDef =
myType.Methods.First(method => method.Name == "PropertyLogging");
MethodReference propertyLoggingRef = module.Import(propertyLoggingDef);

// clear the method (variables and instructions )
property.Body.Variables.Clear();
property.Body.Instructions.Clear();

// define and init the locals
property.Body.InitLocals = true;
var tempVar = new VariableDefinition("temp", module.TypeSystem.String);
property.Body.Variables.Add(tempVar);

// write the IL
var processor = property.Body.GetILProcessor();
processor.Emit(OpCodes.Ldarg_0);
processor.Emit(OpCodes.Ldfld, dataStringRef);
processor.Emit(OpCodes.Stloc_0);
processor.Emit(OpCodes.Ldstr, "DataString");
processor.Emit(OpCodes.Ldloca_S, tempVar);
processor.Emit(OpCodes.Call, propertyLoggingRef);
processor.Emit(OpCodes.Ldloca_S, tempVar);
processor.Emit(OpCodes.Ret);
processor.Body.OptimizeMacros();

// save the new module
module.Write(@"C:\temp\ClassLibrary1_new.dll");
}

之后的 C# 代码:

public string DataString
{
get
{
string dataString = this._dataString;
Class1.PropertyLogging("DataString", ref dataString);
return dataString;
}
}

之后的 IL 代码:

.method public hidebysig specialname 
instance string get_DataString () cil managed
{
// Method begins at RVA 0x2050
// Code size 22 (0x16)
.maxstack 2
.locals init (
[0] string
)
IL_0000: ldarg.0
IL_0001: ldfld string ClassLibrary1.Class1::_dataString
IL_0006: stloc.0
IL_0007: ldstr "DataString"
IL_000c: ldloca.s 0
IL_000e: call void ClassLibrary1.Class1::PropertyLogging(string, string&)
IL_0013: ldloca.s 0
IL_0015: ret
} // end of method Class1::get_DataString

使用 ILSpy 反编译。

这不是唯一的方法(例如,你不必全部清除,你可以添加 nop 指令并使用指令的非短版本)但我认为它会帮助你理解如何做到这一点。

关于c# - Mono.Cecil:创建局部变量并更改返回语句,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37664948/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com