javascript 异步请求封装成同步请求
此方法是异步请求封装成同步请求,加上token验证,环境试用微信小程序,可以修改文件中的ajax,进行封装自己的,比如用axios等
成功码采用标准的 200 到 300 和304 ,需要可以自行修改
同步任务接入之后,每个任务会进行token的验证,每个任务之间都是同步请求,包括token
/**
* 同步流请求
* token验证每个接口
* 柯里化添加同步任务
* resolve返回res,cb
* reject 返回res,cb
* 通过任务中断测试
* 通过成功失败回调函数测试
*
* 任务流 任务接入柯里化函数 currying(()=>{}) -->
* 开始执行 currying() 进入g函数循环同步执行异步方法 -->
* 执行异步方法 调用 rp 函数 封装 return new promise -->
* rp 函数 执行判断 token 状态 进行处理
*/ const regeneratorRuntime = require('./regenerator-runtime/runtime.js') //runtime类
const errorMessage = '服务繁忙,稍后再试' //公共提示
const successMessage = '完成' //公共提示 class SuperClass {
constructor() {
Object.assign(this, {})
//等待执行的接口
this.wait = []
} //Promise 请求接口
rp(opts, skipToken = false) {
const _t = this,
data = opts.data || {}
//小程序标识
data.source = opts.sourceFlag ? data.source : 1; //请求信息
console.group(`请求接口---${opts.url}`);
console.log('接口信息', {
opts,
skipToken
})
console.groupEnd(); return new Promise(function(resolve, reject) { opts.header = {
"Content-Type": "application/json;charset=utf-8",
//不需要走token的接口
Authorization: !!opts.skipToken ? '' : (wx.getStorageSync('token') || '')
} wx.request({
url: '域名' + opts.url,
data: data,
header: opts.header,
method: opts.method,
dataType: 'json',
responseType: 'text',
success: function(res) {
const {
data
} = res //成功 400为token失效 处理token失效问题 内层函数收集token失败码归并处理
if (data && (_t.successCode({
code: data.code
}) || data.code == 400)) {
resolve(data)
} else {
//其他不可预测为失败
reject(data)
}
},
fail: function(res) {
reject(res.data)
},
complete: function(res) {
//完成停止加载
wx.hideToast()
opts.complete && opts.complete(res.data)
}
})
})
} //为空参数抛异常
paramNoNull() {
throw new Error('Missing parameter')
} // g函数
async g() {
let _t = this,
//中断任务
isbreak = !1 for (let [i, v] of _t.wait.entries()) {
const r = await v().catch((res) => {
//收集 所有的错误 统一执行 提示 或者错误回调
const {
cb,
message
} = res //同步流中断
_t.clearWait()
isbreak = !0;
//没有回调函数执行错误提示
'cb' in res ? cb() : (wx.factory._toast(message || errorMessage))
}) //任务执行成功
if (!!r && r.constructor === Object && !!Object.keys(r).length) {
const {
res: {
code,
data,
message
},
cb
} = r //外层函数只处理成功状态
if (_t.successCode({
code
})) {
!!cb && cb() //同步流执行完成
if ((i + 1) == _t.wait.length) {
_t.clearWait()
wx.factory._toast(message || successMessage)
}
} else {
//同步流中断
_t.clearWait()
isbreak = !0 wx.factory._toast(message || errorMessage)
}
} if (!!isbreak) {
break
}
}
} //清空任务
clearWait() {
this.wait = []
} //柯里化
currying() {
const _t = this
return (function(arg) {
if (arg.length === 0) {
_t.g()
} else {
[].push.apply(_t.wait, arg);
}
})(arguments)
} //成功码
successCode({
code = 404
}) {
return (code >= 200 && code < 300) || code == 304
}
} //超类,实现多继承
const decorator = Sup => class extends Sup {
constructor(...args) {
super(...args)
} //获取token接口
greatetoken(opts) {
let codeOPts = JSON.parse(JSON.stringify(opts));
codeOPts.url = `/getToken`
codeOPts.method = `POST`
codeOPts.data = {
appIds: "xx",
userId: 5
}
return super.rp(codeOPts, true).then((res) => {
const {
code,
data
} = res
if (super.successCode({
code: cb.code
})) {
if (!!data && !!data.token) {
//全局token存储 方式为 storage
wx.setStorageSync('token', data.token)
} else {
wx.factory._toast('获取token失败')
}
} else {
wx.factory._toast('获取token失败')
}
}).catch(() => {
wx.factory._toast('获取token失败')
})
} /**
* 接口公共类
* 同步请求返回 Promise
* 请求 token 为前提,为空 过期重新请求
*/
async send(opts) {
const token = wx.getStorageSync('token'),
sendFun = (opts) => { //转化http请求 catch捕获promise的reject 展示接口级别错误 只做console打印 其他操作流入g函数(回调,提示)
return super.rp(opts).catch(res => { //此处显示rp reject 报错
console.group(`%c请求接口---${opts.url}---报错`, 'color:red;');
console.log(` --> 参数`, {
data: opts.data
});
console.log(` --> 返回值`, {
res
});
console.groupEnd(); //把错误信息传到 g 函数的catch里面
opts.fail && opts.fail(res)
})
},
successFun = async(opts) => {
const cb = await sendFun(opts) //把成功信息传到g函数里面
super.successCode({
code: cb.code
}) && opts.success && opts.success(cb)
} if (!opts.skipToken) { //需要token请求
if (!token) { //token为空 直接发起token请求
await this.greatetoken(opts)
await successFun(opts)
} else {
const cb = await sendFun(opts)
if (!!cb) { //400 token过期 只有在存在token的情况下才会有时效的情况 归并处理
if (cb.code == 400) {
await this.greatetoken(opts)
await successFun(opts)
} else {
//把成功信息传到g函数里面
super.successCode({
code: cb.code
}) && opts.success && opts.success(cb)
}
}
}
} else {
await successFun(opts)
}
} post(opts) {
opts.method = "POST";
this.send(opts)
} get(opts) {
opts.method = "GET";
this.send(opts);
}
} class Http extends decorator(SuperClass) { //子类
constructor() { //继承参数
super()
}
} export default new Http
引入方式,全局调用
import Http from './public/js/request'
wx.http = Http
调用执行,可用bind函数进行传参
success返回为成功的,code值流入任务进行code值判断区分 resolve
fail返回失败 reject
complete执行的为方法
Page({
onLoad(){
wx.http.currying(this.a.bind(this, {
a: 1
}))
//进行传参bind
wx.http.currying(this.b)
wx.http.currying()
},
a() {
const _t = this
wx.factory._toast('加载a信息中...', 6000)
return new Promise((resolve, reject) => {
wx.http.post({
url: 'a',
data: {
"params": {
id:33
}
},
success(res) {
resolve({
res,
cb()=>{}
})
},
fail(res) {
reject({
res,
cb()=>{}
})
}
})
})
},
b() {
const _t = this
wx.factory._toast('加载b信息中...', 6000)
return new Promise((resolve, reject) => {
wx.http.post({
url: 'b',
data: {
id:33
},
success(res) {
resolve({
res
})
},
fail(res) {
reject({
res
})
}
})
})
}
})
javascript 异步请求封装成同步请求的更多相关文章
- .NET 同步与异步之封装成Task(五)
本随笔续接:.NET 实现并行的几种方式(四) 前篇随笔已经介绍了几种可以实现并发的方式,其中异步方法.是最简便的方式.而 异步方式是基于 Task 和 async修饰符和await运算符实现的. 换 ...
- 如果将get请求转换成post请求
td><a href="emp/${emp.id}">Edit</a></td> <form action="" ...
- 利用meta标签将http请求换成https请求
最近网站升级为https之后,为了防止一些http文件没有修改而引起的问题,可以加一个meta标签: <meta http-equiv="Content-Security-Policy ...
- Jquey里的同步请求和异步请求
1.同步请求 发送了同步请求后 会一直等待 先执行 alert("result:" + d); temp = d; 在执行alert("this is last:& ...
- jQuery 异步和同步请求
在jQuery Ajax里面有一个async 参数 , 默认值 为true , 请求为异步请求 , false 为同步请求 .. 使用ajax加载数据返回页面并赋值,然后前端取出该值 这其中涉及到代码 ...
- iOS网络编程-ASIHTTPRequest框架同步请求-备用
在ASIHTTPRequest框架中与HTTP请求相关的类有:ASIHTTPRequest和ASIFormDataRequest,其中最常用的是ASIHTTPRequest,ASIFormDataRe ...
- iOS项目开发实战——使用同步请求获取网页源码
网络请求一般分为同步请求和异步请求,同步请求假设訪问时间过长,会造成界面卡死状态,用户体验不是非常好.可是请求速度较快的话,也能够考虑使用同步訪问.如今先来学习同步訪问. (1)在viewDidLoa ...
- ASIHttprequest-创建同步请求
ASIHttprequest-创建同步请求 当你发送一个同步请求后,该请求会在当前的应用主线程中运行并获取程序的控制权限,也就说你的程序处于锁定的状态,在这个期间,你进行不了任何的操作,直到该请求返回 ...
- 深入理解 JavaScript 异步——转载
本文章转载于深入理解 JavaScript 异步 前言 2014年秋季写完了<深入理解javascript原型和闭包系列>,已经帮助过很多人走出了 js 原型.作用域.闭包的困惑,至今仍能 ...
随机推荐
- Python 爬虫从入门到进阶之路(十四)
之前的文章我们已经可以根据 re 模块,Xpath 模块和 BeautifulSoup4 模块来爬取网站上我们想要的数据并且存储在本地,但是我们并没有对存储数据的格式有要求,本章我们就来看数据的存储格 ...
- Linux环境下虚拟环境virtualenv安装和使用(转)
virtualenv用于创建独立的Python环境,多个Python相互独立,互不影响,它能够: 1. 在没有权限的情况下安装新套件 2. 不同应用可以使用不同的套件版本 3. 套件升级不影响其他应用 ...
- MyBatis从入门到精通(十一):MyBatis高级结果映射之一对多映射
最近在读刘增辉老师所著的<MyBatis从入门到精通>一书,很有收获,于是将自己学习的过程以博客形式输出,如有错误,欢迎指正,如帮助到你,不胜荣幸! 本篇博客主要讲解MyBatis中如何使 ...
- [Haoi2016]放棋子 题解
4563: [Haoi2016]放棋子 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 440 Solved: 285[Submit][Status] ...
- [Microsoft][ODBC Driver 11 for SQL Server][SQL Server]列名 'user1' 无效
唉,还是自己对php执行sql语句运用不熟练.... 我的错误代码是这样的,(解决办法在最后) $re=sqlsrv_query($conn, "select * from visitor ...
- GPS常识-A版(详)
第一章 绪论 1.简述GPS系统的特点有哪些? GPS在测绘工程中应用的优点 P13 ●定位精度高 应用实践证明,相对静态定位1小时以上观测解,其平面位置:在300-1500m范围内,绝对误差小于1m ...
- 【CYH-01】小奔的国庆练习赛:赛后标程
前排鸣谢@找寻 大佬 emm-由于头一次举办公开赛所以--准备不是很充分,所以说题解也没有备好,在这里表示歉意. 欢迎大家来发布题解,在此我们可以提供AC代码,供大家参考. T1 解析:这一题可能栈溢 ...
- 个人永久性免费-Excel催化剂功能第45波-逻辑判断函数增强
自定义函数的最大的作用是可以按需定制,在Excel的原生函数不提供的场景时,传统方法需要使用大量的嵌套函数去实现,实在太累,今天Excel催化剂再次送上一波绝对十分常用的函数逻辑判断类函数给大家使用! ...
- Excel催化剂开源第29波-在Winform上使用富文本编辑器控件
富文本编辑器,一般都是BS架构专利一般,好像百度有一个开源的比较出名,但无奈这些都只能用在JS上,在BS网页端开发上使用.像Winform开发的VSTO,只能羡慕的份.和一般Winform上用的Ric ...
- 庖丁解牛Linux内核分析 0x00:《庖丁解牛》
庖丁解牛 吾生也有涯,而知也无涯 .以有涯随无涯,殆已!已而为知者,殆而已矣!为善无近名,为恶无近刑.缘督以为经,可以保身,可以全生,可以养亲,可以尽年. 庖丁为文惠君解牛,手之所触,肩之所倚,足之 ...