AJax - Xhr & Fetch

Asynchronous JavaScript and XML(異步的 JavaScript 和 XML)


XHR

XHR對(duì)象

var xhr;
if (window.XMLHttpRequest) {
  xhr = new XMLHttpRequest();    
} else {
  xhr = new ActiveXObject("Microsoft.XMLHTTP");//IE5,IE6
}

XHR對(duì)象的請(qǐng)求及相應(yīng)

request

  • xhr.setRequestHeader(header,value)
  • xhr.open(method='GET/POST',url,async='true/false')
  • xhr.send(DOMString/Document(XML)) //DOMString/Document(XML)僅用于POST

xhr2

  • xhr.send(DOMString/Document/FormData/Blob/File/ArrayBuffer)

response

  • xhr.responseText
  • xhr.responseXML

xhr2

  • xhr.responseType 'json/text/arraybuffer/blob/document'

XHR對(duì)象的事件

  • xhr.onloald()
  • xhr.onerror()

XHR對(duì)象的狀態(tài)

  • xhr.onreadystatechange 當(dāng)readyState屬性改變時(shí)眉孩,就會(huì)調(diào)用該函數(shù)瞒斩。
  • xhr.readyState 存有XMLHttpRequest的狀態(tài)。從04發(fā)生變化稠氮。
  • 0: 請(qǐng)求未初始化
  • 1: 服務(wù)器連接已建立
  • 2: 請(qǐng)求已接收
  • 3: 請(qǐng)求處理中
  • 4: 請(qǐng)求已完成兄猩,且響應(yīng)已就緒
  • xhr.status
  • 200: "OK"
  • 404: 未找到頁(yè)面

代碼

var xhr;
if (window.XMLHttpRequest) {
  xhr = new XMLHttpRequest();    
} else {
  xhr = new ActiveXObject("Microsoft.XMLHTTP");//IE5,IE6
}
xhr.onreadystatechange=function() {
  if (xhr.readyState==4 && xhr.status==200){
    var jsonString = xhr.response.json();
    ...
  }
}

xhr.setRequestHeader(header,value);
xhr.open("GET",url,true);
xhr.send();

利用xhr2特性抓取BLOB數(shù)據(jù)(圖片)

BlobBuilder = window.MozBlobBuilder || window.WebKitBlobBuilder || window.BlobBuilder;

var xhr = new XMLHttpRequest();
xhr.open('GET', '/path/to/image.png', true);
xhr.responseType = 'arraybuffer';

xhr.onload = function(e) {
  if (this.status == 200) {
    var bb = new BlobBuilder();
    bb.append(this.response); // Note: not xhr.responseText

    var blob = bb.getBlob('image/png');
    ...
  }
};

xhr.send();

xhr2提交表單

<form id="myform" name="myform" action="/server">
  <input type="text" name="username" value="johndoe">
  <input type="number" name="id" value="123456">
  <input type="submit" onclick="return sendForm(this.form);">
</form>

function sendForm(form) {
  var formData = new FormData(form);

  formData.append('secret_token', '1234567890'); // Append extra data before send.

  var xhr = new XMLHttpRequest();
  xhr.open('POST', form.action, true);
  xhr.onload = function(e) { ... };

  xhr.send(formData);

  return false; // Prevent page from submitting.
}

xhr2表單上傳文件

function uploadFiles(url, files) {
  var formData = new FormData();

  for (var i = 0, file; file = files[i]; ++i) {
    formData.append(file.name, file);
  }

  var xhr = new XMLHttpRequest();
  xhr.open('POST', url, true);
  xhr.onload = function(e) { ... };

  xhr.send(formData);  // multipart/form-data
}

document.querySelector('input[type="file"]').addEventListener('change', function(e) {
  uploadFiles('/server', this.files);
}, false);

xhr2上傳文件或 blob:xhr.send(Blob)
該示例使用 BlobBuilder API 從頭開(kāi)始創(chuàng)建新的文本文件茉盏,并將該 Blob 上傳到服務(wù)器鉴未。該代碼還設(shè)置了一個(gè)處理程序,用于通知用戶上傳進(jìn)度:

<progress min="0" max="100" value="0">0% complete</progress>
function upload(blobOrFile) {
  var xhr = new XMLHttpRequest();
  xhr.open('POST', '/server', true);
  xhr.onload = function(e) { ... };

  // Listen to the upload progress.
  var progressBar = document.querySelector('progress');
  xhr.upload.onprogress = function(e) {
    if (e.lengthComputable) {
      progressBar.value = (e.loaded / e.total) * 100;
      progressBar.textContent = progressBar.value; // Fallback for unsupported browsers.
    }
  };

  xhr.send(blobOrFile);
}

// Take care of vendor prefixes.
BlobBuilder = window.MozBlobBuilder || window.WebKitBlobBuilder || window.BlobBuilder;

var bb = new BlobBuilder();
bb.append('hello world');

upload(bb.getBlob('text/plain'));

xhr2上傳字節(jié):xhr.send(ArrayBuffer)

function sendArrayBuffer() {
  var xhr = new XMLHttpRequest();
  xhr.open('POST', '/server', true);
  xhr.onload = function(e) { ... };

  var uInt8Array = new Uint8Array([1, 2, 3]);

  xhr.send(uInt8Array.buffer);
}

xhr2分割文件并上傳各個(gè)部分

window.BlobBuilder = window.MozBlobBuilder || window.WebKitBlobBuilder ||
                     window.BlobBuilder;

function upload(blobOrFile) {
  var xhr = new XMLHttpRequest();
  xhr.open('POST', '/server', true);
  xhr.onload = function(e) { ... };
  xhr.send(blobOrFile);
}

document.querySelector('input[type="file"]').addEventListener('change', function(e) {
  var blob = this.files[0];

  const BYTES_PER_CHUNK = 1024 * 1024; // 1MB chunk sizes.
  const SIZE = blob.size;

  var start = 0;
  var end = BYTES_PER_CHUNK;

  while(start < SIZE) {

    // Note: blob.slice has changed semantics and been prefixed. See http://goo.gl/U9mE5.
    if ('mozSlice' in blob) {
      var chunk = blob.mozSlice(start, end);
    } else {
      var chunk = blob.webkitSlice(start, end);
    }

    upload(chunk);

    start = end;
    end = start + BYTES_PER_CHUNK;
  }
}, false);

})();

參考


fetch

fetch() 方法用于發(fā)起獲取資源的請(qǐng)求鸠姨。它返回一個(gè) promise铜秆,這個(gè)promise會(huì)在請(qǐng)求響應(yīng)后被resolve,并傳回 Response 對(duì)象讶迁。

當(dāng)遇到網(wǎng)絡(luò)錯(cuò)誤時(shí)连茧,fetch() 返回的 promise 會(huì)被 reject,并傳回 TypeError巍糯,雖然這也可能因?yàn)闄?quán)限或其它問(wèn)題導(dǎo)致啸驯。成功的 fetch() 檢查不僅要包括 promise 被 resolve,還要包括 Response.ok 屬性為 true祟峦。HTTP 404 狀態(tài)并不被認(rèn)為是網(wǎng)絡(luò)錯(cuò)誤罚斗。

可以使用isomorphic-fetch進(jìn)行前后端同構(gòu)應(yīng)用

語(yǔ)法

fetch(input[, init]).then(function(response) { ... });

  • input URLString/RequestObj
  • init(可選)
    一個(gè)配置項(xiàng)對(duì)象,包括所有對(duì)請(qǐng)求的設(shè)置宅楞≌胱耍可選的參數(shù)有:
  • method: 請(qǐng)求使用的方法,如 GET厌衙、POST距淫。
  • headers: 請(qǐng)求的頭信息,形式為 Headers 對(duì)象或 ByteString迅箩。
  • body: 請(qǐng)求的 body 信息:可能是一個(gè) Blob溉愁、BufferSource、FormData饲趋、URLSearchParams 或者 USVString 對(duì)象拐揭。注意 GETHEAD 方法的請(qǐng)求不能包含 body 信息。
  • mode: 請(qǐng)求的模式奕塑,如 cors堂污、 no-cors 或者 same-origin。
  • credentials: 請(qǐng)求的 credentials龄砰,如 omit盟猖、same-origin 或者 include。
  • cache: 請(qǐng)求的 cache 模式: default, no-store, reload, no-cache, force-cache, or only-if-cached.

返回值 一個(gè) Promise换棚,resolve 時(shí)回傳 Response 對(duì)象式镐。

代碼示例

var myImage = document.querySelector('img');
var myHeaders = new Headers();
myHeaders.append('Content-Type', 'image/jpeg');

var myInit = { method: 'GET',
               headers: myHeaders,
               mode: 'cors',
               cache: 'default' };

var myRequest = new Request('flowers.jpg');

fetch(myRequest,myInit).then(function(response) {
  ... 
});

// 也可以傳入:
var myRequest = new Request('flowers.jpg',myInit);
fetch(url).then(function(response) {
  return response.json();
}).then(function(data) {
  console.log(data);
}).catch(function(e) {
  console.log("Oops, error");
});
// ES6 =>
fetch(url).then(response => response.json())
  .then(data => console.log(data))
  .catch(e => console.log("Oops, error", e));
// ES7 async await
async function(){
  try {
  let response = await fetch(url);
  let data = await response.json();
  console.log(data);
  } catch(e) {
    console.log("Oops, error", e);
  }
}

兼容性

fetch-caniuse
fetch-caniuse

需要引入下面polyfill后完美支持IE8+

  1. 由于 IE8 是 ES3,需要引入 ES5 的 polyfill: es5-shim, es5-sham
  2. 引入 Promise 的 polyfill: es6-promise
  3. 引入 fetch 探測(cè)庫(kù):fetch-detector
  4. 引入 fetch 的 polyfill: fetch-ie8
  5. 可選:如果你還使用了 jsonp固蚤,引入 fetch-jsonp
  6. 可選:開(kāi)啟 Babel 的 runtime 模式娘汞,現(xiàn)在就使用 async/await

  • Fetch 請(qǐng)求默認(rèn)是不帶 cookie 的,需要設(shè)置 fetch(url, {credentials: 'include'})
  • 服務(wù)器返回 400夕玩,500 錯(cuò)誤碼時(shí)并不會(huì) reject你弦,只有網(wǎng)絡(luò)錯(cuò)誤這些導(dǎo)致請(qǐng)求不能完成時(shí)惊豺,fetch 才會(huì)被 reject。
  • IE8 IE9XHR不支持CORS跨域,推薦使用fetch-jsonp

不足

  • 沒(méi)有獲取狀態(tài)方法:isRejected禽作,isResolved
  • 缺少其它一些方法:always尸昧,progress,finally
  • 不能中斷旷偿,沒(méi)有 abort烹俗、terminate、onTimeout 或 cancel 方法

參考

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末衷蜓,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子尘喝,更是在濱河造成了極大的恐慌磁浇,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,544評(píng)論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件朽褪,死亡現(xiàn)場(chǎng)離奇詭異置吓,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)缔赠,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,430評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門衍锚,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人嗤堰,你說(shuō)我怎么就攤上這事戴质。” “怎么了踢匣?”我有些...
    開(kāi)封第一講書(shū)人閱讀 162,764評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵告匠,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我离唬,道長(zhǎng)后专,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,193評(píng)論 1 292
  • 正文 為了忘掉前任输莺,我火速辦了婚禮戚哎,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘嫂用。我一直安慰自己型凳,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,216評(píng)論 6 388
  • 文/花漫 我一把揭開(kāi)白布嘱函。 她就那樣靜靜地躺著啰脚,像睡著了一般。 火紅的嫁衣襯著肌膚如雪实夹。 梳的紋絲不亂的頭發(fā)上橄浓,一...
    開(kāi)封第一講書(shū)人閱讀 51,182評(píng)論 1 299
  • 那天,我揣著相機(jī)與錄音亮航,去河邊找鬼荸实。 笑死,一個(gè)胖子當(dāng)著我的面吹牛缴淋,可吹牛的內(nèi)容都是我干的准给。 我是一名探鬼主播,決...
    沈念sama閱讀 40,063評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼重抖,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼露氮!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起钟沛,我...
    開(kāi)封第一講書(shū)人閱讀 38,917評(píng)論 0 274
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤畔规,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后恨统,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體叁扫,經(jīng)...
    沈念sama閱讀 45,329評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,543評(píng)論 2 332
  • 正文 我和宋清朗相戀三年畜埋,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了莫绣。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,722評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡悠鞍,死狀恐怖对室,靈堂內(nèi)的尸體忽然破棺而出咖祭,到底是詐尸還是另有隱情心肪,我是刑警寧澤,帶...
    沈念sama閱讀 35,425評(píng)論 5 343
  • 正文 年R本政府宣布,位于F島的核電站锅减,受9級(jí)特大地震影響伐坏,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜每瞒,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,019評(píng)論 3 326
  • 文/蒙蒙 一代芜、第九天 我趴在偏房一處隱蔽的房頂上張望浓利。 院中可真熱鬧贷掖,春花似錦、人聲如沸苹威。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,671評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)腹暖。三九已至,卻和暖如春糕殉,著一層夾襖步出監(jiān)牢的瞬間殖告,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,825評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留筑煮,地道東北人真仲。 一個(gè)月前我還...
    沈念sama閱讀 47,729評(píng)論 2 368
  • 正文 我出身青樓秸应,卻偏偏與公主長(zhǎng)得像软啼,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子霉囚,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,614評(píng)論 2 353

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