我希望能够在运行时使用反射动态获取overriden属性的值。例如,
class A {
public virtual int Foo => 5;
//This implementation doesn't work
public int ParentFoo => (int)this.GetType().BaseType.GetProperty(nameof(Foo)).GetValue(this);
}
class B : A {
public override int Foo => 7;
}
var test = new B();
Console.WriteLine(test.Foo); //"7"
Console.WriteLine(test.ParentFoo); //Should display "5"
这样做的关键是类型层次结构相当深,我不希望扩展器每次都必须使用完全相同的逻辑(ParentFoo
)重新实现public int ParentFoo => base.Foo;
。我不介意为反思支付性能成本 - 这个属性不需要具备高性能。
是否有可能使用反射来完成我需要的东西?
答案 0 :(得分:1)
可以始终使用反射为属性调用原始定义类的方法。这是一个坏主意。以下代码说明了这个概念,但不值得战斗,也不应该让战斗变得有价值。
void Main()
{
var a = new A();
Console.WriteLine(GetNoVCall<A, int>(a, z => z.Foo)); // prints 5
var b = new B();
Console.WriteLine(GetNoVCall<A, int>(b, z => z.Foo)); // prints 5
Console.WriteLine(GetNoVCall<B, int>(b, z => z.Foo)); // prints 5
}
class A
{
public virtual int Foo { get { return 5; } }
}
class B : A
{
public override int Foo { get { return 7; } }
}
public static TProp GetNoVCall<TClass, TProp>(TClass c, Expression<Func<TClass, TProp>> f)
{
var expr = f.Body as MemberExpression;
var prop = expr.Member as PropertyInfo;
var meth = prop.GetGetMethod(true);
var src = expr.Expression as ParameterExpression;
if (src == null || prop == null || expr == null)
throw new Exception();
var dyn = new DynamicMethod("GetNoVCallHelper", typeof(TProp), new Type[]{ typeof(TClass) }, typeof(string).Module, true);
var il = dyn.GetILGenerator();
il.Emit(OpCodes.Ldarg_0);
il.Emit(OpCodes.Call, meth);
il.Emit(OpCodes.Ret);
return ((Func<TClass, TProp>)dyn.CreateDelegate(typeof(Func<TClass, TProp>)))(c);
}