我有一个实现IDictionary<T, K>
的实例,我在编译时不知道T和K,并希望从中获取所有元素。出于某种原因,我不想使用IEnumerable
,这是IDictionary
实现的唯一非通用接口。
我到目前为止的代码:
// 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,但我认为应该可以调用[]运算符。有人能帮助我吗?
答案 0 :(得分:22)
最好弄清楚 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);
}
}
}
答案 1 :(得分:6)
只是为了完成,即使Marc Gravell的解决方案更好,这就是我的工作方式:
object value = dictType.GetMethod("get_Item")
.Invoke(instance, new object[] { key });
这会调用字典的[]运算符。