ฉันมีสตริงเช่นนี้:
abc=foo&def=%5Basf%5D&xyz=5
ฉันจะแปลงเป็นวัตถุ JavaScript เช่นนี้ได้อย่างไร
{
abc: 'foo',
def: '[asf]',
xyz: 5
}
ฉันมีสตริงเช่นนี้:
abc=foo&def=%5Basf%5D&xyz=5
ฉันจะแปลงเป็นวัตถุ JavaScript เช่นนี้ได้อย่างไร
{
abc: 'foo',
def: '[asf]',
xyz: 5
}
คำตอบ:
การแก้ไขนี้ปรับปรุงและอธิบายคำตอบตามความคิดเห็น
var search = location.search.substring(1);
JSON.parse('{"' + decodeURI(search).replace(/"/g, '\\"').replace(/&/g, '","').replace(/=/g,'":"') + '"}')
ตัวอย่าง
แยกวิเคราะห์abc=foo&def=%5Basf%5D&xyz=5
ในห้าขั้นตอน:
abc=foo","def=[asf]","xyz=5
abc":"foo","def":"[asf]","xyz":"5
{"abc":"foo","def":"[asf]","xyz":"5"}
ซึ่งเป็น JSON ที่ถูกกฎหมาย
โซลูชันที่ได้รับการปรับปรุงช่วยให้สามารถค้นหาอักขระได้มากขึ้นในสตริงการค้นหา มันใช้ฟังก์ชั่น reviver สำหรับการถอดรหัส URI:
var search = location.search.substring(1);
JSON.parse('{"' + search.replace(/&/g, '","').replace(/=/g,'":"') + '"}', function(key, value) { return key===""?value:decodeURIComponent(value) })
ตัวอย่าง
search = "abc=foo&def=%5Basf%5D&xyz=5&foo=b%3Dar";
จะช่วยให้
Object {abc: "foo", def: "[asf]", xyz: "5", foo: "b=ar"}
ซับหนึ่ง:
JSON.parse('{"' + decodeURI("abc=foo&def=%5Basf%5D&xyz=5".replace(/&/g, "\",\"").replace(/=/g,"\":\"")) + '"}')
JSON.parse('{"' + decodeURI(location.search.substring(1).replace(/&/g, "\",\"").replace(/=/g, "\":\"")) + '"}')
เริ่มตั้งแต่ ES6 และเปิดตัว Javascript มีโครงสร้างหลายอย่างเพื่อสร้างโซลูชันที่มีประสิทธิภาพสำหรับปัญหานี้
ซึ่งรวมถึงการใช้URLSearchParamsและตัววนซ้ำ
let params = new URLSearchParams('abc=foo&def=%5Basf%5D&xyz=5');
params.get("abc"); // "foo"
หากกรณีการใช้งานของคุณต้องการให้คุณแปลงเป็นวัตถุจริงคุณสามารถใช้ฟังก์ชันต่อไปนี้:
function paramsToObject(entries) {
let result = {}
for(let entry of entries) { // each 'entry' is a [key, value] tupple
const [key, value] = entry;
result[key] = value;
}
return result;
}
const urlParams = new URLSearchParams('abc=foo&def=%5Basf%5D&xyz=5');
const entries = urlParams.entries(); //returns an iterator of decoded [key,value] tuples
const params = paramsToObject(entries); //{abc:"foo",def:"[asf]",xyz:"5"}
เราสามารถใช้Object.fromEntries (ซึ่งขณะนี้อยู่ในขั้นตอนที่ 4) แทนที่ด้วยparamsToObject
Object.fromEntries(entries)
คู่ของค่าที่จะวนซ้ำเป็นคู่ของชื่อ - ค่าของรายการที่มีคีย์เป็นชื่อและค่าที่เป็นค่า
ตั้งแต่URLParams
ส่งคืนออบเจกต์ที่ทำซ้ำได้การใช้ตัวดำเนินการสเปรดแทนการเรียก.entries
จะให้ผลลัพธ์ตามข้อมูลจำเพาะ:
const urlParams = new URLSearchParams('abc=foo&def=%5Basf%5D&xyz=5');
const params = Object.fromEntries(urlParams); // {abc: "foo", def: "[asf]", xyz: "5"}
หมายเหตุ:ค่าทั้งหมดจะถูกสตริงโดยอัตโนมัติตามข้อมูลจำเพาะ URLSearchParams
ในฐานะที่เป็น@siipeชี้สตริงที่มีค่าเดียวกันที่สำคัญหลายจะถูกบังคับให้คุ้มค่าที่มีอยู่ล่าสุด: จะในสาระสำคัญกลายเป็น:foo=first_value&foo=second_value
{foo: "second_value"}
ตามคำตอบนี้: https://stackoverflow.com/a/1746566/1194694ไม่มีข้อมูลจำเพาะสำหรับการตัดสินใจว่าจะทำอย่างไรและแต่ละเฟรมเวิร์กสามารถทำงานแตกต่างกันได้
กรณีการใช้งานทั่วไปจะเป็นการรวมสองค่าเดียวกันเข้าในอาร์เรย์ทำให้วัตถุเอาต์พุตเป็น:
{foo: ["first_value", "second_value"]}
สามารถทำได้ด้วยรหัสต่อไปนี้:
const groupParamsByKey = (params) => [...params.entries()].reduce((acc, tuple) => {
// getting the key and value from each tuple
const [key, val] = tuple;
if(acc.hasOwnProperty(key)) {
// if the current key is already an array, we'll add the value to it
if(Array.isArray(acc[key])) {
acc[key] = [...acc[key], val]
} else {
// if it's not an array, but contains a value, we'll convert it into an array
// and add the current value to it
acc[key] = [acc[key], val];
}
} else {
// plain assignment if no special case is present
acc[key] = val;
}
return acc;
}, {});
const params = new URLSearchParams('abc=foo&def=%5Basf%5D&xyz=5&def=dude');
const output = groupParamsByKey(params) // {abc: "foo", def: ["[asf]", "dude"], xyz: 5}
search string
แยกวิเคราะห์ - ซึ่งเป็นสิ่งที่มันถูกออกแบบมาเพื่อทำโดยไม่คำนึงถึงการเชื่อมโยงกับ URL
Object.fromEntries
ใช้ไม่ได้กับกุญแจซ้ำ ถ้าเราพยายามที่จะทำบางสิ่งบางอย่างเช่นเราจะได้รับเพียง?foo=bar1&foo=bar2
{ foo: 'bar2' }
ตัวอย่างเช่นวัตถุคำขอ Node.js แยกวิเคราะห์เป็น{ foo: ['bar1', 'bar2'] }
let temp={};Object.keys(params).map(key=>{temp[key]=urlParams.getAll(key)})
ซับ ES6 หนึ่ง สะอาดและเรียบง่าย
Object.fromEntries(new URLSearchParams(location.search));
สำหรับกรณีเฉพาะของคุณมันจะเป็น:
console.log(
Object.fromEntries(new URLSearchParams('abc=foo&def=%5Basf%5D&xyz=5'))
);
"http://place.com?foo=bar&hello=%40world&showThing"
ผลิต{ hello: "@world", http://place.com?foo: "bar", showThing: "" }
?someValue=false
จะกลายเป็น{ someValue: "false" }
?foo=bar1&foo=bar2
{ foo: 'bar2' }
วัตถุคำขอ Node.js แยกวิเคราะห์เป็น{ foo: ['bar1', 'bar2'] }
location.search
แยกบน &
ที่จะได้รับคู่ชื่อ / =
ค่าแล้วแยกแต่ละคู่ใน นี่คือตัวอย่าง:
var str = "abc=foo&def=%5Basf%5D&xy%5Bz=5"
var obj = str.split("&").reduce(function(prev, curr, i, arr) {
var p = curr.split("=");
prev[decodeURIComponent(p[0])] = decodeURIComponent(p[1]);
return prev;
}, {});
วิธีอื่นโดยใช้นิพจน์ทั่วไป:
var obj = {};
str.replace(/([^=&]+)=([^&]*)/g, function(m, key, value) {
obj[decodeURIComponent(key)] = decodeURIComponent(value);
});
โซลูชันที่รัดกุม:
location.search
.slice(1)
.split('&')
.map(p => p.split('='))
.reduce((obj, pair) => {
const [key, value] = pair.map(decodeURIComponent);
return ({ ...obj, [key]: value })
}, {});
โซลูชันที่นำเสนอที่ฉันพบจนถึงขณะนี้ไม่ครอบคลุมสถานการณ์ที่ซับซ้อนมากขึ้น
ฉันต้องการแปลงสตริงแบบสอบถามเช่น
https://random.url.com?Target=Offer&Method=findAll&filters%5Bhas_goals_enabled%5D%5BTRUE%5D=1&filters%5Bstatus%5D=active&fields%5B%5D=id&fields%5B%5D=name&fields%5B%5D=default_goal_name
เป็นวัตถุที่ชอบ:
{
"Target": "Offer",
"Method": "findAll",
"fields": [
"id",
"name",
"default_goal_name"
],
"filters": {
"has_goals_enabled": {
"TRUE": "1"
},
"status": "active"
}
}
หรือ:
https://random.url.com?Target=Report&Method=getStats&fields%5B%5D=Offer.name&fields%5B%5D=Advertiser.company&fields%5B%5D=Stat.clicks&fields%5B%5D=Stat.conversions&fields%5B%5D=Stat.cpa&fields%5B%5D=Stat.payout&fields%5B%5D=Stat.date&fields%5B%5D=Stat.offer_id&fields%5B%5D=Affiliate.company&groups%5B%5D=Stat.offer_id&groups%5B%5D=Stat.date&filters%5BStat.affiliate_id%5D%5Bconditional%5D=EQUAL_TO&filters%5BStat.affiliate_id%5D%5Bvalues%5D=1831&limit=9999
เป็น:
{
"Target": "Report",
"Method": "getStats",
"fields": [
"Offer.name",
"Advertiser.company",
"Stat.clicks",
"Stat.conversions",
"Stat.cpa",
"Stat.payout",
"Stat.date",
"Stat.offer_id",
"Affiliate.company"
],
"groups": [
"Stat.offer_id",
"Stat.date"
],
"limit": "9999",
"filters": {
"Stat.affiliate_id": {
"conditional": "EQUAL_TO",
"values": "1831"
}
}
}
รหัส:
var getParamsAsObject = function (query) {
query = query.substring(query.indexOf('?') + 1);
var re = /([^&=]+)=?([^&]*)/g;
var decodeRE = /\+/g;
var decode = function (str) {
return decodeURIComponent(str.replace(decodeRE, " "));
};
var params = {}, e;
while (e = re.exec(query)) {
var k = decode(e[1]), v = decode(e[2]);
if (k.substring(k.length - 2) === '[]') {
k = k.substring(0, k.length - 2);
(params[k] || (params[k] = [])).push(v);
}
else params[k] = v;
}
var assign = function (obj, keyPath, value) {
var lastKeyIndex = keyPath.length - 1;
for (var i = 0; i < lastKeyIndex; ++i) {
var key = keyPath[i];
if (!(key in obj))
obj[key] = {}
obj = obj[key];
}
obj[keyPath[lastKeyIndex]] = value;
}
for (var prop in params) {
var structure = prop.split('[');
if (structure.length > 1) {
var levels = [];
structure.forEach(function (item, i) {
var key = item.replace(/[?[\]\\ ]/g, '');
levels.push(key);
});
assign(params, levels, params[prop]);
delete(params[prop]);
}
}
return params;
};
obj=encodeURIComponent(JSON.stringify({what:{ever:','},i:['like']}))
ผมคิดว่านี่เป็นเพียงแค่ความซับซ้อนสิ่งที่ฉันต้องการเพียงแค่ผ่าน
นี่เป็นรุ่นง่าย ๆ แน่นอนคุณจะต้องเพิ่มการตรวจสอบข้อผิดพลาด:
var obj = {};
var pairs = queryString.split('&');
for(i in pairs){
var split = pairs[i].split('=');
obj[decodeURIComponent(split[0])] = decodeURIComponent(split[1]);
}
JSON.parse('{"' + decodeURIComponent(query.replace(/"/g, '\\"').replace(/&/g, '","').replace(/=/g,'":"') + '"}'));
ทำงานให้ฉัน
name[]=test1&name[]=test2
และจะส่งผลให้name[]=test2
ฉันพบ$ .String.deparamโซลูชั่นที่สร้างไว้ล่วงหน้าที่สมบูรณ์แบบที่สุด ตรวจสอบเอกสาร
สำหรับกรณีเฉพาะของคุณ:
Object.fromEntries(new URLSearchParams('abc=foo&def=%5Basf%5D&xyz=5'));
สำหรับกรณีทั่วไปที่มีคนต้องการแยกวิเคราะห์แบบสอบถามไปยังวัตถุ:
Object.fromEntries(new URLSearchParams(location.search));
หากคุณไม่สามารถใช้ Object.fromEntries สิ่งนี้จะทำงาน:
Array.from(new URLSearchParams(window.location.search)).reduce((o, i) => ({ ...o, [i[0]]: i[1] }), {});
[...new URLSearchParams(window.location.search)].reduce((o, i) => ({ ...o, [i[0]]: i[1] }), {});
"http://place.com?foo=bar&hello=%40world&showThing
{ hello: "@world", http://place.com?foo: "bar", showThing: "" }
ลองเพิ่มstr.split("?").pop()
โซลูชันอื่นที่อ้างอิงมาตรฐานล่าสุดของ URLSearchParams ( https://developer.mozilla.org/en-US/docs/Web/API/URLSearchParams )
function getQueryParamsObject() {
const searchParams = new URLSearchParams(location.search.slice(1));
return searchParams
? _.fromPairs(Array.from(searchParams.entries()))
: {};
}
โปรดทราบว่าโซลูชันนี้กำลังใช้งานอยู่
Array.from ( https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/from )
และ_.fromPairs ( https://lodash.com/docs#fromPairs ) ของ lodash เพื่อความเรียบง่าย
ควรสร้างโซลูชันที่เข้ากันได้ง่ายขึ้นเนื่องจากคุณสามารถเข้าถึงตัววนซ้ำsearchParams.entries ()
ฉันมีปัญหาเดียวกันลองใช้วิธีแก้ปัญหาที่นี่ แต่ก็ไม่ได้ผลเพราะฉันมีอาร์เรย์ในพารามิเตอร์ URL เช่นนี้:
?param[]=5¶m[]=8&othr_param=abc¶m[]=string
ดังนั้นฉันจึงสิ้นสุดการเขียนฟังก์ชัน JS ของตัวเองซึ่งสร้างอาร์เรย์จากพารามิเตอร์ใน URI:
/**
* Creates an object from URL encoded data
*/
var createObjFromURI = function() {
var uri = decodeURI(location.search.substr(1));
var chunks = uri.split('&');
var params = Object();
for (var i=0; i < chunks.length ; i++) {
var chunk = chunks[i].split('=');
if(chunk[0].search("\\[\\]") !== -1) {
if( typeof params[chunk[0]] === 'undefined' ) {
params[chunk[0]] = [chunk[1]];
} else {
params[chunk[0]].push(chunk[1]);
}
} else {
params[chunk[0]] = chunk[1];
}
}
return params;
}
if(chunk[0].search("\\[\\]") !== -1) {
นั้นคือchunk[0]=chunk[0].replace(/\[\]$/,'');
const
แทนvar
เพราะบางคนอาจทำcreateObjFromURI = 'some text'
แล้วพวกเขาก็จะทำให้รหัสสับสน หากคุณใช้งานconst
บางคนที่ทำงานอยู่createObjFromURI = 'some text'
จะทำให้ข้อผิดพลาดไม่สามารถกำหนดค่าให้กับตัวแปรคงที่ได้
ใช้ ES6, URL API และ URLSearchParams API
function objectifyQueryString(url) {
let _url = new URL(url);
let _params = new URLSearchParams(_url.search);
let query = Array.from(_params.keys()).reduce((sum, value)=>{
return Object.assign({[value]: _params.get(value)}, sum);
}, {});
return query;
}
ซับ ES6 หนึ่ง (ถ้าเราเรียกได้ว่าเป็นวิธีที่เห็นเส้นยาว)
[...new URLSearchParams(location.search).entries()].reduce((prev, [key,val]) => {prev[key] = val; return prev}, {})
cur
เพื่อความชัดเจนเพิ่ม .reduce((prev, [key, val]) => {prev[key] = val})
ค่อนข้างง่ายด้วยURLSearchParams
JavaScript Web API
var paramsString = "q=forum&topic=api";
//returns an iterator object
var searchParams = new URLSearchParams(paramsString);
//Usage
for (let p of searchParams) {
console.log(p);
}
//Get the query strings
console.log(searchParams.toString());
//You can also pass in objects
var paramsObject = {q:"forum",topic:"api"}
//returns an iterator object
var searchParams = new URLSearchParams(paramsObject);
//Usage
for (let p of searchParams) {
console.log(p);
}
//Get the query strings
console.log(searchParams.toString());
หมายเหตุ : ไม่รองรับใน IE
สำหรับ Node JS คุณสามารถใช้ Node JS API querystring
:
const querystring = require('querystring');
querystring.parse('abc=foo&def=%5Basf%5D&xyz=5&foo=b%3Dar');
// returns the object
เอกสารประกอบ: https://nodejs.org/api/querystring.html
ไม่มีวิธีแก้ปัญหาดั้งเดิมที่ฉันรู้ Dojo มีวิธี unserialization ในตัวหากคุณใช้เฟรมเวิร์กนั้นโดยบังเอิญ
มิฉะนั้นคุณสามารถใช้มันได้ด้วยตัวเอง:
function unserialize(str) {
str = decodeURIComponent(str);
var chunks = str.split('&'),
obj = {};
for(var c=0; c < chunks.length; c++) {
var split = chunks[c].split('=', 2);
obj[split[0]] = split[1];
}
return obj;
}
แก้ไข: เพิ่ม decodeURIComponent ()
มีไลบรารี่ขนาดเล็กที่เรียกว่าYouAreI.jsซึ่งผ่านการทดสอบแล้วและทำให้ง่ายมาก
YouAreI = require('YouAreI')
uri = new YouAreI('http://user:pass@www.example.com:3000/a/b/c?d=dad&e=1&f=12.3#fragment');
uri.query_get() => { d: 'dad', e: '1', f: '12.3' }
หนึ่งในวิธีที่ง่ายที่สุดในการทำเช่นนี้โดยใช้ส่วนต่อประสาน URLSearchParam
ด้านล่างเป็นข้อมูลโค้ดที่ใช้งานได้:
let paramObj={},
querystring=window.location.search,
searchParams = new URLSearchParams(querystring);
//*** :loop to add key and values to the param object.
searchParams.forEach(function(value, key) {
paramObj[key] = value;
});
นี่น่าจะเป็นทางออกที่ดีที่สุดเนื่องจากต้องคำนึงถึงพารามิเตอร์หลายตัวที่มีชื่อเดียวกันในการพิจารณา
function paramsToJSON(str) {
var pairs = str.split('&');
var result = {};
pairs.forEach(function(pair) {
pair = pair.split('=');
var name = pair[0]
var value = pair[1]
if( name.length )
if (result[name] !== undefined) {
if (!result[name].push) {
result[name] = [result[name]];
}
result[name].push(value || '');
} else {
result[name] = value || '';
}
});
return( result );
}
<a href="index.html?x=1&x=2&x=3&y=blah">something</a>
paramsToJSON("x=1&x=2&x=3&y=blah");
console yields => {x: Array[3], y: "blah"} where x is an array as is proper JSON
ฉันภายหลังตัดสินใจแปลงเป็นปลั๊กอิน jQuery ด้วย ...
$.fn.serializeURLParams = function() {
var result = {};
if( !this.is("a") || this.attr("href").indexOf("?") == -1 )
return( result );
var pairs = this.attr("href").split("?")[1].split('&');
pairs.forEach(function(pair) {
pair = pair.split('=');
var name = decodeURI(pair[0])
var value = decodeURI(pair[1])
if( name.length )
if (result[name] !== undefined) {
if (!result[name].push) {
result[name] = [result[name]];
}
result[name].push(value || '');
} else {
result[name] = value || '';
}
});
return( result )
}
<a href="index.html?x=1&x=2&x=3&y=blah">something</a>
$("a").serializeURLParams();
console yields => {x: Array[3], y: "blah"} where x is an array as is proper JSON
ตอนนี้คนแรกจะยอมรับพารามิเตอร์เท่านั้น แต่ปลั๊กอิน jQuery จะใช้เวลา URL ทั้งหมดและกลับพารามิเตอร์อนุกรม
นี่คือสิ่งที่ฉันใช้:
var params = {};
window.location.search.substring(1).split('&').forEach(function(pair) {
pair = pair.split('=');
if (pair[1] !== undefined) {
var key = decodeURIComponent(pair[0]),
val = decodeURIComponent(pair[1]),
val = val ? val.replace(/\++/g,' ').trim() : '';
if (key.length === 0) {
return;
}
if (params[key] === undefined) {
params[key] = val;
}
else {
if ("function" !== typeof params[key].push) {
params[key] = [params[key]];
}
params[key].push(val);
}
}
});
console.log(params);
การใช้งานขั้นพื้นฐานเช่น
?a=aa&b=bb
Object {a: "aa", b: "bb"}
ทำซ้ำพารามิเตอร์เช่น
?a=aa&b=bb&c=cc&c=potato
Object {a: "aa", b: "bb", c: ["cc","potato"]}
คีย์ที่หายไปเช่น
?a=aa&b=bb&=cc
Object {a: "aa", b: "bb"}
ค่าที่ขาดหายไปเช่น
?a=aa&b=bb&c
Object {a: "aa", b: "bb"}
โซลูชัน JSON / regex ด้านบนมีข้อผิดพลาดทางไวยากรณ์ใน url แปลกประหลาดนี้:
?a=aa&b=bb&c=&=dd&e
Object {a: "aa", b: "bb", c: ""}
นี่คือเวอร์ชันที่รวดเร็วและสกปรกของฉันโดยทั่วไปแล้วมันแยกพารามิเตอร์ URL ที่คั่นด้วย '&' ลงในองค์ประกอบอาร์เรย์แล้ววนซ้ำไปตามอาร์เรย์นั้นโดยเพิ่มคู่คีย์ / ค่าคั่นด้วย '=' ลงในวัตถุ ฉันใช้ decodeURIComponent () เพื่อแปลอักขระที่เข้ารหัสเป็นสตริงเทียบเท่าปกติ (ดังนั้น% 20 กลายเป็นช่องว่าง,% 26 กลายเป็น '&' ฯลฯ ):
function deparam(paramStr) {
let paramArr = paramStr.split('&');
let paramObj = {};
paramArr.forEach(e=>{
let param = e.split('=');
paramObj[param[0]] = decodeURIComponent(param[1]);
});
return paramObj;
}
ตัวอย่าง:
deparam('abc=foo&def=%5Basf%5D&xyz=5')
ผลตอบแทน
{
abc: "foo"
def:"[asf]"
xyz :"5"
}
ปัญหาเดียวคือ xyz เป็นสตริงและไม่ใช่ตัวเลข (เนื่องจากการใช้ decodeURIComponent ()) แต่เกินกว่านั้นจะไม่ใช่จุดเริ่มต้นที่ไม่ดี
//under ES6
const getUrlParamAsObject = (url = window.location.href) => {
let searchParams = url.split('?')[1];
const result = {};
//in case the queryString is empty
if (searchParams!==undefined) {
const paramParts = searchParams.split('&');
for(let part of paramParts) {
let paramValuePair = part.split('=');
//exclude the case when the param has no value
if(paramValuePair.length===2) {
result[paramValuePair[0]] = decodeURIComponent(paramValuePair[1]);
}
}
}
return result;
}
Babel
ความช่วยเหลือของคุณสามารถทำได้ดีภายใต้สภาพแวดล้อมอื่น ๆ
ใช้ phpjs
function parse_str(str, array) {
// discuss at: http://phpjs.org/functions/parse_str/
// original by: Cagri Ekin
// improved by: Michael White (http://getsprink.com)
// improved by: Jack
// improved by: Brett Zamir (http://brett-zamir.me)
// bugfixed by: Onno Marsman
// bugfixed by: Brett Zamir (http://brett-zamir.me)
// bugfixed by: stag019
// bugfixed by: Brett Zamir (http://brett-zamir.me)
// bugfixed by: MIO_KODUKI (http://mio-koduki.blogspot.com/)
// reimplemented by: stag019
// input by: Dreamer
// input by: Zaide (http://zaidesthings.com/)
// input by: David Pesta (http://davidpesta.com/)
// input by: jeicquest
// note: When no argument is specified, will put variables in global scope.
// note: When a particular argument has been passed, and the returned value is different parse_str of PHP. For example, a=b=c&d====c
// test: skip
// example 1: var arr = {};
// example 1: parse_str('first=foo&second=bar', arr);
// example 1: $result = arr
// returns 1: { first: 'foo', second: 'bar' }
// example 2: var arr = {};
// example 2: parse_str('str_a=Jack+and+Jill+didn%27t+see+the+well.', arr);
// example 2: $result = arr
// returns 2: { str_a: "Jack and Jill didn't see the well." }
// example 3: var abc = {3:'a'};
// example 3: parse_str('abc[a][b]["c"]=def&abc[q]=t+5');
// returns 3: {"3":"a","a":{"b":{"c":"def"}},"q":"t 5"}
var strArr = String(str)
.replace(/^&/, '')
.replace(/&$/, '')
.split('&'),
sal = strArr.length,
i, j, ct, p, lastObj, obj, lastIter, undef, chr, tmp, key, value,
postLeftBracketPos, keys, keysLen,
fixStr = function(str) {
return decodeURIComponent(str.replace(/\+/g, '%20'));
};
if (!array) {
array = this.window;
}
for (i = 0; i < sal; i++) {
tmp = strArr[i].split('=');
key = fixStr(tmp[0]);
value = (tmp.length < 2) ? '' : fixStr(tmp[1]);
while (key.charAt(0) === ' ') {
key = key.slice(1);
}
if (key.indexOf('\x00') > -1) {
key = key.slice(0, key.indexOf('\x00'));
}
if (key && key.charAt(0) !== '[') {
keys = [];
postLeftBracketPos = 0;
for (j = 0; j < key.length; j++) {
if (key.charAt(j) === '[' && !postLeftBracketPos) {
postLeftBracketPos = j + 1;
} else if (key.charAt(j) === ']') {
if (postLeftBracketPos) {
if (!keys.length) {
keys.push(key.slice(0, postLeftBracketPos - 1));
}
keys.push(key.substr(postLeftBracketPos, j - postLeftBracketPos));
postLeftBracketPos = 0;
if (key.charAt(j + 1) !== '[') {
break;
}
}
}
}
if (!keys.length) {
keys = [key];
}
for (j = 0; j < keys[0].length; j++) {
chr = keys[0].charAt(j);
if (chr === ' ' || chr === '.' || chr === '[') {
keys[0] = keys[0].substr(0, j) + '_' + keys[0].substr(j + 1);
}
if (chr === '[') {
break;
}
}
obj = array;
for (j = 0, keysLen = keys.length; j < keysLen; j++) {
key = keys[j].replace(/^['"]/, '')
.replace(/['"]$/, '');
lastIter = j !== keys.length - 1;
lastObj = obj;
if ((key !== '' && key !== ' ') || j === 0) {
if (obj[key] === undef) {
obj[key] = {};
}
obj = obj[key];
} else { // To insert new dimension
ct = -1;
for (p in obj) {
if (obj.hasOwnProperty(p)) {
if (+p > ct && p.match(/^\d+$/g)) {
ct = +p;
}
}
}
key = ct + 1;
}
}
lastObj[key] = value;
}
}
}
การสร้างคำตอบของMike Causerฉันได้ทำฟังก์ชั่นนี้ซึ่งคำนึงถึงหลายfoo=bar&foo=baz
พารามิเตอร์ด้วยคีย์เดียวกัน ( ) และพารามิเตอร์ที่คั่นด้วยเครื่องหมายจุลภาค ( foo=bar,baz,bin
) นอกจากนี้ยังช่วยให้คุณค้นหาคีย์คิวรีที่แน่นอน
function getQueryParams(queryKey) {
var queryString = window.location.search;
var query = {};
var pairs = (queryString[0] === '?' ? queryString.substr(1) : queryString).split('&');
for (var i = 0; i < pairs.length; i++) {
var pair = pairs[i].split('=');
var key = decodeURIComponent(pair[0]);
var value = decodeURIComponent(pair[1] || '');
// Se possui uma vírgula no valor, converter em um array
value = (value.indexOf(',') === -1 ? value : value.split(','));
// Se a key já existe, tratar ela como um array
if (query[key]) {
if (query[key].constructor === Array) {
// Array.concat() faz merge se o valor inserido for um array
query[key] = query[key].concat(value);
} else {
// Se não for um array, criar um array contendo o valor anterior e o novo valor
query[key] = [query[key], value];
}
} else {
query[key] = value;
}
}
if (typeof queryKey === 'undefined') {
return query;
} else {
return query[queryKey];
}
}
อินพุตตัวอย่าง:
foo.html?foo=bar&foo=baz&foo=bez,boz,buz&bar=1,2,3
ตัวอย่างผลลัพธ์
{
foo: ["bar","baz","bez","boz","buz"],
bar: ["1","2","3"]
}
หากคุณใช้ URI.js คุณสามารถใช้:
https://medialize.github.io/URI.js/docs.html#static-parseQuery
var result = URI.parseQuery("?foo=bar&hello=world&hello=mars&bam=&yup");
result === {
foo: "bar",
hello: ["world", "mars"],
bam: "",
yup: null
};
console.log(decodeURI('abc=foo&def=%5Basf%5D&xyz=5')
.split('&')
.reduce((result, current) => {
const [key, value] = current.split('=');
result[key] = value;
return result
}, {}))
ครั้งแรกที่คุณต้องกำหนดสิ่งที่ได้รับ VAR:
function getVar()
{
this.length = 0;
this.keys = [];
this.push = function(key, value)
{
if(key=="") key = this.length++;
this[key] = value;
this.keys.push(key);
return this[key];
}
}
กว่าแค่อ่าน:
function urlElement()
{
var thisPrototype = window.location;
for(var prototypeI in thisPrototype) this[prototypeI] = thisPrototype[prototypeI];
this.Variables = new getVar();
if(!this.search) return this;
var variables = this.search.replace(/\?/g,'').split('&');
for(var varI=0; varI<variables.length; varI++)
{
var nameval = variables[varI].split('=');
var name = nameval[0].replace(/\]/g,'').split('[');
var pVariable = this.Variables;
for(var nameI=0;nameI<name.length;nameI++)
{
if(name.length-1==nameI) pVariable.push(name[nameI],nameval[1]);
else var pVariable = (typeof pVariable[name[nameI]] != 'object')? pVariable.push(name[nameI],new getVar()) : pVariable[name[nameI]];
}
}
}
และใช้เช่น:
var mlocation = new urlElement();
mlocation = mlocation.Variables;
for(var key=0;key<mlocation.keys.length;key++)
{
console.log(key);
console.log(mlocation[mlocation.keys[key]];
}
ฉันจำเป็นต้องจัดการกับ+
ในส่วนการค้นหาของ URL ( decodeURIComponent ไม่ได้ ) ดังนั้นฉันจึงปรับรหัสของ Wolfgang ให้เป็น:
var search = location.search.substring(1);
search = search?JSON.parse('{"' + search.replace(/\+/g, ' ').replace(/&/g, '","').replace(/=/g,'":"') + '"}',
function(key, value) { return key===""?value:decodeURIComponent(value)}):{};
ในกรณีของฉันฉันใช้ jQuery เพื่อรับพารามิเตอร์แบบฟอร์มพร้อม URL จากนั้นเคล็ดลับในการสร้างวัตถุจากนั้นฉันสามารถอัปเดตพารามิเตอร์บนวัตถุและสร้าง URL แบบสอบถามได้อย่างง่ายดายเช่น:
var objForm = JSON.parse('{"' + $myForm.serialize().replace(/\+/g, ' ').replace(/&/g, '","').replace(/=/g,'":"') + '"}',
function(key, value) { return key===""?value:decodeURIComponent(value)});
objForm.anyParam += stringToAddToTheParam;
var serializedForm = $.param(objForm);
ฉันทำอย่างนี้:
const uri = new URL('https://example.org/?myvar1=longValue&myvar2=value')
const result = {}
for (let p of uri.searchParams) {
result[p[0]] = p[1]
}
หากคุณต้องการการสอบถามซ้ำคุณสามารถใช้ไลบรารีjs-extension-lingจิ๋ว
npm i js-extension-ling
const jsx = require("js-extension-ling");
console.log(jsx.queryStringToObject("a=1"));
console.log(jsx.queryStringToObject("a=1&a=3"));
console.log(jsx.queryStringToObject("a[]=1"));
console.log(jsx.queryStringToObject("a[]=1&a[]=pomme"));
console.log(jsx.queryStringToObject("a[0]=one&a[1]=five"));
console.log(jsx.queryStringToObject("http://blabla?foo=bar&number=1234"));
console.log(jsx.queryStringToObject("a[fruits][red][]=strawberry"));
console.log(jsx.queryStringToObject("a[fruits][red][]=strawberry&a[1]=five&a[fruits][red][]=cherry&a[fruits][yellow][]=lemon&a[fruits][yellow][688]=banana"));
สิ่งนี้จะให้ผลลัพธ์ดังนี้:
{ a: '1' }
{ a: '3' }
{ a: { '0': '1' } }
{ a: { '0': '1', '1': 'pomme' } }
{ a: { '0': 'one', '1': 'five' } }
{ foo: 'bar', number: '1234' }
{
a: { fruits: { red: { '0': 'strawberry' } } }
}
{
a: {
'1': 'five',
fruits: {
red: { '0': 'strawberry', '1': 'cherry' },
yellow: { '0': 'lemon', '688': 'banana' }
}
}
}
หมายเหตุ: มันขึ้นอยู่กับฟังก์ชั่น locutus parse_str ( https://locutus.io/php/strings/parse_str/ )