- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
遵循 this post 上的示例和 its follow-up question ,我正在尝试使用编译表达式创建字段 getter/setter。
getter 工作得很好,但我被 setter 卡住了,因为我需要 setter 来分配任何类型的字段。
这是我的 setter-action 构建器:
public static Action<T1, T2> GetFieldSetter<T1, T2>(this FieldInfo fieldInfo) {
if (typeof(T1) != fieldInfo.DeclaringType && !typeof(T1).IsSubclassOf(fieldInfo.DeclaringType)) {
throw new ArgumentException();
}
ParameterExpression targetExp = Expression.Parameter(typeof(T1), "target");
ParameterExpression valueExp = Expression.Parameter(typeof(T2), "value");
//
// Expression.Property can be used here as well
MemberExpression fieldExp = Expression.Field(targetExp, fieldInfo);
BinaryExpression assignExp = Expression.Assign(fieldExp, valueExp);
//
return Expression.Lambda<Action<T1, T2>> (assignExp, targetExp, valueExp).Compile();
}
现在,我将通用 setter 存储到缓存列表中(当然,因为每次构建 setter 都是性能 killer ),我将它们转换为简单的“对象”:
// initialization of the setters dictionary
Dictionary<string, object> setters = new Dictionary(string, object)();
Dictionary<string, FieldInfo> fldInfos = new Dictionary(string, FieldInfo)();
FieldInfo f = this.GetType().GetField("my_int_field");
setters.Add(f.Name, GetFieldSetter<object, int>(f);
fldInfos.Add(f.Name, f);
//
f = this.GetType().GetField("my_string_field");
setters.Add(f.Name, GetFieldSetter<object, string>(f);
fldInfos.Add(f.Name, f);
现在我尝试像这样设置一个字段值:
void setFieldValue(string fieldName, object value) {
var setterAction = setters[fieldName];
// TODO: now the problem => how do I invoke "setterAction" with
// object and fldInfos[fieldName] as parameters...?
}
我可以简单地调用一个通用方法并每次都进行转换,但我担心性能开销...有什么建议吗?
-- 编辑后的答案基于Mr Anderson's answer ,我创建了一个小测试程序,它比较直接设置值、缓存反射(缓存 FieldInfo 的地方)和缓存多类型代码。我使用最多 3 级继承的对象继承 ( ObjectC : ObjectB : ObjectA
)。
Full code is of the example can be found here.
测试的单次迭代给出以下输出:
-------------------------
--- OBJECT A ---
-------------------------
Set direct: 0.0036 ms
Set reflection: 2.319 ms
Set ref.Emit: 1.8186 ms
Set Accessor: 4.3622 ms
-------------------------
--- OBJECT B ---
-------------------------
Set direct: 0.0004 ms
Set reflection: 0.1179 ms
Set ref.Emit: 1.2197 ms
Set Accessor: 2.8819 ms
-------------------------
--- OBJECT C ---
-------------------------
Set direct: 0.0024 ms
Set reflection: 0.1106 ms
Set ref.Emit: 1.1577 ms
Set Accessor: 2.9451 ms
当然,这只是显示了创建对象的成本 - 这使我们能够衡量创建反射和表达式缓存版本的偏移量。
接下来,让我们运行 1.000.000 次:
-------------------------
--- OBJECT A ---
-------------------------
Set direct: 33.2744 ms
Set reflection: 1259.9551 ms
Set ref.Emit: 531.0168 ms
Set Accessor: 505.5682 ms
-------------------------
--- OBJECT B ---
-------------------------
Set direct: 38.7921 ms
Set reflection: 2584.2972 ms
Set ref.Emit: 971.773 ms
Set Accessor: 901.7656 ms
-------------------------
--- OBJECT C ---
-------------------------
Set direct: 40.3942 ms
Set reflection: 3796.3436 ms
Set ref.Emit: 1510.1819 ms
Set Accessor: 1469.4459 ms
为了完整起见:我删除了对“set”方法的调用以突出显示获取 setter 的成本(反射方法为 FieldInfo
,表达式情况为 Action<object, object>
)。结果如下:
-------------------------
--- OBJECT A ---
-------------------------
Set direct: 3.6849 ms
Set reflection: 44.5447 ms
Set ref.Emit: 47.1925 ms
Set Accessor: 49.2954 ms
-------------------------
--- OBJECT B ---
-------------------------
Set direct: 4.1016 ms
Set reflection: 76.6444 ms
Set ref.Emit: 79.4697 ms
Set Accessor: 83.3695 ms
-------------------------
--- OBJECT C ---
-------------------------
Set direct: 4.2907 ms
Set reflection: 128.5679 ms
Set ref.Emit: 126.6639 ms
Set Accessor: 132.5919 ms
注意:这里的时间增加并不是因为更大的词典访问时间更慢(因为它们有 O(1)
访问时间),而是因为我们访问它的次数增加了(4 ObjectA
的每次迭代次数,ObjectB
的 8 次,ObjectC
的每次迭代次数 12 次)...如您所见,只有创建偏移量在这里有所不同(这是可以预料的)。
底线:我们确实将性能提高了 2 倍或更多,但我们与直接字段集的性能相去甚远……在列表中检索正确的 setter 代表了 10% 的时间。
我会尝试用表达式树代替 Reflection.Emit 看看我们是否可以进一步缩小差距...欢迎任何评论。
最佳答案
如果你希望它支持对多种类型的操作,你的函数缓存应该由 Type
和字段名(string
)索引,并且应该延迟创建函数.试试这个:
private static Dictionary<Type, Dictionary<string, Action<object, object>>> _typeMapper = new Dictionary<Type, Dictionary<string, Action<object, object>>>();
public static void Set(object obj, string fieldName, object newValue)
{
if (obj == null)
{
throw new ArgumentNullException("obj");
}
Type type = obj.GetType();
Dictionary<string, Action<object, object>> fieldMapper;
Action<object, object> action;
if (_typeMapper.TryGetValue(type, out fieldMapper))
{
// entry has been created for this type.
if (!fieldMapper.TryGetValue(fieldName, out action))
{
// method has not been created yet, must build it.
FieldInfo fld = type.GetField(fieldName, BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic);
if (fld == null)
{
throw new ArgumentException("No field " + fieldName);
}
action = buildSetter(fld);
fieldMapper.Add(fieldName, action); // add it to method cache for future use.
}
}
else
{
// -- ADDED CODE: forgot to create the new fieldMapper.....
fieldMapper = new Dictionary<string, Action<object, object>>();
// type has not been added yet, so we know method has not been built yet either.
FieldInfo fld = type.GetField(fieldName, BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic);
if (fld == null)
{
throw new ArgumentException("No field " + fieldName);
}
action = buildSetter(fld);
fieldMapper.Add(fieldName, action); // add it to method cache for future use.
_typeMapper.Add(type, fieldMapper); // add it to type cache for future use.
}
action(obj, newValue); // invoke the method.
}
// this is my preferred setter-builder, feel free to use expressions instead.
private static Action<object, object> buildSetter(FieldInfo fld)
{
DynamicMethod dyn = new DynamicMethod("set_" + fld, typeof(void), new[] { typeof(object), typeof(object) }, fld.DeclaringType);
ILGenerator gen = dyn.GetILGenerator();
gen.Emit(OpCodes.Ldarg_0);
gen.Emit(OpCodes.Castclass, fld.DeclaringType);
gen.Emit(OpCodes.Ldarg_1);
if (fld.FieldType.IsClass)
{
gen.Emit(OpCodes.Castclass, fld.FieldType);
}
else
{
gen.Emit(OpCodes.Unbox_Any, fld.FieldType);
}
gen.Emit(OpCodes.Stfld, fld);
gen.Emit(OpCodes.Ret);
return (Action<object, object>)dyn.CreateDelegate(typeof(Action<object, object>));
}
否则,如果您只需要对一种类型执行此操作,您的过程将变为:
private static Dictionary<string, Action<MyType, object>> _mapper = new Dictionary<string, Action<MyType, object>>();
public static void Set(MyType obj, string fieldName, object newValue)
{
if (obj == null)
{
throw new ArgumentNullException("obj");
}
Action<MyType, object> action;
if (!_mapper.TryGetValue(fieldName, out action))
{
FieldInfo fld = typeof(MyType).GetField(fieldName, BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic);
if (fld == null)
{
throw new ArgumentException("No field " + fieldName);
}
action = buildSetter(fld);
_mapper.Add(fieldName, action);
}
action(obj, newValue); // invoke the method.
}
private static Action<MyType, object> buildSetter(FieldInfo fld)
{
DynamicMethod dyn = new DynamicMethod("set_" + fld, typeof(void), new[] { typeof(MyType), typeof(object) }, typeof(MyType));
ILGenerator gen = dyn.GetILGenerator();
gen.Emit(OpCodes.Ldarg_0);
gen.Emit(OpCodes.Ldarg_1);
if (fld.FieldType.IsClass)
{
gen.Emit(OpCodes.Castclass, fld.FieldType);
}
else
{
gen.Emit(OpCodes.Unbox_Any, fld.FieldType);
}
gen.Emit(OpCodes.Stfld, fld);
gen.Emit(OpCodes.Ret);
return (Action<MyType, object>)dyn.CreateDelegate(typeof(Action<MyType, object>));
}
关于c# - 基类中带有表达式树的字段 getter/setter,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38527266/
如果您有一个具有一些普通 get/set 属性的类,是否有任何理由在类方法中使用 getter,或者您应该只使用私有(private)成员变量?我认为关于setter(验证逻辑?)可能会有更多争论,但
我是 Android 编程的新手(~ 2 个月)有必要为几十个不同的变量设置 getter 吗? 例如—— //Yes I realise that this isn't 'dozens' publi
我是VUEX新手,在学习VUEX的同时搭建了一个测试应用。我已将我的 VUEX 存储拆分为多个模块,每个模块都有自己的 getter.js 文件。 Getters、 Action 、突变被导入到每个模
我有一项服务正在加载我想在多个组件之间共享的基本系统信息。 如果我在 getter 中创建一个 getter 方法 this 如果我不在服务中使用双箭头,则 getter 中的 this 成为组件的
tl;博士 如何从参数化的 getter 访问其他 getter? 通常可以使用this.myGetter ;但是参数化的 getter 被实现为箭头函数,其中 this未定义。 在 Pinia 中处
我第一次尝试做一些 OOP,这不是简单的练习,而是用 java 编写一个程序来存储有关人员的信息(如通讯录)。下面是我正在开发的名为 Person 的类的片段。 Person 将在将来的某个时间被其他
我在某处看到类似下面的内容,想知道它是什么意思。我知道他们是getter和setter,但是想知道为什么字符串Type是这样定义的。谢谢你帮助我。 public string Type { get;
Public class Example { private int number; public Example(int number){ this.number =
getter 应该只返回对象吗: public MyObject getMyObject() { return myObject; } 或者它应该复制它返回的对象并返回该副本? public
我目前正在处理大量数据输入,包括很多值,我希望在 getter 中接收这些值以供以后使用。 在编写了一些方法之后,我想知道仅使用一个 get 方法是否是一个更好的主意,并使用一个包含所有可能值的枚举类
我正在使用新的 Class Public Field Declarations可用 Chrome 72我遇到了这种真正奇怪的行为: class Extended { property = 5; }
我有一个这样的表达式 setter/getter : var expression = () => SomeInstance.Nr; 它被传递到一个方法中: public void AddExpres
我的一个类(class)中有以下 getter: get password(){ if(this._public) return null; var text = ""
我已经设法测试了与其他代码隔离的 Vuex getter。当 getter 依赖于其他 getter 时,我现在面临一些问题,请参见以下示例: getters.js export const gett
有时我的任务是查找 getter 返回的值中的某些嵌套 getter 是否具有某些属性。经典的 C++ 会是这样的: for (const auto& label: labels) for (co
我有一个像这样的基类“Parent”: using System; using System.Collections.Generic; using System.Text; namespace Con
我一直在努力了解 getter 和 setter,但没有深入了解。我读过 JavaScript Getters and Setters和 Defining Getters and Setters只是没
考虑一个简单的 Vue 博客: 我使用 Vuex 作为我的数据存储,我需要设置两个 getters :一个 getPost getter,用于通过 ID 检索 post,以及一个 listFeatur
我有一个 VueX 商店,有两个模块,user.js 和merchant.js,顶层是index.js。 user.js 中的 getter 是: 重构 const getters = { s
我正在尝试向 jsp 添加一个复选框以在进入站点之前接受条款和条件。尽管我有一个 getter 方法并且没有看到任何拼写错误,但我一直收到关于没有 getter 方法的相同错误。我不明白我错过了什么。
我是一名优秀的程序员,十分优秀!