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 原型.作用域.闭包的困惑,至今仍能 ...
随机推荐
- 吐槽:那些Java设计中不得不说的槽点
1. 求长度各有千秋 你是否曾经在面试的时候,经常被问到:数组有没有 length() 方法?字符串有没有 length() 方法? 集合有没有 length() 方法? 面对这个问题,那么不得不吐槽 ...
- extern和static区别
1. 声明和定义 当定义一个变量的时候,就包含了对该变量声明的过程,同时在内存张申请了一块内存空间.如果在多个文件中使用相同的变量,为了避免重复定义,就必须将声明和定义分离开来.定义是创建与名字关 ...
- git rebase VS git merge? 更优雅的 git 合并方式值得拥有
写在前面 如果你不能很好的应用 Git,那么这里为你提供一个非常棒的 Git 在线练习工具 Git Online ,你可以更直观的看到你所使用的命令会产生什么效果 另外,你在使用 Git 合并分支时只 ...
- JVM监控jconsole
1. 描述 程序在开发过程中,有可能会发生CPU飙高.内存溢出等问题或系统在后期调优阶段,不可避免的要监控JVM情况,JDK自带的Jconsole监控工具,结合Tomcat使用非常方便,占用内存小 ...
- Centos6.5安装Redis3.2.8
1 - Redis安装 redis安装 在网上一搜一大把,但是还是在这里想要能够统一吧,所以这个安装步骤是在Centos6.5 Minimal 上安装redis3.4.8,本次安装是在root 用户下 ...
- 重新认识 async/await 语法糖
提起.Net中的 async/await,相信很多.neter 第一反应都会是异步编程,其本质是语法糖,但继续追查下去,既然是语法糖,那么经过编译之后,真正的代码是什么样的,如何执行的?带着这些疑问, ...
- form 利用BeginCollectionItem提交集合List<T>数据 以及提交的集合中含有集合的数据类型 如List<List<T>> 数据的解决方案
例子: public class IssArgs { public List<IssTabArgs> Tabs { get; set; } } public class IssTabArg ...
- Python 爬虫:豆瓣电影Top250,包括电影导演、类型、年份、主演
结果输出到文本文件中. import codecs import requests from bs4 import BeautifulSoup headers={'User-Agent': 'Mozi ...
- SQLite的一些体会
SQLite遵循sql语法,所以如果接触过数据库,使用它进行增删改查几乎没障碍.在.net中,它与Mysql.sql server的类也相似,比如连接类名字是SQLiteConnection,不过它S ...
- 个人永久性免费-Excel催化剂功能第85波-灵活便捷的批量发送短信功能(使用腾讯云接口)
微信时代的今天,短信一样不可缺席,大系统都有集成短信接口.若只是临时用一下,若能够直接在Excel上加工好内容就可以直接发送,这些假设在此篇批量群发短信功能中都为大家带来完美答案. 业务场景 不多说, ...