HTTP請(qǐng)求參數(shù)的編碼和解碼(瀏覽器和服務(wù)端聯(lián)調(diào)手冊(cè))

使用瀏覽器猜煮,查看頁(yè)面撼班,點(diǎn)擊按鈕提交信息歧匈,實(shí)際上是一次次http請(qǐng)求。瀏覽器在發(fā)送這些請(qǐng)求時(shí)砰嘁,是需要對(duì)請(qǐng)求參數(shù)中的特殊字符做編碼的件炉,服務(wù)端需要對(duì)參數(shù)解碼,才能知道參數(shù)的原始內(nèi)容矮湘,再進(jìn)行處理斟冕。不同的場(chǎng)景,編碼方式是存在差別的缅阳。

GET請(qǐng)求

get請(qǐng)求的參數(shù)會(huì)在URL的中磕蛇,在“?”的后面十办。一般來(lái)說(shuō)秀撇,URL只能使用英文字母、阿拉伯?dāng)?shù)字和某些標(biāo)點(diǎn)符號(hào)向族,不能使用其他文字和符號(hào)捌袜。比如,世界上有英文字母的網(wǎng)址 http://www.abc.com炸枣,但是沒(méi)有中文網(wǎng)址 http://www.我是網(wǎng)站.com虏等。

這是因?yàn)榫W(wǎng)絡(luò)標(biāo)準(zhǔn)RFC 1738做了硬性規(guī)定:

"只有字母和數(shù)字[0-9a-zA-Z]、一些特殊符號(hào)"-_.!*'()+"[不包括雙引號(hào)]适肠,和保留字(&霍衫,?之類的)才可以不經(jīng)過(guò)編碼直接用于URL侯养。"

這個(gè)編碼是由瀏覽器完成的敦跌,編碼方式叫做“URL編碼”

場(chǎng)景:點(diǎn)擊鏈接

注意參數(shù)name含有空格

<a href="/?name=前  ++端">前端</a>

抓包得到:

GET /?name=%E5%89%8D%20%20++%E7%AB%AF HTTP/1.1

瀏覽器html文件中head標(biāo)簽中的頁(yè)面編碼方式 utf-8

<meta content="text/html; charset=utf-8" http-equiv="content-type">

把中文的“前端” 用 utf-8字符集的十六進(jìn)制 “E5898D”(前), “E7ABAF”(端)逛揩,然后用”%“分割柠傍,空格轉(zhuǎn)化成了“%20”

場(chǎng)景:form表單提交,method默認(rèn)為“get”辩稽,enctype編碼方式默認(rèn)為“application/x-www-form-urlencoded”

“application/x-www-form-urlencoded”惧笛,特殊字符編碼方式和“點(diǎn)擊鏈接”場(chǎng)景是一樣的,唯一的區(qū)別是空格會(huì)轉(zhuǎn)化為“+”逞泄,內(nèi)容中的
“+”url編碼為“%2B”

例如:

value值為“前 ++端”含有空格和+

        <h2>get</h2>
        <form action="/">
            <label>name</label><input name="name" value="前  ++端"/>
            <input type="submit" id="submit" value="提交"/>
        </form>

點(diǎn)擊提交表單患整,抓包得到:

GET /?name=%E5%89%8D++%2B%2B%E7%AB%AF HTTP/1.1

ajax異步提交

let xhr = new XMLHttpRequest()
xhr.open('get', '/?name=前  ++端', true)
xhr.send()

抓包得到:

GET /?name=%E5%89%8D%20%20++%E7%AB%AF HTTP/1.1

編碼的結(jié)果和點(diǎn)擊鏈接場(chǎng)景是一致的拜效。

結(jié)論:除了form表單的get方式提交的編碼方式不一樣,其他的都一樣各谚。

POST請(qǐng)求

form表單提交

非文件表單項(xiàng)提交

默認(rèn)的enctype="application/x-www-form-urlencoded"

        <h2>post</h2>
        <form action="/post" method="post">
            <label>name</label><input name="name" value="前  ++端"/>
            <input type="submit" id="submit" value="提交"/>
        </form>

點(diǎn)擊提交后紧憾,抓包得到:

POST /post HTTP/1.1
Host: localhost:3001
Content-Length: 31
Pragma: no-cache
Cache-Control: no-cache
Origin: http://localhost:3001
Upgrade-Insecure-Requests: 1
**Content-Type: application/x-www-form-urlencoded**
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_13_3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/64.0.3282.186 Safari/537.36
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8
DNT: 1
Accept-Encoding: gzip, deflate, br

**name=%E5%89%8D++%2B%2B%E7%AB%AF**

post請(qǐng)求傳輸?shù)膮?shù)在請(qǐng)求體中,編碼成了 “name=%E5%89%8D++%2B%2B%E7%AB%AF”昌渤,說(shuō)明使用enctype="application/x-www-form-urlencoded"方式
編碼赴穗,get和post的結(jié)果是一樣的。

另外post提交需要關(guān)注“Content-Type”這個(gè)請(qǐng)求頭膀息,這個(gè)請(qǐng)求頭告訴服務(wù)端望抽,我請(qǐng)求體內(nèi)容和編碼格式。

文件上傳

文件上傳必須使用enctype="multipart/form-data"履婉,表示請(qǐng)求體內(nèi)容不做URL編碼,原文傳輸

        <h2>file</h2>
        <form action="/post" id="fileupload" method="post" enctype="multipart/form-data">
            <label>name</label><input name="name" value="前  ++端"/>
            <label>file</label><input type="file" name="file"/>
            <input type="submit" id="submit" value="提交"/>
        </form>

點(diǎn)擊提交按鈕后斟览,抓包得到:

alt

注意:Content-Type為: multipart/form-data; boundary=----WebKitFormBoundaryIAobIVW9hww2sV7i
表示編碼是multipart/form-data毁腿,而且每一個(gè)表單項(xiàng)使用“----WebKitFormBoundaryIAobIVW9hww2sV7i”作為分割線。
然后name的值“前 ++端”苛茂,也沒(méi)有做編碼處理已烤。

ajax異步提交

原生ajax

xhr.open('post', '/post', true)
xhr.send(JSON.stringify({name:'前  ++端'}))

抓包得到:

POST /post HTTP/1.1
Host: localhost:3001
Content-Length: 21
Pragma: no-cache
Cache-Control: no-cache
Origin: http://localhost:3001
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_13_3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/64.0.3282.186 Safari/537.36
Content-Type: text/plain;charset=UTF-8
Accept: */*
DNT: 1

{"name":"前  ++端"}

ajax原生post請(qǐng)求,瀏覽器對(duì)請(qǐng)求體不做任何編碼處理妓羊,Content-Type: text/plain;charset=UTF-8

Jquery

$.ajax({
    url: '/post',
    method: 'post',
    data: {
        name: '前  ++端',
    }
})

jquery對(duì)于ajax的請(qǐng)求內(nèi)容的處理胯究,默認(rèn)的編碼方式application/x-www-form-urlencoded;

POST /post HTTP/1.1
Host: localhost:3001
Content-Length: 31
Pragma: no-cache
Cache-Control: no-cache
Accept: */*
Origin: http://localhost:3001
X-Requested-With: XMLHttpRequest
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_13_3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/64.0.3282.186 Safari/537.36
Content-Type: application/x-www-form-urlencoded; charset=UTF-8
DNT: 1

name=%E5%89%8D++%2B%2B%E7%AB%AF

因?yàn)閖query默認(rèn)是application/x-www-form-urlencoded,但是對(duì)于使用formdata做圖片上傳的話,他不能自動(dòng)識(shí)別出formdata躁绸,需要使用enctype="multipart/form-data"裕循,只能手動(dòng)調(diào)整參數(shù)processData,contentType來(lái)解決這個(gè)問(wèn)題:

let formdata = new FormData(document.querySelector("form"));
$.ajax({
  url: "/post",
  type: "post",
  data: formdata,
  processData: false,  // 不處理數(shù)據(jù)
  contentType: false   // 不設(shè)置內(nèi)容類型
});

axios

axios是最近在前端圈內(nèi)大面積使用的ajax框架,功能強(qiáng)悍净刮,可以自行g(shù)oogle搜索剥哑。

axios.post('/post', {
    name: '前  ++端',
})

得到:

POST /post HTTP/1.1
Host: localhost:3001
Content-Length: 21
Pragma: no-cache
Cache-Control: no-cache
Accept: application/json, text/plain, */*
Origin: http://localhost:3001
User-Agent: Mozilla/5.0 (Macintosh; Intel Mac OS X 10_13_3) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/64.0.3282.186 Safari/537.36
Content-Type: application/json;charset=UTF-8
DNT: 1

{"name":"前  ++端"}

axios默認(rèn)編碼方式是將請(qǐng)求體內(nèi)容json stringfy,然后設(shè)置Content-Type: application/json;charset=UTF-8

在聯(lián)調(diào)是淹父,會(huì)出現(xiàn)很多服務(wù)端不支持json格式的數(shù)據(jù)的處理株婴,只支持傳統(tǒng)的application/x-www-form-urlencoded,和傳統(tǒng)的表單提交一樣的格式

這時(shí)就需要使用QS框架了

import qs from 'qs'

axios.post('/post', qs.stringify({
    name: '前  ++端'
}))

這樣能夠提交application/x-www-form-urlencoded編碼的數(shù)據(jù)暑认。

axios能夠智能判斷需要提交的數(shù)據(jù)困介,如果是formdata,會(huì)自動(dòng)切換成使用multipart/form-data編碼

let formdata = new FormData(document.querySelector("form"));
axios.post('/post', formdata) //不要做任何處理蘸际,這個(gè)很贊Wā!

服務(wù)端解碼

目前使用nodejs框架koa和koa-body中間件做post請(qǐng)求參數(shù)的解碼粮彤,其他語(yǔ)言框架的處理思維也是一致的八回。

GET請(qǐng)求

decodeURIComponent(str.replace(/\+/g, ' '));

將使用form表單get請(qǐng)求酷愧,使用‘a(chǎn)pplication/x-www-form-urlencoded’編碼后的’+’,轉(zhuǎn)化成空格缠诅,再URL解碼

so easyH茉 !管引!

POST請(qǐng)求

alt

ctx.is 是做判斷士败,進(jìn)入不同的處理邏輯,ctx.js是什么褥伴?

var value = req.headers['content-type’]
return typeis(value, types)

ctx.js是服務(wù)端取請(qǐng)求頭中content-type谅将,做類型判斷。

所以只要 content-type 和 請(qǐng)求體內(nèi)容正確的一一對(duì)應(yīng)重慢,那么服務(wù)端就能正確的解碼出瀏覽器傳輸過(guò)來(lái)的內(nèi)容

json字符串:Content-Type: application/json;charset=UTF-8

alt

表單提交饥臂,URL編碼:Content-Type: application/x-www-form-urlencoded; charset=UTF-8

alt

文件上傳:Content-Type為: multipart/form-data; boundary=----WebKitFormBoundaryIAobIVW9hww2sV7i

alt
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市似踱,隨后出現(xiàn)的幾起案子隅熙,更是在濱河造成了極大的恐慌,老刑警劉巖核芽,帶你破解...
    沈念sama閱讀 211,194評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件囚戚,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡轧简,警方通過(guò)查閱死者的電腦和手機(jī)驰坊,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,058評(píng)論 2 385
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)哮独,“玉大人拳芙,你說(shuō)我怎么就攤上這事∑よ担” “怎么了态鳖?”我有些...
    開封第一講書人閱讀 156,780評(píng)論 0 346
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)恶导。 經(jīng)常有香客問(wèn)我浆竭,道長(zhǎng),這世上最難降的妖魔是什么惨寿? 我笑而不...
    開封第一講書人閱讀 56,388評(píng)論 1 283
  • 正文 為了忘掉前任邦泄,我火速辦了婚禮,結(jié)果婚禮上裂垦,老公的妹妹穿的比我還像新娘顺囊。我一直安慰自己,他們只是感情好蕉拢,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,430評(píng)論 5 384
  • 文/花漫 我一把揭開白布特碳。 她就那樣靜靜地躺著诚亚,像睡著了一般。 火紅的嫁衣襯著肌膚如雪午乓。 梳的紋絲不亂的頭發(fā)上站宗,一...
    開封第一講書人閱讀 49,764評(píng)論 1 290
  • 那天,我揣著相機(jī)與錄音益愈,去河邊找鬼梢灭。 笑死,一個(gè)胖子當(dāng)著我的面吹牛蒸其,可吹牛的內(nèi)容都是我干的敏释。 我是一名探鬼主播,決...
    沈念sama閱讀 38,907評(píng)論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼摸袁,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼钥顽!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起靠汁,我...
    開封第一講書人閱讀 37,679評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤蜂大,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后膀曾,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,122評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡阳啥,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,459評(píng)論 2 325
  • 正文 我和宋清朗相戀三年添谊,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片察迟。...
    茶點(diǎn)故事閱讀 38,605評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡斩狱,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出扎瓶,到底是詐尸還是另有隱情所踊,我是刑警寧澤,帶...
    沈念sama閱讀 34,270評(píng)論 4 329
  • 正文 年R本政府宣布概荷,位于F島的核電站秕岛,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏误证。R本人自食惡果不足惜继薛,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,867評(píng)論 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望愈捅。 院中可真熱鬧遏考,春花似錦、人聲如沸蓝谨。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,734評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至咖楣,卻和暖如春督笆,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背截歉。 一陣腳步聲響...
    開封第一講書人閱讀 31,961評(píng)論 1 265
  • 我被黑心中介騙來(lái)泰國(guó)打工胖腾, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人瘪松。 一個(gè)月前我還...
    沈念sama閱讀 46,297評(píng)論 2 360
  • 正文 我出身青樓咸作,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親宵睦。 傳聞我的和親對(duì)象是個(gè)殘疾皇子记罚,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,472評(píng)論 2 348

推薦閱讀更多精彩內(nèi)容