Як проаналізувати рядок запиту в NameValueCollection в .NET


186

Я хотів би проаналізувати рядок типу p1=6&p2=7&p3=8a NameValueCollection.

Який найелегантніший спосіб зробити це, коли у вас немає доступу до Page.Requestоб'єкта?

Відповіді:


356

Для цього є вбудована утиліта .NET: HttpUtility.ParseQueryString

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

Можливо, вам доведеться замінити querystringна new Uri(fullUrl).Query.


26
Омар, для мене це не працювало на ASP.NET 4, він повернув ключ " stackoverflow.com?para " замість "para". Тому я використовую HttpUtility.ParseQueryString (новий Uri (fullUrl) .Query), який правильно працює для мене.
Майкл

2
qscoll ["p1"], qscoll ["p2"] та qscoll ["p3"]
SMUsamaShah

5
ParseQueryString насправді погана ідея для використання у настільних додатках, оскільки вона не включена у профіль клієнта; чому для встановлення 100 M додаткових бібліотек на клієнтському комп'ютері просто використовувати один простий метод? Однак, схоже, Microsoft не має кращої ідеї. Єдиний спосіб - реалізувати власний метод або використовувати реалізацію з відкритим кодом.
Віталій

2
VASoftOnline: У цьому випадку ви можете використовувати Mono реалізацію ParseQueryString: github.com/mono/mono/blob/master/mcs/class/System.Web/… ліцензією на це є MIT X11: github.com/mono/mono / крап / майстер / ЛІЦЕНЗІЯ
sw.

3
HttpUtility.ParseQueryStringбуде моєю рекомендацією, за винятком того, що у випадку HttpUtility.ParseQueryString("&X=1&X=2&X=3")результату є наявність ....X=1,2,3...декількох однотипних парам - це нечасто, але це потрібно для підтримки параметрів контролера, таких як int [], IEnumerable <int> і т.д. (такі параметри можуть використовуватися для підтримки декількох прапорців) див. "Кілька входів однієї змінної рядка запиту консолідовано в одному записі", як на сайті MS . Версія методу ручної роботи може бути вашим єдиним варіантом
dunxz

43

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);
   }
}

5
ви повинні перевірити, чи є деталі. Довжина> 1, щоб бути впевненим, що ви можете зателефонувати до деталей [1]
Александру Пупса

2
Що робити, якщо немає цінності? наприклад, рядок запиту може виглядати так ?foo=1&bar. HttpUtilityрозберемо це як{ key = null, value = "bar" }
Томас Левеск

Це чудово порівняно з HttpUtility.ParseQueryString, оскільки він не розшифровує значення (тому збережені закодовані значення base64)
CRice

1
Насправді вам все-таки потрібно дозволити значення "=", наприклад & code = A0SYw34Hj / m ++ lH0s7r0l / yg6GWdymzSCbI2zOn3V4o = буде видалено останній символ "=". Я переписав частину вашого коду, щоб отримати перший індекс '=' і підстрочити ключ і значення, щоб виправити це (замість того, щоб використовувати split).
CRice

28

Багато відповідей надають власні приклади через залежність прийнятої відповіді від 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 і не хочете розгорнути свою власну, це хороший варіант.


3
Ця ж функція існує як розширення в просторі імен System.Net.Http (див. Мою відповідь нижче), немає потреби в цілій іншій залежності ...
jvenema

@jvenema Звідки ви додаєте залежність System.Net.Http.Формування, я вважаю, що це забезпечується лише додаванням пакета Microsoft.AspNet.WebApi.Client NuGet.
Джеймс Скімінг

19

Я хотів зняти залежність від System.Web, щоб я міг проаналізувати рядок запитів розгортання ClickOnce, маючи при цьому передумови, обмежені "Підмножиною рамки лише для клієнта".

Мені сподобалась відповідь rp. Я додав трохи додаткової логіки.

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;
    }

це дуже корисно для Windows Phone, ви просто повинні замінити "NameValueCollection" на "SortedDictionnary <string, string>"
Майк Брайант

4
Будьте обережні, перемикаючи NameValueCollection для словника - вони не однакові! Рядки запитів підтримують декілька клавіш з однаковим значенням, так само, як і NameValueCollection.
Метт Декре

7

Мені потрібна була функція, яка трохи більш універсальна, ніж те, що було надано вже під час роботи з запитами OLSC.

  • Значення можуть містити кілька рівних знаків
  • Розшифруйте закодовані символи як у імені, так і у значенні
  • Можливість роботи на Client 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()>за це теж, щоб додати можливості самому Урі.


7

Щоб зробити це без того 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


3
Звідки ви додаєте залежність System.Net.Http.Formatting, я вважаю, що це забезпечується лише додаванням пакета Microsoft.AspNet.WebApi.Client NuGet.
Джеймс Скімінг

Ага, мій поганий. Я здогадуюсь, що в рамках MVC встановлено автоматичне встановлення, тому мені не потрібно було додавати будь-які додаткові пакети (програмні файли (x86) \ Microsoft ASP.NET \ ASP.NET MVC 4 \ Assemblies \ System.Net). Http.Formatting.dll)
jvenema

System.Net.Formatting EXTENSION VS2019 - це окремий список або вкладка з традиційних посилань на рамки
Sql Surfer

3

Якщо ви хочете уникнути залежності від System.Web, необхідної для використання HttpUtility.ParseQueryString , ви можете використовувати Uriметод розширення, ParseQueryStringзнайдений у System.Net.Http.

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

Зауважте, що ви повинні перетворити тіло відповідей у ​​дійсне, Uriщоб ParseQueryString(in System.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();

System.Net.Formatting EXTENSION VS2019 ставить посилання на розширення окремо від традиційних посилань на рамки.
Sql Surfer

2
    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 ] );    
            }    
        }
    }

5
крапка з комою дозволена також як роздільник параметрів у http, краще не винаходити колесо
Matthew Lock

2

Я щойно зрозумів, що клієнт Web API має ParseQueryStringметод розширення, який працює на a Uriі повертає HttpValueCollection:

var parameters = uri.ParseQueryString();
string foo = parameters["foo"];

1

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


1

HttpUtility.ParseQueryString(Request.Url.Query)return є HttpValueCollection(внутрішній клас). Він успадковує від NameValueCollection.

    var qs = HttpUtility.ParseQueryString(Request.Url.Query);
    qs.Remove("foo"); 

    string url = "~/Default.aspx"; 
    if (qs.Count > 0)
       url = url + "?" + qs.ToString();

    Response.Redirect(url); 

1

Якщо ви не хочете залежності від System.Web, просто вставте цей вихідний код із класу HttpUtility.

Я щойно виграв це разом із вихідного коду Mono . Він містить HttpUtility і всі його залежності (як IHtmlString, Helpers, HttpEncoder, HttpQSCollection).

Потім використовуйте HttpUtility.ParseQueryString.

https://gist.github.com/bjorn-ali-goransson/b04a7c44808bb2de8cca3fc9a3762f9c


1

Оскільки всі, здається, вкладають його рішення .. ось моє :-) Мені це було потрібно з бібліотеки класів без System.Web отримання параметрів id із збережених гіперпосилань.

Думав, я поділюсь, тому що я знаходжу це рішення швидше і краще виглядаю.

public static class Statics
    public static Dictionary<string, string> QueryParse(string url)
    {
        Dictionary<string, string> qDict = new Dictionary<string, string>();
        foreach (string qPair in url.Substring(url.IndexOf('?') + 1).Split('&'))
        {
            string[] qVal = qPair.Split('=');
            qDict.Add(qVal[0], Uri.UnescapeDataString(qVal[1]));
        }
        return qDict;
    }

    public static string QueryGet(string url, string param)
    {
        var qDict = QueryParse(url);
        return qDict[param];
    }
}

Використання:

Statics.QueryGet(url, "id")

1
Лише одна проблема з цим методом: рядок запиту може мати більше одного значення для заданого параметра. У цьому випадку ваш метод видасть помилку повторюваного ключа.
Томас Левеск

так, принаймні використовуйте NameValueCollection, запити з повторюваними ключами цілком законно.
Чад Грант

Також у вашому словнику є великі регістри, тому X = 1 і x = 1 будуть різними ключами. Потрібен новий словник <string, string> (StringComparer.OrdinalIgnoreCase);
Чад Грант


0

Щоб отримати всі значення запитів, спробуйте:

    Dim qscoll As NameValueCollection = HttpUtility.ParseQueryString(querystring)

Dim sb As New StringBuilder("<br />")
For Each s As String In qscoll.AllKeys

  Response.Write(s & " - " & qscoll(s) & "<br />")

Next s


0
let search = window.location.search;

console.log(search);

let qString = search.substring(1);

while(qString.indexOf("+") !== -1)

   qString  = qString.replace("+", "");

let qArray = qString.split("&");

let values = [];

for(let i = 0; i < qArray.length; i++){
   let pos = qArray[i].search("=");
   let keyVal = qArray[i].substring(0, pos);
   let dataVal = qArray[i].substring(pos + 1);
   dataVal = decodeURIComponent(dataVal);
   values[keyVal] = dataVal;
}

-1

Я перекладаю на версію C # josh-brown у VB

private System.Collections.Specialized.NameValueCollection ParseQueryString(Uri uri)
{
    var result = new System.Collections.Specialized.NameValueCollection(4);
    var query = uri.Query;
    if (!String.IsNullOrEmpty(query))
    {
        var pairs = query.Substring(1).Split("&".ToCharArray());
        foreach (var pair in pairs)
        {
            var parts = pair.Split("=".ToCharArray(), 2);
            var name = System.Uri.UnescapeDataString(parts[0]);
            var value = (parts.Length == 1) ? String.Empty : System.Uri.UnescapeDataString(parts[1]);
            result.Add(name, value);
        }
    }
    return result;
}

-2

Це мій код, я думаю, що це дуже корисно:

public String GetQueryString(string ItemToRemoveOrInsert = null, string InsertValue = null )
{
    System.Collections.Specialized.NameValueCollection filtered = new System.Collections.Specialized.NameValueCollection(Request.QueryString);
    if (ItemToRemoveOrInsert != null)
    {
        filtered.Remove(ItemToRemoveOrInsert);
        if (!string.IsNullOrWhiteSpace(InsertValue))
        {
            filtered.Add(ItemToRemoveOrInsert, InsertValue);
        }
    }

    string StrQr = string.Join("&", filtered.AllKeys.Select(key => key + "=" + filtered[key]).ToArray());
    if (!string.IsNullOrWhiteSpace(StrQr)){
        StrQr="?" + StrQr;
    }

    return StrQr;
}
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.