Я хотел бы проанализировать строку, такую как p1=6&p2=7&p3=8, в NameValueCollection.
Каков самый элегантный способ сделать это, когда у вас нет доступа к объекту Page.Request?
Я хотел бы проанализировать строку, такую как p1=6&p2=7&p3=8, в NameValueCollection.
Каков самый элегантный способ сделать это, когда у вас нет доступа к объекту Page.Request?
Там есть встроенная утилита .NET: HttpUtility.ParseQueryString
// C#
NameValueCollection qscoll = HttpUtility.ParseQueryString(querystring);
' VB.NET
Dim qscoll As NameValueCollection = HttpUtility.ParseQueryString(querystring)
Вам может потребоваться заменить querystring на new Uri(fullUrl).Query.
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);
   }
}
Многие ответы предоставляют пользовательские примеры из-за принятой зависимости от ответа на 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 зависимости и не хотите откатывать свои собственные, это хороший вариант.
Я хотел бы удалить зависимость от 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;
    }
Мне нужна была функция, которая немного более универсальна, чем то, что было предоставлено уже при работе с запросами OLSC.
Вот мое решение:
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()> к этому, чтобы добавить возможность самому Uri.
Чтобы сделать это без System.Web, не записывая его самостоятельно и без дополнительных пакетов NuGet:
System.Net.Http.Formattingusing System.Net.Http;Используйте этот код:
new Uri(uri).ParseQueryString()
https://msdn.microsoft.com/en-us/library/system.net.http.uriextensions(v=vs.118).aspx
    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 ] );    
            }    
        }
    }
Я только понял, что Клиент веб-API имеет метод расширения ParseQueryString, который работает на Uri и возвращает HttpValueCollection:
var parameters = uri.ParseQueryString();
string foo = parameters["foo"];
Просто выполните запрос Request.QueryString. AllKeys, упомянутый как еще один ответ, просто дает вам набор ключей.
 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); 
Так как все, кажется, вставляют свое решение.. здесь мои:-)
Мне понадобилось это из библиотеки классов без 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")
Я просто взбивал это вместе из исходного кода Mono. Он содержит HttpUtility и все его зависимости (например, IHtmlString, Helpers, HttpEncoder, HttpQSCollection).
Затем используйте HttpUtility.ParseQueryString.
https://gist.github.com/bjorn-ali-goransson/b04a7c44808bb2de8cca3fc9a3762f9c
Хит вверх Request.QueryString.Keys для NameValueCollection всех параметров строки запроса.
Чтобы получить все значения Querystring, попробуйте следующее:
    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
        var q = Request.QueryString;
        NameValueCollection qscoll = HttpUtility.ParseQueryString(q.ToString());
Если вы хотите избежать зависимости от System.Web, который должен использовать HttpUtility.ParseQueryString, вы можете использовать Uri метод расширения ParseQueryString найден в System.Net.Http.
Обязательно добавьте ссылку (если она еще не указана) в System.Net.Http в вашем проекте.
Обратите внимание, что вам нужно преобразовать тело ответа в действительный Uri, чтобы ParseQueryString (в 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();
Я переводю версию С# 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;
}
Это мой код, я думаю, что это очень полезно:
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;
}