如何将查询字符串解析为.NET中的NameValueCollection

时间:2008-09-16 01:42:35

标签: .net parsing query-string

我想将p1=6&p2=7&p3=8之类的字符串解析为NameValueCollection

当您无法访问Page.Request对象时,最优雅的方法是什么?

19 个答案:

答案 0 :(得分:345)

有一个内置的.NET实用程序:HttpUtility.ParseQueryString

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

您可能需要将querystring替换为new Uri(fullUrl).Query

答案 1 :(得分:43)

只要您在Web应用程序中,或者不介意包含对System.Web的依赖,HttpUtility.ParseQueryString就可以正常工作。另一种方法是:

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

答案 2 :(得分:25)

许多答案都提供自定义示例,因为已接受的答案依赖于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 :(得分: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;
    }

答案 4 :(得分:7)

我需要的功能比使用OLSC查询时提供的功能更加通用。

  • 值可能包含多个等号
  • 解码名称和值中的编码字符
  • 能够在Client Framework上运行
  • 能够在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()>添加到Uri本身也可能不是一个坏主意。

答案 5 :(得分:7)

要在没有System.Web的情况下执行此操作,无需自己编写,也无需额外的NuGet包:

  1. 添加对System.Net.Http.Formatting
  2. 的引用
  3. 添加using System.Net.Http;
  4. 使用此代码:

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

答案 6 :(得分:2)

如果您想避免使用HttpUtility.ParseQueryString所需的System.Web依赖,您可以使用Uri中的ParseQueryString扩展方法System.Net.Http。< / p>

确保在项目中向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();

答案 7 :(得分: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 ] );    
            }    
        }
    }

答案 8 :(得分:2)

我刚才意识到Web API Client有一个ParseQueryString扩展方法适用于Uri并返回HttpValueCollection

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

答案 9 :(得分:1)

只需访问Request.QueryString。 AllKeys提到的另一个答案就是获取一系列密钥。

答案 10 :(得分:1)

如果您不想要System.Web依赖项,只需从HttpUtility类中粘贴此源代码。

我只是从Mono的源代码中一起鞭打了这个。它包含HttpUtility及其所有依赖项(如IHtmlString,Helpers,HttpEncoder,HttpQSCollection)。

然后使用HttpUtility.ParseQueryString

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

答案 11 :(得分: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")

答案 12 :(得分:1)

HttpUtility.ParseQueryString(Request.Url.Query)返回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); 

答案 13 :(得分:0)

要获取所有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

答案 14 :(得分:0)

        var q = Request.QueryString;
        NameValueCollection qscoll = HttpUtility.ParseQueryString(q.ToString());

答案 15 :(得分:0)

为所有查询字符串参数的NameValueCollection命中Request.QueryString.Keys。

答案 16 :(得分: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;
}

答案 17 :(得分:-1)

我在VB中翻译成josh-brown的C#版本

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

答案 18 :(得分:-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;
}