.NET에서 쿼리 문자열을 NameValueCollection으로 구문 분석하는 방법

StackOverflow https://stackoverflow.com/questions/68624

  •  09-06-2019
  •  | 
  •  

문제

다음과 같은 문자열을 구문 분석하고 싶습니다. 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);
   }
}

허용되는 답변의 의존성 때문에 많은 답변이 사용자 정의 예제를 제공하고 있습니다. 시스템.웹.로부터 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();

그러니 피하고 싶다면 시스템.웹 의존성이 있고 자신의 것을 굴리고 싶지 않다면 이것은 좋은 선택입니다.

전제 조건을 "클라이언트 전용 프레임워크 하위 집합"으로 제한하면서 ClickOnce 배포의 쿼리 문자열을 구문 분석할 수 있도록 System.Web에 대한 종속성을 제거하고 싶었습니다.

나는 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 패키지 없이:

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

방금 그걸 깨달았어요 웹 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를 누르십시오.

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

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

이것은 내 코드입니다. 매우 유용하다고 생각합니다.

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