Mam ten kod:
NameValueCollection nv = HttpUtility.ParseQueryString(queryString);
foreach (KeyValuePair<String,String> pr in nv) {
//process KeyValuePair
}
To się kompiluje, ale kiedy próbuję go uruchomić, pojawia się plik InvalidCastException
.
Dlaczego to? Dlaczego nie mogę użyć KeyValuePair
do iteracji po a NameValueCollection
i czego powinienem użyć zamiast tego?
var nv = HttpUtility.ParseQueryString(Request.Url.Query); var qsDic = nv.Cast<object>().ToDictionary<object, string, object>(key => (string) key, key => nv[(string) key]);
Odpowiedzi:
Przede wszystkim
NameValueCollection
nie używaKeyValuePair<String,String>
. Ponadtoforeach
ujawnia tylko klucz:NameValueCollection nv = HttpUtility.ParseQueryString(queryString); foreach (string key in nv) { var value = nv[key]; }
źródło
KeyValuePair<String,String>
ponieważ NameValueCollection używa tylko ciągu, ciągu. Nie trzeba tego określać.Nie możesz tego zrobić bezpośrednio, ale możesz utworzyć metodę rozszerzenia w następujący sposób:
public static IEnumerable<KeyValuePair<string, string>> AsKVP( this NameValueCollection source ) { return source.AllKeys.SelectMany( source.GetValues, (k, v) => new KeyValuePair<string, string>(k, v)); }
Następnie możesz:
NameValueCollection nv = HttpUtility.ParseQueryString(queryString); foreach (KeyValuePair<String,String> pr in nv.AsKVP()) { //process KeyValuePair }
Uwaga: zainspirowany tym . SelectMany jest wymagane do obsługi zduplikowanych kluczy.
Wersja vb.net:
<Extension> Public Function AsKVP( source As Specialized.NameValueCollection ) As IEnumerable(Of KeyValuePair(Of String, String)) Dim result = source.AllKeys.SelectMany( AddressOf source.GetValues, Function(k, v) New KeyValuePair(Of String, String)(k, v)) Return result End Function
źródło
Na przyszłość możesz również użyć następującej składni:
foreach(string key in Request.QueryString) { var value = Request.QueryString[key]; }
źródło
HttpRequest
, możesz pominąć,AllKeys
ponieważQueryString
nieruchomość toNameValueCollection
.AllKeys
jak wspomniano @jgauffinjgauffin
a odpowiedzi nie są duplikatami, jeśli przyjrzysz się bliżej.Inna metoda rozszerzenia do celów edukacyjnych:
public static IEnumerable<KeyValuePair<string, string>> ToIEnumerable(this NameValueCollection nvc) { foreach (string key in nvc.AllKeys) { yield return new KeyValuePair<string, string>(key, nvc[key]); } }
źródło
nvc
zamiastnvc.AllKeys
. Zastanawiam się, dlaczego chłopaki z Microsoftu tego nie dodaliSystem.Linq
. :)NameValueCollection używa oldskulowego modułu wyliczającego:
var enu = ConfigurationManager.AppSettings.GetEnumerator(); while(enu.MoveNext()) { string key = (string)enu.Current; string value = ConfigurationManager.AppSettings[key]; }
źródło
Tak mi się podobało i działa:
foreach (string akey in request.Query.Keys.Cast<string>()) writer.WriteLine(akey + " = " + request.Query[akey]);
źródło
Należy pamiętać, że nazwa klucza może pojawić się więcej niż raz w ciągu zapytania, a porównanie zwykle rozróżnia wielkość liter.
Jeśli chcesz po prostu uzyskać wartość pierwszego pasującego klucza i nie przejmować się wielkością liter, użyj tego:
public string GetQueryValue(string queryKey) { foreach (string key in QueryItems) { if(queryKey.Equals(key, StringComparison.OrdinalIgnoreCase)) return QueryItems.GetValues(key).First(); // There might be multiple keys of the same name, but just return the first match } return null; }
źródło
public static void PrintKeysAndValues2( NameValueCollection myCol ) { Console.WriteLine( " [INDEX] KEY VALUE" ); for ( int i = 0; i < myCol.Count; i++ ) Console.WriteLine( " [{0}] {1,-10} {2}", i, myCol.GetKey(i), myCol.Get(i) ); Console.WriteLine(); }
http://msdn.microsoft.com/en-us/library/system.collections.specialized.namevaluecollection.aspx
źródło
NameValueCollection
danych. Ale pytanie brzmi, dlaczego nie możesz konwertowaćNameValueCollection
elementów naKeyValuePair<string, string>
. Może niezbyt uważnie przeczytałeś pytanie. Najlepszej odpowiedzi udzieliłjgauffin
. Ciekawe rozwiązanie zaproponowałAdrian Carneiro
.