我有一个实现IDictionary<T, K>
的实例,我在编译时不知道T和K,并且想要从中获取所有元素。由于某些原因,我不想使用IEnumerable
,这将是IDictionary
实现的唯一非通用接口(interface)。
我到目前为止的代码:
// getting types
Type iDictType = instance.GetType().GetInterface("IDictionary`2");
Type keyType = iDictType.GetGenericArguments()[0];
Type valueType = iDictType.GetGenericArguments()[1];
// getting the keys
IEnumerable keys = (IEnumerable)dictType.GetProperty("Keys")
.GetValue(instance, null);
foreach (object key in keys)
{
// ==> this does not work: calling the [] operator
object value = dictType.GetProperty("Item")
.GetValue(instance, new object[] {key } );
// getting the value from another instance with TryGet
MethodInfo tryGetValue = iDictType.GetMethod("TryGetValue");
object[] arguments = new object[] { key, null };
bool hasElement = (bool)tryGetValue.Invoke(otherInstance, arguments);
object anotherValue = arguments[1];
}
我也可以调用TryGetValue,但是我认为应该可以调用[]运算符。有谁能够帮我?
最佳答案
最好找出TKey
/TValue
,然后通过MakeGenericMethod
切换为常规代码-像这样:
(编辑-如果它们的类型相同,您也可以将otherInstance
作为参数传递)
static class Program
{
static void Main()
{
object obj = new Dictionary<int, string> {
{ 123, "abc" }, { 456, "def" } };
foreach (Type iType in obj.GetType().GetInterfaces())
{
if (iType.IsGenericType && iType.GetGenericTypeDefinition()
== typeof(IDictionary<,>))
{
typeof(Program).GetMethod("ShowContents")
.MakeGenericMethod(iType.GetGenericArguments())
.Invoke(null, new object[] { obj });
break;
}
}
}
public static void ShowContents<TKey, TValue>(
IDictionary<TKey, TValue> data)
{
foreach (var pair in data)
{
Console.WriteLine(pair.Key + " = " + pair.Value);
}
}
}
关于c# - 使用反射获取通用IDictionary的值,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/852233/