繁体   English   中英

IL Emit-在notifypropertychanged更改之前,将现有属性设置为布尔值

[英]IL Emit - set an existing property with a boolean value before notifypropertychanged

我正在为包含虚拟自动属性的POCO对象实现一个发出的propertychanged处理程序,并且我的代码可以工作到每当我更改基础属性时引发propertychanged的地步。 这样做的原因是,我正在与服务器共享POCO对象(无论好坏),在那里我将修改后的对象发送到服务器。 我不能用属性修饰POCO对象(因为服务器也将具有这些修饰符,因为我们共享公共类),并且由于政策原因,我无法使用Fody或PostSharp等第三方工具。 我需要跟踪对象是否已被修改,而我一直坚持下去。

这是用更改通知包装我的虚拟自动属性的发射:

    MethodBuilder setMethodBuilder = typeBuilder.DefineMethod(setMethod.Name, setMethod.Attributes, setMethod.ReturnType, types.ToArray());
    typeBuilder.DefineMethodOverride(setMethodBuilder, setMethod);
    ILGenerator wrapper = setMethodBuilder.GetILGenerator();

    ...Emit if property <> value IsModified=true here...

    wrapper.Emit(OpCodes.Ldarg_0);
    wrapper.Emit(OpCodes.Ldarg_1);
    wrapper.EmitCall(OpCodes.Call, setMethod, null);

我需要做的是获取现有“ IsModified”布尔属性的set方法,并在属性值<>值的情况下进行设置。

这是我要发出的示例(当前定义为具有虚拟自动属性的POCO):

public class AnEntity
{
    string _myData;
    public string MyData
    {
        get
        {
            return _myData;
        }
        set
        {
            if(_myData <> value) 
            {
                IsModified = true;
                _myData = value;
                OnPropertyChanged("MyData");                
            }
        }
    }

    bool _isModified;
    public bool IsModified { get; set; }
    {
        get
        {
            return _isModified;
        }
        set
        {
            _isModified = value;
            OnPropertyChanged("IsModified");
        }
    }
}

我已经坚持了一段时间……我设法在创建的新代理类中创建一个名为“ NewIsModified”的新属性,但是,我非常想在我的数据库中重用现有的IsModified属性。原始POCO。

我希望我已经正确解释了我的问题,并且易于理解。 任何帮助将不胜感激,我希望它也能帮助其他人。

亲切的问候。

这是在Mono.Cecil中执行此操作的工作代码

之前的C#代码:

public class AnEntityVirtual
{
    public virtual string MyData { get; set; }
    public virtual bool IsModified { get; set; }
}

set_MyData IL代码之前:

IL_0000: ldarg.0
IL_0001: ldarg.1
IL_0002: stfld string ClassLibrary1.AnEntityVirtual::'<MyData>k__BackingField'
IL_0007: ret

改写:

// Read the module and get the relevant type
var assemblyPath = $"{Environment.CurrentDirectory}\\ClassLibrary1.dll";
var module = ModuleDefinition.ReadModule(assemblyPath);
var type = module.Types.Single(t => t.Name == "AnEntityVirtual");

// Get the method to rewrite
var myDataProperty = type.Properties.Single(prop => prop.Name == "MyData");
var isModifiedSetMethod = type.Properties.Single(prop => prop.Name == "IsModified").SetMethod;
var setMethodBody = myDataProperty.SetMethod.Body;

// Initilize before rewriting (clear pre instructions, create locals and init them)
setMethodBody.Instructions.Clear();
var localDef = new VariableDefinition(module.TypeSystem.Boolean);
setMethodBody.Variables.Add(localDef);
setMethodBody.InitLocals = true;

 // Get fields\methos to use in the new method body
 var propBackingField = type.Fields.Single(field => field.Name == $"<{myDataProperty.Name}>k__BackingField");
var equalMethod =
            myDataProperty.PropertyType.Resolve().Methods.FirstOrDefault(method => method.Name == "Equals") ??
            module.ImportReference(typeof(object)).Resolve().Methods.Single(method => method.Name == "Equales");
var equalMethodReference = module.ImportReference(equalMethod);

// Start the rewriting
var ilProcessor = setMethodBody.GetILProcessor();

// First emit a Ret instruction. This is beacause we want a label to jump if the values are equals
ilProcessor.Emit(OpCodes.Ret);
var ret = setMethodBody.Instructions.First();

ilProcessor.InsertBefore(ret, ilProcessor.Create(OpCodes.Ldarg_0)); // load 'this'
ilProcessor.InsertBefore(ret, ilProcessor.Create(OpCodes.Ldfld, propBackingField)); // load backing field
ilProcessor.InsertBefore(ret, ilProcessor.Create(OpCodes.Ldarg_1)); // load 'value'
ilProcessor.InsertBefore(ret, ilProcessor.Create(equalMethod.IsStatic ? OpCodes.Call : OpCodes.Callvirt, equalMethodReference)); // call equals
ilProcessor.InsertBefore(ret, ilProcessor.Create(OpCodes.Stloc_0)); // store result
ilProcessor.InsertBefore(ret, ilProcessor.Create(OpCodes.Ldloc_0)); // load result
ilProcessor.InsertBefore(ret, ilProcessor.Create(OpCodes.Brtrue_S, ret)); // check result and jump to Ret if are equals
ilProcessor.InsertBefore(ret, ilProcessor.Create(OpCodes.Ldarg_0)); // load 'this'
ilProcessor.InsertBefore(ret, ilProcessor.Create(OpCodes.Ldc_I4_1)); // load 1 ('true')
ilProcessor.InsertBefore(ret, ilProcessor.Create(OpCodes.Call, isModifiedSetMethod)); // set IsModified to 'true'
ilProcessor.InsertBefore(ret, ilProcessor.Create(OpCodes.Ldarg_0)); // load this
ilProcessor.InsertBefore(ret, ilProcessor.Create(OpCodes.Ldarg_1)); // load 'value'
ilProcessor.InsertBefore(ret, ilProcessor.Create(OpCodes.Stfld, propBackingField)); // store 'value' in backing field
// here you can call to Notify or whatever you want
module.Write(assemblyPath.Replace(".dll", "_new") + ".dll"); // save the new assembly

以下的C#代码:

public virtual string MyData
{
    [CompilerGenerated]
    get
    {
        return this.<MyData>k__BackingField;
    }
    [CompilerGenerated]
    set
    {
        if (!this.<MyData>k__BackingField.Equals(value))
        {
            this.IsModified = true;
            this.<MyData>k__BackingField = value;
        }
    }
}

IL代码后:

IL_0000: ldarg.0
IL_0001: ldfld string ClassLibrary1.AnEntityVirtual::'<MyData>k__BackingField'
IL_0006: ldarg.1
IL_0007: callvirt instance bool [mscorlib]System.String::Equals(object)
IL_000c: stloc.0
IL_000d: ldloc.0
IL_000e: brtrue.s IL_001e

IL_0010: ldarg.0
IL_0011: ldc.i4.1
IL_0012: call instance void ClassLibrary1.AnEntityVirtual::set_IsModified(bool)
IL_0017: ldarg.0
IL_0018: ldarg.1
IL_0019: stfld string ClassLibrary1.AnEntityVirtual::'<MyData>k__BackingField'

IL_001e: ret

如我所写,这是在Cecil中执行此操作的示例。 在您的真实代码中,您可以在此基础上进行一些更改。

例如,您可以为属性创建私有字段,而不使用编译器生成的后备字段。

您可以致电OptimizeMacros。

另外,如果您确切知道需要重写的属性,则可以调用其他equal方法,例如,如果是string ,则可以调用字符串op_Equalityop?_Inequality类型的静态方法,这是string==!=

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM