vue中promise的使用及異步請求數(shù)據(jù)的方法
下面給大家介紹vue中promise的使用
promise是處理異步的利器,在之前的文章《ES6之promise》中,我詳細介紹了promise的使用, 在文章《js動畫實現(xiàn)&&回調(diào)地獄&&promise》中也提到了promise的then的鏈?zhǔn)秸{(diào)用, 這篇文章主要是介紹在實際項目中關(guān)于異步我遇到的一些問題以及解決方法,由此來加深對promise的進一步理解。
背景
進入商品頁,商品頁的左側(cè)是分類,右側(cè)是具體的商品,一旦進入商品頁,就把所有分類的商品請求出來,注意: 必須要按照順序。
實現(xiàn)思路
在商品頁, created 鉤子函數(shù)觸發(fā)獲取分類的http請求,請求到結(jié)果后,開始請求所有的具體商品并渲染。
遇到的問題
•由于請求商品分類是異步的, 怎么判斷異步請求完成, 也就是說請求具體商品的時機是什么時候。
•獲取到所有的商品必須發(fā)送請求,請求時異步的,怎么保證能夠按照順序獲取到。
解決問題 --- 問題一
針對問題一,最好的方式還是使用promise,大致實現(xiàn)如下:
getClassify: function () {
var that = this;
// 使用promise處理異步。
this.updateKinds().then(function () {
console.log("獲取分類結(jié)束!");
that.updateAllContent();
});
},
其中g(shù)etClassify是在created時就會調(diào)用的,而updateKinds是actions中的方法,我們先看看actions中是怎么寫的:
updateKinds ({commit, state}) {
return new Promise(function (resolve, reject) {
axios.get('/bbg/shop/get_classify', {
params: {
sid: 13729792
}
})
.then(function (response) {
if (response.data.code == 130) {
commit(UPDATE_KINDS, response.data.data)
console.log(response.data.data);
resolve()
}
}).catch(function (error) {
console.log(error);
});
});
即返回一個promise,當(dāng)請求到數(shù)據(jù),并且commit之后,我們就額可以resolve()了,這樣,就可以在then中執(zhí)行獲取所有內(nèi)容的方法了。
雖然實現(xiàn)起來比較簡單,但是這個思想更好。
解決問題 --- 問題二
在問題一中,我們看到resolve之后就可以調(diào)用updateAllContent() 了,那么這個應(yīng)該怎么寫呢?
首先可以確定的是: 因為需要請求的分類不只一個,所以要使用promise, 并且一定要返回一個promise,這樣才能繼續(xù)鏈?zhǔn)秸{(diào)用,其中一部分如下:
ar items = state.items;
function getItemPromise(id) {
return new Promise(function (resolve, reject) {
var content = {
"isSingle": 1,
"sbid": 13729792,
"catalog3": id,
"offset": 0,
"pageSize": 10
};
axios.post('/bbg/goods/get_goods_list_wechat', qs.stringify({"data": JSON.stringify(content)}))
.then(function (response) {
if (response.data.code == 626) {
for (let i = 0; i < response.data.data.length; i++) {
commit(UPDATE_ALL_CONTENT, response.data.data[i]);
}
resolve();
}
}).catch(function (error) {
console.log(error);
});
});
}
即調(diào)用這個函數(shù),傳入一個分類的id,然后就可以發(fā)送請求了,獲取到數(shù)據(jù)之后,就把數(shù)據(jù)插入到 內(nèi)容的數(shù)組中, 最后resolve()還告訴then可以執(zhí)行了。
注意: 如何更新一個數(shù)組呢?
[UPDATE_ALL_CONTENT] (state, item) {
state.contentItems = [...state.contentItems, Object.assign({}, item)];
},
這樣就相當(dāng)于push了。
上面的這個函數(shù)的意義在于封裝請求,那么對于請求多個時,如何做到呢?
我之前嘗試了下面兩種方法:
FIRST
// first method
var promise = getItemPromise(items[0].id)
for (let j = 1; j < items.length; j++) {
promise.then(function () {
return getItemPromise(items[j].id);
})
}
思路就是先請求第一個分類,然后循環(huán),實際上和下面的效果是一樣的:
var promise = getItemPromise(items[0].id);
promise.then(function () {
console.log("1", window.performance.now());
return getItemPromise(items[1].id);
});
promise.then(function () {
console.log("2", window.performance.now());
return getItemPromise(items[2].id);
});
promise.then(function () {
console.log("3", window.performance.now());
return getItemPromise(items[3].id);
});
promise.then(function () {
console.log("4", window.performance.now());
return getItemPromise(items[4].id);
});
promise.then(function () {
console.log("5", window.performance.now());
return getItemPromise(items[5].id);
});
promise.then(function () {
console.log("6", window.performance.now());
return getItemPromise(items[6].id);
});
問題: 通過這樣的方法最終請求的數(shù)據(jù)是可以請求到的,但是順序并沒有按照我們預(yù)想的思路來執(zhí)行,因為這樣的執(zhí)行方式會在getItemPromise執(zhí)行之后就立即同時執(zhí)行后面幾個then,所以最終得到的順序是不能確定的。
方法二:
// second method
var somePromise = getItemPromise(items[0].id);
for (let k = 1; k < items.length; k++) {
somePromise = somePromise.then(function () {
return getItemPromise(items[k].id);
});
}
這種方法的結(jié)構(gòu)類似于下面這樣:
getItemPromise(items[0].id)
.then(function () {
console.log("1", window.performance.now());
return getItemPromise(items[1].id);
})
.then(function () {
console.log("2", window.performance.now());
return getItemPromise(items[2].id);
})
.then(function () {
console.log("3", window.performance.now());
return getItemPromise(items[3].id);
})
.then(function () {
console.log("4", window.performance.now());
return getItemPromise(items[4].id);
})
.then(function () {
console.log("5", window.performance.now());
return getItemPromise(items[5].id);
})
.then(function () {
console.log("6", window.performance.now());
return getItemPromise(items[6].id);
})
.then(function () {
console.log("7", window.performance.now());
return getItemPromise(items[7].id);
})
.then(function () {
return getItemPromise(items[8].id);
})
.then(function () {
return getItemPromise(items[9].id);
})
.then(function () {
return getItemPromise(items[10].id);
})
.then(function () {
return getItemPromise(items[11].id);
})
這樣請求得到的順序就是相同的了。 但是通過for循環(huán),不論分類有多少,我們都可以請求到。
也就是說,通過鏈?zhǔn)秸{(diào)用的方式,即.then().then()這樣才會在一個異步執(zhí)行完之后執(zhí)行下一個,值得注意。
下面看下vue 中promise 異步請求數(shù)據(jù)的方法
export function getTypes(type) {
return listDictItems({ code: type }).then((res) => {
if (res.code == 200) {
let list = res.body;
// console.log('list',list);
return list;
}
})
};
組件中:
getTypes('EP_TYPE').then((data) => {console.log('data',data)});//成功
總結(jié)
以上所述是小編給大家介紹的vue中promise的使用及異步請求數(shù)據(jù)的方法,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!
相關(guān)文章
用element的upload組件實現(xiàn)多圖片上傳和壓縮的示例代碼
這篇文章主要介紹了用element的upload組件實現(xiàn)多圖片上傳和壓縮的示例代碼,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-02-02
django使用channels2.x實現(xiàn)實時通訊
這篇文章主要介紹了django使用channels2.x實現(xiàn)實時通訊,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-11-11
詳解使用mpvue開發(fā)github小程序總結(jié)
這篇文章主要介紹了詳解使用mpvue開發(fā)github小程序總結(jié),小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-07-07
Vue3中動態(tài)修改樣式與級聯(lián)樣式優(yōu)先順序圖文詳解
在項目中,我們時常會遇到動態(tài)的去綁定操作切換不同的CSS樣式,下面這篇文章主要給大家介紹了關(guān)于Vue3中動態(tài)修改樣式與級聯(lián)樣式優(yōu)先順序的相關(guān)資料,文中通過實例代碼介紹的非常詳細,需要的朋友可以參考下2023-04-04

