c# – 使用FieldInfo.SetValue与LINQ表达式在结构中设置字段
内容导读
互联网集市收集整理的这篇技术教程文章主要介绍了c# – 使用FieldInfo.SetValue与LINQ表达式在结构中设置字段,小编现在分享给大家,供广大互联网技能从业者学习和参考。文章包含2619字,纯文字阅读大概需要4分钟。
内容图文
我想使用LINQ表达式设置私有字段.我有这个代码:
//parameter "target", the object on which to set the field `field`
ParameterExpression targetExp = Expression.Parameter(typeof(object), "target");
//parameter "value" the value to be set in the `field` on "target"
ParameterExpression valueExp = Expression.Parameter(typeof(object), "value");
//cast the target from object to its correct type
Expression castTartgetExp = Expression.Convert(targetExp, type);
//cast the value to its correct type
Expression castValueExp = Expression.Convert(valueExp, field.FieldType);
//the field `field` on "target"
MemberExpression fieldExp = Expression.Field(castTartgetExp, field);
//assign the "value" to the `field`
BinaryExpression assignExp = Expression.Assign(fieldExp, castValueExp);
//compile the whole thing
var setter = Expression.Lambda<Action<object, object>> (assignExp, targetExp, valueExp).Compile();
这将编译一个带有两个对象的委托,即目标和值:
setter(someObject, someValue);
type变量指定目标的Type,而field变量是FieldInfo,它指定要设置的字段.
这适用于引用类型,但如果目标是结构,那么这个东西会将目标作为副本传递给setter委托并在副本上设置值,而不是像我想要的那样在原始目标上设置值. (至少这是我的想法.)
另一方面,
field.SetValue(someObject, someValue);
工作得很好,即使是结构.
为了使用编译的表达式设置目标字段,我能做些什么吗?
解决方法:
对于值类型,请使用Expression.Unbox而不是Expression.Convert.
//cast the target from object to its correct type
Expression castTartgetExp = type.IsValueType
? Expression.Unbox(targetExp, type)
: Expression.Convert(targetExp, type);
这是一个演示:.NET Fiddle
问:setter方法没有ref参数.它如何更新原始结构?
答:虽然没有ref关键字,值类型通常按值传递并因此被复制,但这里目标参数的类型是对象.如果参数是盒装结构,则对该方框的引用(按值)传递给该方法.
现在,使用纯C#来改变盒装结构是不可能的,因为C#取消装箱转换总是产生盒装值的副本.但是有可能使用IL或Reflection:
public struct S { public int I; }
public void M(object o, int i)
{
// ((S)o).I = i; // DOESN'T COMPILE
typeof(S).GetField("I").SetValue(o, i);
}
public void N()
{
S s = new S();
object o = s; // create a boxed copy of s
M(o, 1); // mutate o (but not s)
Console.WriteLine(((S)o).I); // "1"
Console.WriteLine(s.I); // "0"
M(s, 2); // mutate a TEMPORARY boxed copy of s (BEWARE!)
Console.WriteLine(s.I); // "0"
}
问:如果LINQ表达式使用Expression.Convert,为什么setter不起作用?
答:Expression.Convert编译为unbox.any IL指令,该指令返回target引用的结构的副本.然后,setter更新此副本(随后将其丢弃).
问:为什么Expression.Unbox会解决问题?
答:Expression.Unbox(当用作Expression.Assign的目标时)编译为unbox IL指令,该指令返回指向目标引用的结构的指针.然后,setter使用指针直接修改该结构.
内容总结
以上是互联网集市为您收集整理的c# – 使用FieldInfo.SetValue与LINQ表达式在结构中设置字段全部内容,希望文章能够帮你解决c# – 使用FieldInfo.SetValue与LINQ表达式在结构中设置字段所遇到的程序开发问题。 如果觉得互联网集市技术教程内容还不错,欢迎将互联网集市网站推荐给程序员好友。
内容备注
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 gblab@vip.qq.com 举报,一经查实,本站将立刻删除。
内容手机端
扫描二维码推送至手机访问。