Как разобрать строку запроса в NameValueCollection в .NET - PullRequest
181 голосов
/ 16 сентября 2008

Я бы хотел разобрать строку типа p1=6&p2=7&p3=8 в NameValueCollection.

Какой самый элегантный способ сделать это, когда у вас нет доступа к Page.Request объекту?

Ответы [ 18 ]

345 голосов
/ 16 сентября 2008

Для этого есть встроенная утилита .NET: HttpUtility.ParseQueryString

// C#
NameValueCollection qscoll = HttpUtility.ParseQueryString(querystring);
' VB.NET
Dim qscoll As NameValueCollection = HttpUtility.ParseQueryString(querystring)

Возможно, вам придется заменить querystring на new Uri(fullUrl).Query.

43 голосов
/ 16 сентября 2008

HttpUtility.ParseQueryString будет работать до тех пор, пока вы находитесь в веб-приложении или не против включения зависимости от System.Web. Еще один способ сделать это:

NameValueCollection queryParameters = new NameValueCollection();
string[] querySegments = queryString.Split('&');
foreach(string segment in querySegments)
{
   string[] parts = segment.Split('=');
   if (parts.Length > 0)
   {
      string key = parts[0].Trim(new char[] { '?', ' ' });
      string val = parts[1].Trim();

      queryParameters.Add(key, val);
   }
}
25 голосов
/ 04 марта 2014

Многие ответы содержат пользовательские примеры из-за принятой зависимости ответа от System.Web . Из пакета Microsoft.AspNet.WebApi.Client NuGet существует метод UriExtensions.ParseQueryString , который также можно использовать:

var uri = new Uri("https://stackoverflow.com/a/22167748?p1=6&p2=7&p3=8");
NameValueCollection query = uri.ParseQueryString();

Так что, если вы хотите избежать зависимости System.Web и не хотите создавать свою собственную, это хороший вариант.

19 голосов
/ 24 августа 2009

Я хотел удалить зависимость от System.Web, чтобы можно было проанализировать строку запроса развертывания ClickOnce, при этом предварительные условия ограничены «Подмножеством клиентских рамок».

Мне понравился ответ Р.П. Я добавил дополнительную логику.

public static NameValueCollection ParseQueryString(string s)
    {
        NameValueCollection nvc = new NameValueCollection();

        // remove anything other than query string from url
        if(s.Contains("?"))
        {
            s = s.Substring(s.IndexOf('?') + 1);
        }

        foreach (string vp in Regex.Split(s, "&"))
        {
            string[] singlePair = Regex.Split(vp, "=");
            if (singlePair.Length == 2)
            {
                nvc.Add(singlePair[0], singlePair[1]);
            }
            else
            {
                // only one key with no value specified in query string
                nvc.Add(singlePair[0], string.Empty);
            }
        }

        return nvc;
    }
7 голосов
/ 20 октября 2015

Для этого без System.Web, без написания самостоятельно и без дополнительных пакетов NuGet:

  1. Добавить ссылку на System.Net.Http.Formatting
  2. Добавить using System.Net.Http;
  3. Используйте этот код:

    new Uri(uri).ParseQueryString()
    

https://msdn.microsoft.com/en-us/library/system.net.http.uriextensions(v=vs.118).aspx

7 голосов
/ 15 мая 2013

Мне нужна была функция, которая немного более универсальна, чем та, которая была предоставлена ​​уже при работе с OLSC-запросами.

  • Значения могут содержать несколько знаков равенства
  • Декодировать закодированные символы в имени и значении
  • Возможность работы на клиентской платформе
  • Возможность работы на Mobile Framework.

Вот мое решение:

Public Shared Function ParseQueryString(ByVal uri As Uri) As System.Collections.Specialized.NameValueCollection
    Dim result = New System.Collections.Specialized.NameValueCollection(4)
    Dim query = uri.Query
    If Not String.IsNullOrEmpty(query) Then
        Dim pairs = query.Substring(1).Split("&"c)
        For Each pair In pairs
            Dim parts = pair.Split({"="c}, 2)

            Dim name = System.Uri.UnescapeDataString(parts(0))
            Dim value = If(parts.Length = 1, String.Empty,
                System.Uri.UnescapeDataString(parts(1)))

            result.Add(name, value)
        Next
    End If
    Return result
End Function

Возможно, неплохо было бы добавить к этому <Extension()>, чтобы добавить возможность к самому Ури.

2 голосов
/ 16 сентября 2008
    private void button1_Click( object sender, EventArgs e )
    {
        string s = @"p1=6&p2=7&p3=8";
        NameValueCollection nvc = new NameValueCollection();

        foreach ( string vp in Regex.Split( s, "&" ) )
        {
            string[] singlePair = Regex.Split( vp, "=" );
            if ( singlePair.Length == 2 )
            {
                nvc.Add( singlePair[ 0 ], singlePair[ 1 ] );    
            }    
        }
    }
2 голосов
/ 24 июля 2014

Я только что понял, что Клиент веб-API имеет метод расширения ParseQueryString, который работает на Uri и возвращает HttpValueCollection:

var parameters = uri.ParseQueryString();
string foo = parameters["foo"];
2 голосов
/ 16 августа 2017

Если вы хотите избежать зависимости от System.Web, которая требуется для использования HttpUtility.ParseQueryString , вы можете использовать Uri метод расширения ParseQueryString из System.Net.Http.

Обязательно добавьте ссылку (если вы этого еще не сделали) на System.Net.Http в своем проекте.

Обратите внимание, что вы должны преобразовать тело ответа в действительный Uri, чтобы ParseQueryStringSystem.Net.Http) работал.

string body = "value1=randomvalue1&value2=randomValue2";

// "http://localhost/query?" is added to the string "body" in order to create a valid Uri.
string urlBody = "http://localhost/query?" + body;
NameValueCollection coll = new Uri(urlBody).ParseQueryString();
1 голос
/ 16 сентября 2008

Просто получите доступ к Request.QueryString. AllKeys, упомянутый в качестве другого ответа, просто дает вам массив ключей.

...