Eu gostaria de analisar uma string como p1=6&p2=7&p3=8
em a NameValueCollection
.
Qual é a maneira mais elegante de fazer isso quando você não tem acesso ao Page.Request
objeto?
Eu gostaria de analisar uma string como p1=6&p2=7&p3=8
em a NameValueCollection
.
Qual é a maneira mais elegante de fazer isso quando você não tem acesso ao Page.Request
objeto?
Respostas:
Há um utilitário .NET interno para isso: HttpUtility.ParseQueryString
// C#
NameValueCollection qscoll = HttpUtility.ParseQueryString(querystring);
' VB.NET
Dim qscoll As NameValueCollection = HttpUtility.ParseQueryString(querystring)
Pode ser necessário substituir querystring
por new Uri(fullUrl).Query
.
HttpUtility.ParseQueryString
seria minha recomendação, exceto que, no caso do HttpUtility.ParseQueryString("&X=1&X=2&X=3")
resultado, ....X=1,2,3...
Ter vários parâmetros do mesmo nome é incomum, mas necessário para suportar parâmetros do controlador como int [], IEnumerable <int> etc. (esses parâmetros podem ser usados para suportar várias caixas de seleção) consulte "Várias ocorrências da mesma variável de cadeia de consulta são consolidadas em uma entrada", conforme o site da MS . Uma versão artesanal do método pode ser sua única opção
O HttpUtility.ParseQueryString funcionará desde que você esteja em um aplicativo Web ou não se importe de incluir uma dependência no System.Web. Outra maneira de fazer isso é:
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);
}
}
?foo=1&bar
. HttpUtility
analisaria como{ key = null, value = "bar" }
Muitas das respostas estão fornecendo exemplos personalizados devido à dependência da resposta aceita no System.Web . Do Microsoft.AspNet.WebApi.Client pacote NuGet há uma UriExtensions.ParseQueryString , método que também pode ser usado:
var uri = new Uri("https://stackoverflow.com/a/22167748?p1=6&p2=7&p3=8");
NameValueCollection query = uri.ParseQueryString();
Portanto, se você deseja evitar a dependência System.Web e não deseja criar sua própria, essa é uma boa opção.
Eu queria remover a dependência do System.Web para poder analisar a sequência de consultas de uma implantação do ClickOnce, mantendo os pré-requisitos limitados ao "Subconjunto da estrutura do cliente apenas".
Eu gostei da resposta do rp. Eu adicionei alguma lógica adicional.
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;
}
Eu precisava de uma função que seja um pouco mais versátil do que a que já foi fornecida ao trabalhar com consultas OLSC.
Aqui está a minha solução:
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
Pode não ser uma má idéia abordar <Extension()>
isso também para adicionar a capacidade ao próprio Uri.
Para fazer isso sem System.Web
, sem você mesmo escrever e sem pacotes NuGet adicionais:
System.Net.Http.Formatting
using System.Net.Http;
Use este código:
new Uri(uri).ParseQueryString()
https://msdn.microsoft.com/en-us/library/system.net.http.uriextensions(v=vs.118).aspx
Se você quiser evitar a dependência de System.Web que é necessário para usar HttpUtility.ParseQueryString , você poderia usar o Uri
método de extensão ParseQueryString
encontrada em System.Net.Http
.
Certifique-se de adicionar uma referência (se ainda não o fez) ao System.Net.Http
seu projeto.
Observe que você precisa converter o corpo da resposta em um válido Uri
para que ParseQueryString
(in System.Net.Http
) funcione.
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();
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 ] );
}
}
}
Acabei de perceber que o Web API Client tem um ParseQueryString
método de extensão que funciona em um Uri
e retorna um HttpValueCollection
:
var parameters = uri.ParseQueryString();
string foo = parameters["foo"];
Basta acessar Request.QueryString. AllKeys mencionadas como outra resposta apenas fornece uma variedade de chaves.
HttpUtility.ParseQueryString(Request.Url.Query)
return é HttpValueCollection
(classe interna). Herda de 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);
Acabei de juntar isso a partir do código fonte do Mono . Ele contém o HttpUtility e todas as suas dependências (como IHtmlString, Helpers, HttpEncoder, HttpQSCollection).
Então use HttpUtility.ParseQueryString
.
https://gist.github.com/bjorn-ali-goransson/b04a7c44808bb2de8cca3fc9a3762f9c
Como todo mundo parece estar colando sua solução ... aqui está a minha :-) Eu precisava disso dentro de uma biblioteca de classes sem System.Web
buscar parâmetros de identificação dos hiperlinks armazenados.
Pensei em compartilhar porque acho essa solução mais rápida e com melhor aparência.
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];
}
}
Uso:
Statics.QueryGet(url, "id")
Ative Request.QueryString.Keys para um NameValueCollection de todos os parâmetros de seqüência de caracteres de consulta.
Para obter todos os valores de consulta, tente o seguinte:
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());
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;
}
Eu traduzo para a versão c # de josh-brown no 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;
}
Este é o meu código, acho muito útil:
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;
}