【发布时间】:2013-05-23 21:44:15
【问题描述】:
我正在使用第三方 API,我需要访问内部字段 ("_fieldOfA") 的私有字段 ("_fieldOfB")。下面的示例说明了该组合。我要查找的字段是内部类型列表(ClassC 和 ClassD 是内部的并且在同一个程序集中)。
public abstract class ClassA
{
internal ClassB _fieldOfA;
}
public class ClassB
{
private readonly List<ClassC<ClassD, int>> _fieldOfB;
}
我尝试过反射,但我似乎无法正确转换最终类型 - 动态类型保持 Object(请注意,我的类扩展了 ClassA)
var assemblyHandle = typeof (ClassB).Assembly;
var genericTypeC = assemblyHandle.GetType("ApiNamespace.ClassC`2");
var typeD = assemblyHandle.GetType("ApiNamespace.ClassD");
var genericTypesForC = new Type[] { typeD, typeof(int) };
var typeC = genericTypeC.MakeGenericType(genericTypesForC);
var typeOfList = typeof(List<>).MakeGenericType(typeC);
var fieldOfAInfo= typeof(ClassA).GetField("_fieldOfA", BindingFlags.NonPublic | BindingFlags.GetField | BindingFlags.Instance);
var fieldOfAValue = fieldOfAInfo.GetValue(this);
var fieldOfBInfo= typeof(ClassB).GetField("_fieldOfB", BindingFlags.NonPublic | BindingFlags.GetField | BindingFlags.Instance);
var uncastedFieldOfBValue = fieldOfBInfo.GetValue(fieldOfAValue);
dynamic fieldOfBValue = Convert.ChangeType(uncastedFieldOfBValue, typeOfList);
有什么想法吗?
【问题讨论】:
-
预警,如果你还没有弄清楚这一点......因为
ClassC<ClassD, int>的项目是你看不到的内部类型,你将不得不使用反射使用列表和它的对象(不仅仅是获取列表本身),因为你不能在编译时使用这些类型。 -
话虽如此,您是否在调试器中检查了
uncastedFieldOFBValue,以确定您实际上从反射调用中获得了某种形式的List<>? -
是的,我已经使用调试器检查了
uncastedFieldOFBValue,并且数据在那里,但我似乎无法以List<>的身份访问它,它仍然是Object -
我创建了一个解决方法,方法是转换为
IList,然后反射性地从列表中一一获取项目值。这并不漂亮,而且绝对是一个令人讨厌的 hack,但在 API 供应商可以发布他们的库的更新之前,它必须这样做。
标签: c# reflection casting