У меня есть такой код:
NameValueCollection nv = HttpUtility.ParseQueryString(queryString);
foreach (KeyValuePair<String,String> pr in nv) {
//process KeyValuePair
}
Это компилируется, но когда я пытаюсь запустить его, я получаю InvalidCastException
.
Почему это? Почему я не могу использовать KeyValuePair
для итерации по a NameValueCollection
, и что мне использовать вместо этого?
var nv = HttpUtility.ParseQueryString(Request.Url.Query); var qsDic = nv.Cast<object>().ToDictionary<object, string, object>(key => (string) key, key => nv[(string) key]);
Ответы:
Во-первых,
NameValueCollection
не используетKeyValuePair<String,String>
. Кроме того,foreach
раскрывает только ключ:NameValueCollection nv = HttpUtility.ParseQueryString(queryString); foreach (string key in nv) { var value = nv[key]; }
источник
KeyValuePair<String,String>
поскольку NameValueCollection использует только строку, строку. Нет необходимости указывать это.Вы не можете сделать это напрямую, но вы можете создать такой метод расширения:
public static IEnumerable<KeyValuePair<string, string>> AsKVP( this NameValueCollection source ) { return source.AllKeys.SelectMany( source.GetValues, (k, v) => new KeyValuePair<string, string>(k, v)); }
Тогда вы сможете:
NameValueCollection nv = HttpUtility.ParseQueryString(queryString); foreach (KeyValuePair<String,String> pr in nv.AsKVP()) { //process KeyValuePair }
Примечание: вдохновлено этим . SelectMany требуется для обработки повторяющихся ключей.
версия 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
источник
Для справки в будущем вы также можете использовать этот синтаксис:
foreach(string key in Request.QueryString) { var value = Request.QueryString[key]; }
источник
HttpRequest
вы можете пропустить его,AllKeys
поскольку этоQueryString
свойствоNameValueCollection
.AllKeys
как упоминал @jgauffinjgauffin
и ответы не будут дублироваться, если вы присмотритесь.Другой метод расширения для учебных целей:
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]); } }
источник
nvc
вместоnvc.AllKeys
. Мне интересно, почему ребята из Microsoft не добавили этоSystem.Linq
. :)NameValueCollection использует перечислитель old-skool:
var enu = ConfigurationManager.AppSettings.GetEnumerator(); while(enu.MoveNext()) { string key = (string)enu.Current; string value = ConfigurationManager.AppSettings[key]; }
источник
Мне это понравилось, и это работает:
foreach (string akey in request.Query.Keys.Cast<string>()) writer.WriteLine(akey + " = " + request.Query[akey]);
источник
Имейте в виду, что имя ключа может встречаться в строке запроса более одного раза и что при сравнении обычно учитывается регистр.
Если вы хотите просто получить значение первого совпадающего ключа и не беспокоиться о регистре, используйте это:
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; }
источник
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
источник
NameValueCollection
данным. Но вопрос в том, почему нельзя конвертироватьNameValueCollection
предметы вKeyValuePair<string, string>
. Может вы не очень внимательно прочитали вопрос. Лучший ответ далjgauffin
. Интересное решение предложилAdrian Carneiro
.