- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
首先,我有一个类
internal class Parent
{
protected void Write(string message, [CallerMemberName] string caller = null)
{
Console.WriteLine($"{caller} :: {message}");
}
}
我想动态创建一个类,该类有一个属性“Name”,如果属性值改变然后调用 write 方法,看起来像
class Child : Parent
{
private string _name;
public string Name
{
get { return _name; }
set
{
if (_name != value)
{
Write("changedto: " + value);
_name = value;
}
}
}
}
我想知道的是高亮条款。如何在 emit 中写入它。请帮忙。
我想要下面的一些代码
private static MethodBuilder BuildSetter(TypeBuilder typeBuilder, PropertyInfo property, FieldBuilder fieldBuilder, MethodAttributes attributes)
{
var setterBuilder = typeBuilder.DefineMethod($"set_{property.Name}", attributes, null, new Type[] { property.PropertyType });
var ilGenerator = setterBuilder.GetILGenerator();
ilGenerator.Emit(OpCodes.Ldarg_0); //this
ilGenerator.Emit(OpCodes.Ldarg_1); // the first one in arguments list
//code should be here
ilGenerator.Emit(OpCodes.Stfld, fieldBuilder);
ilGenerator.Emit(OpCodes.Ret);
return setterBuilder;
}
已更新
var typeBuilder = moduleBuilder.DefineType(typeName, TypeAttributes.Public,typeof(PropertyChanged));
var t = typeof(PropertyChanged);
var m2 = t.GetMethod("ValueChanged");
你看,当我创建类型时,我使用父类型,在父类型中有一个名为“ValueChanged”的方法
protected void ValueChanged(object value,[CallerMemberName] string property = null)
我想在 set 方法中调用它。
更新 2
private static MethodBuilder BuildSetter(TypeBuilder typeBuilder, PropertyInfo property, FieldBuilder fieldBuilder, MethodAttributes attributes)
{
var propertyType = property.PropertyType;
var setterBuilder = typeBuilder.DefineMethod($"set_{property.Name}", attributes, null, new Type[] { propertyType });
var setIl = setterBuilder.GetILGenerator();
Label exitSet = setIl.DefineLabel(); // define label to jump in case condition is false
setIl.Emit(OpCodes.Ldarg_0); // this
setIl.Emit(OpCodes.Ldfld, fieldBuilder); // _name field
setIl.Emit(OpCodes.Ldarg_1); // value
var inequality = propertyType.GetMethod("Equals", new[] { propertyType});
setIl.Emit(OpCodes.Callvirt, inequality); // '!=' method
setIl.Emit(OpCodes.Brtrue_S, exitSet); // check for inequality
setIl.Emit(OpCodes.Ldarg_0); // load string literal
setIl.Emit(OpCodes.Ldarg_1); // value
setIl.Emit(OpCodes.Ldstr, property.Name);
var m = typeBuilder.BaseType.GetMethod("ValueChanged", new Type[] {typeof(object),typeof(string) });
setIl.Emit(OpCodes.Call, m);
setIl.Emit(OpCodes.Ldarg_0); // this
setIl.Emit(OpCodes.Ldarg_1); // value
setIl.Emit(OpCodes.Stfld, fieldBuilder); // save the new value into _name
setIl.MarkLabel(exitSet); // mark the label
setIl.Emit(OpCodes.Ret); // return
return setterBuilder;
}
更新 4
也许,我找到了错误原因,请参阅Reflection Emit and Type Inheritance: Calling Base Type Constructors
更新 5
最后,我得到了错误原因。这不是我在更新 4 时猜到的,它是由调用父方法“ValueChanged”引起的。在我们将参数传递给方法之前,如果原始数据类型是 IsValueType,我们应该将其Box 为 Object。请参阅下面的引用,
最佳答案
@AlbertK 的回答向您展示了正确的方法,我将添加完整的代码。希望对您有所帮助。
我把所有东西都放在一个方法中..按照你的意愿重构它。
// define assembly and module
var propertyName = "Name";
var propertyType = typeof(string);
var ab = AssemblyBuilder.DefineDynamicAssembly(
new AssemblyName("dynamicAssembly"),
AssemblyBuilderAccess.Save);
var mb = ab.DefineDynamicModule("dynamicModule", "dynamicModule.dll");
// define type, field and property
var tb = mb.DefineType("dynamicType");
var fb = tb.DefineField("_name", propertyType, FieldAttributes.Private);
var pb = tb.DefineProperty(propertyName, PropertyAttributes.HasDefault, propertyType, Type.EmptyTypes);
var get = tb.DefineMethod("get_" + propertyName, MethodAttributes.Public | MethodAttributes.SpecialName | MethodAttributes.HideBySig, propertyType, Type.EmptyTypes);
var set = tb.DefineMethod("set_" + propertyName, MethodAttributes.Public | MethodAttributes.SpecialName | MethodAttributes.HideBySig, null, new[] { propertyType });
// write the IL for the get method
var getIl = get.GetILGenerator();
getIl.Emit(OpCodes.Ldarg_0); // this
getIl.Emit(OpCodes.Ldfld, fb); // _name field
getIl.Emit(OpCodes.Ret);
// write the IL for the set method
var setIl = set.GetILGenerator();
Label exitSet = setIl.DefineLabel(); // define label to jump in case condition is false
setIl.Emit(OpCodes.Ldarg_0); // this
setIl.Emit(OpCodes.Ldfld, fb); // _name field
setIl.Emit(OpCodes.Ldarg_1); // value
var inequality = propertyType.GetMethod("op_Inequality", new[] { propertyType, propertyType });
setIl.Emit(OpCodes.Call, inequality); // '!=' method
setIl.Emit(OpCodes.Brfalse_S, exitSet); // check for inequality
setIl.Emit(OpCodes.Ldstr, "changedto:"); // load string literal
setIl.Emit(OpCodes.Ldarg_1); // value
var concat = propertyType.GetMethod("Concat", new[] { propertyType, propertyType });
setIl.Emit(OpCodes.Call, concat); // concat two strings (literal + value)
var writeline = typeof(Console).GetMethod("WriteLine", new[] { propertyType });
setIl.Emit(OpCodes.Call, writeline); // write
setIl.Emit(OpCodes.Ldarg_0); // this
setIl.Emit(OpCodes.Ldarg_1); // value
setIl.Emit(OpCodes.Stfld, fb); // save the new value into _name
setIl.MarkLabel(exitSet); // mark the label
setIl.Emit(OpCodes.Ret); // return
pb.SetGetMethod(get);
pb.SetSetMethod(set);
tb.CreateType(); // complete the type
ab.Save("dynamicModule.dll"); // save the assembly to disk
结果
internal class dynamicType
{
private string _name;
public string Name
{
get
{
return this._name;
}
set
{
if (this._name != value)
{
Console.WriteLine("changedto:" + value);
this._name = value;
}
}
}
}
关于C# Emit ,如何编写 if 语句,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54646543/
我只是偶然发现了一个似乎广为人知的compsci关键字“emit”。但是我找不到用通用计算机科学术语对其的任何明确定义,也找不到任何特定编程语言中的“emit()”函数或关键字的特定定义。 我在这里找
在 Qt 中,它们都是有效的,并且表现相同: emit someSignal(value); 对比 emit(someSignal(value)); 有什么区别吗? 最佳答案 Is there any
是否可以同步执行并在调用方法本身时返回结果。所以,我想在 $emit 完成后执行下一条语句。其如下: Parent component has method, doC
我是 socket.io 的新手,遇到了一些看起来很奇怪的事情。我实际上不知道 socket.emit 和 io.emit 之间的区别,但我在任何地方都找不到解释。 io.on('connection
有谁知道如何显式实现接口(interface)的属性 使用反射。发射? 最佳答案 有关 TypeBuilder.DefineMethodOverride 的信息,请参阅 MSDN 文档,其中包括使用
我有多个可观察对象进行网络调用,并且仅当所有可观察对象发出错误时,我才需要从组合器可观察对象发出错误。如果至少有一个可观察完成,则应传递结果。 我当前的流功能如下: Observable.fromIt
我想在这里实现的是当 IF 语句为 true 时 fightID 更改为其他值,因此它不会匹配它之前分配的值,因此我无法在战斗结束后立即通过控制台运行 win-fight 发出的消息。但是我现在得到的
当我们需要自定义组件在 VueTable2 中发出事件时,我们必须使用: this.$parent.$emit('myCustomEvent') // instead of this.$emit('m
This is my production webpack config .标题中的两个引号分别指的是 webpack2 和 webpack。两者都因类似的错误而挂起。 这是我触发构建的命令 设置 N
我正在尝试构建一个发布者,它会在其他 5 个发布者中的任何一个发布者发出 true 时发出 true。我已经设法构建了一个工作版本,但感觉非常恶心,使用 CombineLatest4 + Combin
我正在尝试创建用于教育目的的简单 .net 编译器。在解析、扫描和构建 AST 之后,我正在使用 Reflection.Emit.ILGenerator 生成 .net 程序集。 这是我的程序集生成示
以下 node.js 脚本不工作 var EventEmitter = require('events').EventEmitter; var util = require('util'); var
我使用的是Laravel 10,Livewire 2.x,我试图使用Livewire组件中的一个简单函数中的emit或emitTo事件来更新愿望列表计数。。WishlistCount控制器:。当我运行
我收到以下警告: [Vue warn]: Extraneous non-emits event listeners (addData) were passed to component but cou
我不明白 io.emit 和 io.sockets.emit 之间有什么区别。 有时它们的行为相同,但有时它们的行为不同。 最佳答案 socket.emit 仅将消息发送到发送者客户端 io.emit
我刚刚成功地使用 socket.io 连接到一个服务器脚本,所以我很高兴。我对我的脚本生成的奇怪行为不太满意:我在单击按钮时向服务器脚本发送一个发射,然后服务器测试脚本向控制台日志发回一条消息 6x。
这听起来可能有点模糊,但我仍在学习并试图理解它是如何工作的。 我开始使用“侧菜单”启动 ionic 项目,它已经设置了带有“应用程序”抽象状态/ View / Controller 的 urlrout
我遵循了一些 Nodejs 教程并完成了一个简单的 REST Web 服务。此 Web 服务监听/api/accounts 中的电子邮件和密码,然后继续在 Cassandra 集群中进行查找。我使用
我正在尝试建立一种模式,在这种模式下,我的产生某些对象的可观察对象被转换为领域事件,例如围绕可观察对象产生的 Started、Success、Error,如果这有意义的话 public Observa
这个问题已经有答案了: How can I let the javascript catch a signal from node and prompt a window right after th
我是一名优秀的程序员,十分优秀!