質問

次のような文字列を解析したいのですが、 p1=6&p2=7&p3=8NameValueCollection.

にアクセスできない場合にこれを行う最もエレガントな方法は何ですか? Page.Request 物体?

役に立ちましたか?

解決

これには、組み込みの .NET ユーティリティがあります。 HttpUtility.ParseQueryString

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

交換が必要な場合があります querystringnew Uri(fullUrl).Query.

他のヒント

HttpUtility.ParseQueryString は、Web アプリを使用しているか、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);
   }
}

受け入れられた回答は次のものに依存しているため、多くの回答ではカスタム例が提供されています。 システム.ウェブ. 。から 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 への依存関係を削除して、前提条件を「クライアント専用フレームワーク サブセット」に限定しながら、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 クエリを使用するときに、すでに提供されているものよりももう少し汎用性の高い関数が必要でした。

  • 値には複数の等号が含まれる場合があります
  • 名前と値の両方でエンコードされた文字をデコードします
  • クライアントフレームワーク上で実行可能
  • 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 自体に機能を追加することもできます。

なしでこれを行うには 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

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

今気づいたのですが Web APIクライアント があります ParseQueryString 上で動作する拡張メソッド Uri そして、を返します HttpValueCollection:

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

使用するために必要な 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();

Request.QueryString にアクセスするだけです。別の回答として言及されている AllKeys は、キーの配列を取得するだけです。

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

System.Web の依存関係が不要な場合は、HttpUtility クラスからこのソース コードを貼り付けるだけです。

これをソースコードからまとめただけです 単核症. 。これには、HttpUtility とそのすべての依存関係 (IHtmlString、Helpers、HttpEncoder、HttpQSCollection など) が含まれています。

それから 使用 HttpUtility.ParseQueryString.

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

誰もが自分の解決策を貼り付けているようなので..これが私のものです:-)私はクラスライブラリ内からこれが必要でした 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")

すべてのクエリ文字列パラメータの NameValueCollection を取得するには、Request.QueryString.Keys を実行します。

すべてのクエリ文字列値を取得するには、これを試してください。

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

josh-brown を VB で 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;
}

これは私のコードです。非常に役立つと思います。

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;
}
ライセンス: CC-BY-SA帰属
所属していません StackOverflow
scroll top