Jak widać w źródłach referencyjnych, NameValueCollection dziedziczy po NameObjectCollectionBase .
Więc bierzesz typ podstawowy, uzyskujesz prywatną tablicę haszującą przez odbicie i sprawdzasz, czy zawiera określony klucz.
Aby to działało również w Mono, musisz zobaczyć, jaka nazwa tablicy hashy jest w mono, co można zobaczyć tutaj (m_ItemsContainer), i uzyskać mono-pole, jeśli początkowe FieldInfo ma wartość null (mono- runtime).
Lubię to
public static class ParameterExtensions
{
private static System.Reflection.FieldInfo InitFieldInfo()
{
System.Type t = typeof(System.Collections.Specialized.NameObjectCollectionBase);
System.Reflection.FieldInfo fi = t.GetField("_entriesTable", System.Reflection.BindingFlags.Instance | System.Reflection.BindingFlags.NonPublic);
if(fi == null) // Mono
fi = t.GetField("m_ItemsContainer", System.Reflection.BindingFlags.Instance | System.Reflection.BindingFlags.NonPublic);
return fi;
}
private static System.Reflection.FieldInfo m_fi = InitFieldInfo();
public static bool Contains(this System.Collections.Specialized.NameValueCollection nvc, string key)
{
//System.Collections.Specialized.NameValueCollection nvc = new System.Collections.Specialized.NameValueCollection();
//nvc.Add("hello", "world");
//nvc.Add("test", "case");
// The Hashtable is case-INsensitive
System.Collections.Hashtable ent = (System.Collections.Hashtable)m_fi.GetValue(nvc);
return ent.ContainsKey(key);
}
}
w przypadku ultra-czystego, nieodblaskowego kodu .NET 2.0, możesz zapętlić klucze, zamiast używać tablicy mieszania, ale jest to powolne.
private static bool ContainsKey(System.Collections.Specialized.NameValueCollection nvc, string key)
{
foreach (string str in nvc.AllKeys)
{
if (System.StringComparer.InvariantCultureIgnoreCase.Equals(str, key))
return true;
}
return false;
}