วิธีการแยกวิเคราะห์สตริงแบบสอบถามลงใน NameValueCollection ใน. NET


186

ฉันต้องการที่จะแยกสตริงเช่นเป็นp1=6&p2=7&p3=8NameValueCollection

อะไรคือวิธีที่สง่างามที่สุดในการทำสิ่งนี้เมื่อคุณไม่มีสิทธิ์เข้าถึงPage.Requestวัตถุ?

คำตอบ:


356

มียูทิลิตี้. NET ในตัวสำหรับสิ่งนี้: HttpUtility.ParseQueryString

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

คุณอาจจะต้องเปลี่ยนด้วยquerystringnew Uri(fullUrl).Query


26
โอมาร์มันใช้งานไม่ได้สำหรับฉันบน ASP.NET 4 มันส่งคืนคีย์ของ " stackoverflow.com?para " แทนที่จะเป็น "para" ดังนั้นฉันใช้ HttpUtility.ParseQueryString (ใหม่ Uri (fullUrl) .Query) ซึ่งทำงานได้อย่างถูกต้องสำหรับฉัน
Michael

2
qscoll ["p1"], qscoll ["p2"] และ qscoll ["p3"]
SMUsamaShah

5
ParseQueryString เป็นความคิดที่ไม่ดีที่จะใช้ในแอปพลิเคชันเดสก์ท็อปเนื่องจากไม่มีอยู่ในโปรไฟล์ลูกค้า เหตุใดจึงต้องติดตั้งไลบรารีเพิ่มเติม 100 M บนคอมพิวเตอร์ไคลเอนต์เพื่อใช้วิธีการง่ายๆเพียงวิธีเดียว อย่างไรก็ตามดูเหมือนว่า Microsoft ไม่มีความคิดที่ดีกว่านี้ วิธีเดียวคือการใช้วิธีการของตัวเองหรือใช้การใช้งานโอเพนซอร์ส
Vitalii

2
VASoftOnline: ในกรณีนั้นคุณสามารถใช้ Mono ในการใช้งาน ParseQueryString: github.com/mono/mono/blob/master/mcs/class/System.Web/?hl=thใบอนุญาตสำหรับ MIT X11: github.com/mono/mono / blob / master / LICENSE
sw

3
HttpUtility.ParseQueryStringจะเป็นคำแนะนำของฉันยกเว้นว่าในกรณีของHttpUtility.ParseQueryString("&X=1&X=2&X=3")ผลลัพธ์คือ....X=1,2,3...การมีหลาย params ในชื่อเดียวกันนั้นเป็นเรื่องแปลก แต่จำเป็นต้องสนับสนุนพารามิเตอร์ของตัวควบคุมเช่น int [], IE จำนวนนับ <int> ฯลฯ (เช่น params อาจใช้เพื่อรองรับหลายช่องทำเครื่องหมาย) โปรดดูที่ "การเกิดขึ้นของหลายตัวแปรสตริงแบบสอบถามเดียวกันจะถูกรวมอยู่ในหนึ่งรายการ" ตามเว็บไซต์ 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]
Alexandru Pupsa

2
เกิดอะไรขึ้นถ้าไม่มีค่า? ?foo=1&barเช่นสตริงแบบสอบถามจะมีลักษณะเหมือน HttpUtilityจะแยกเป็น{ key = null, value = "bar" }
โทมัส Levesque

นี้ดีมากเมื่อเทียบกับ HttpUtility.ParseQueryString เพราะมันไม่ได้ถอดรหัสค่า (ดังนั้นค่าที่เข้ารหัส base64 จะถูกเก็บไว้)
CRice

1
ที่จริงแล้วคุณยังต้องอนุญาตให้ใช้ค่า '=' เช่น & code = A0SYw34Hj / m ++ lH0s7r0l / yg6GWdymzSCbI2zOn3V4o = จะลบอักขระ '=' ตัวสุดท้าย ฉันเขียนส่วนหนึ่งของรหัสของคุณใหม่เพื่อรับดัชนีแรกของ '=' และซับสตริงคีย์และค่าเพื่อแก้ไขปัญหานี้ (แทนที่จะใช้การแยก)
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.Formatting พึ่งพาฉันเชื่อว่ามันมีให้เฉพาะโดยการเพิ่ม Microsoft.AspNet.WebApi.Client NuGet แพ็คเกจ
James Skimming

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 คุณเพียงแค่แทนที่ "NameValueCollection" ด้วย "SortedDictionnary <string, string>"
Mike Bryant

4
ระวังเมื่อเปลี่ยน NameValueCollection สำหรับพจนานุกรม - มันไม่เหมือนกัน! สตริงแบบสอบถามสนับสนุนหลายคีย์ที่มีค่าเท่ากันและ NameValueCollection
Matt DeKrey

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 เอง


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 แพ็คเกจ
James Skimming

อืมฉันไม่ดี ฉันเดาว่ามันมาพร้อมกับเฟรมเวิร์ก MVC ที่ติดตั้งโดยอัตโนมัติดังนั้นฉันไม่ต้องเพิ่มแพ็คเกจใด ๆ (Program Files (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(ใน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

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, ดีกว่าที่จะไม่บูรณาการล้อ
แมทธิวล็อค



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

ฉันเพียงแค่วิปปิ้งด้วยกันนี้จากซอร์สโค้ดของโมโน มันมี HttpUtility และการพึ่งพาทั้งหมด (เช่น IHtmlString, ผู้ช่วยเหลือ, 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
เพียงหนึ่งปัญหากับวิธีนี้: สตริงแบบสอบถามสามารถมีได้มากกว่าหนึ่งค่าสำหรับพารามิเตอร์ที่กำหนด ในกรณีนี้วิธีการของคุณจะผิดพลาดซ้ำคีย์
Thomas Levesque

ใช่อย่างน้อยใช้ NameValueCollection การสอบถามที่มีคีย์ซ้ำกันนั้นถูกกฎหมายอย่างสมบูรณ์
Chad Grant

นอกจากนี้พจนานุกรมของคุณจะต้องตรงตามตัวพิมพ์ใหญ่ - เล็กดังนั้น X = 1 และ x = 1 จะเป็นปุ่มที่แตกต่างกัน ต้องการพจนานุกรมใหม่ <สตริง, สตริง> (StringComparer.OrdinalIgnoreCase);
ชาดแกรนท์


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


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

ฉันแปล Josh-brown เวอร์ชัน C # ใน 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;
}
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.