当前位置: 首页 > article >正文

Uniapp基于uni拦截器+Promise的请求函数封装

最近在学Uniapp,到封装请求的时候本来还想用axios,但是看到一些教学视频有更简单的方法,

基于uni的拦截器和Promise封装的请求函数

但是他们是用TS写的,还没学到TS,我就把JS写了,最终也是请求成功

// src/utils/http.js


import { useUserStore } from "../stores/modules/user"

// 请求基地址
const baseURL = 'https://pcapi-xiaotuxian-front-devtest.itheima.net'

// 拦截器配置
const httpInterceptor = {
  // 拦截前触发
  invoke(options) {
    // 1. 非 http 开头需拼接地址
    if (!options.url.startsWith('http')) {
      options.url = baseURL + options.url
    }
    // 2. 请求超时
    options.timeout = 10000
    // 3. 添加小程序端请求头标识
    options.header = {
      'source-client': 'miniapp',
      ...options.header,
    }
    // 4. 添加 token 请求头标识
    const userStore = useUserStore()
    const token = userStore.getToken() ? userStore.getToken() : ''
    if (token) {
      options.header.Authorization = token
    }
  },
}

// 拦截 request 请求
uni.addInterceptor('request', httpInterceptor)
// 拦截 uploadFile 文件上传
uni.addInterceptor('uploadFile', httpInterceptor)

export const http = (options) => {
    return new Promise((resolve, reject) => {
        uni.request({
            ...options,
            success(res) {
                if (res.statusCode >= 200 && res.statusCode < 300){
                    resolve(res.data)
                } else if ( res.statusCode === 401){
                    const userStore = useUserStore()
                    userStore.removeToken()
                    uni.navigateTo({ url: '/pages/login/login' })
                    reject(res)
                } else {
                    uni.showToast({
                        icon: 'none',
                        title: (res.data).msg || '请求错误',
                      })
                      reject(res)
                }
            },
            fail(err) {
                uni.showToast({
                    icon: 'none',
                    title: '请求失败',
                })
                reject(err)
            }
        })
    })
}

组件引入http方法进行请求

最后附上TS的写法,感觉其实差不多,只是用TS语法进行了一下泛型的判断

// src/utils/http.ts

// 请求基地址
const baseURL = 'https://pcapi-xiaotuxian-front-devtest.itheima.net'

// 拦截器配置
const httpInterceptor = {
  // 拦截前触发
  invoke(options: UniApp.RequestOptions) {
    // 1. 非 http 开头需拼接地址
    if (!options.url.startsWith('http')) {
      options.url = baseURL + options.url
    }
    // 2. 请求超时
    options.timeout = 10000
    // 3. 添加小程序端请求头标识
    options.header = {
      'source-client': 'miniapp',
      ...options.header,
    }
    // 4. 添加 token 请求头标识
    const memberStore = useMemberStore()
    const token = memberStore.profile?.token
    if (token) {
      options.header.Authorization = token
    }
  },
}

// 拦截 request 请求
uni.addInterceptor('request', httpInterceptor)
// 拦截 uploadFile 文件上传
uni.addInterceptor('uploadFile', httpInterceptor)

type Data<T> = {
  code: string
  msg: string
  result: T
}
// 2.2 添加类型,支持泛型
export const http = <T>(options: UniApp.RequestOptions) => {
  // 1. 返回 Promise 对象
  return new Promise<Data<T>>((resolve, reject) => {
    uni.request({
      ...options,
      // 响应成功
      success(res) {
        // 状态码 2xx,参考 axios 的设计
        if (res.statusCode >= 200 && res.statusCode < 300) {
          // 2.1 提取核心数据 res.data
          resolve(res.data as Data<T>)
        } else if (res.statusCode === 401) {
          // 401错误  -> 清理用户信息,跳转到登录页
          const memberStore = useMemberStore()
          memberStore.clearProfile()
          uni.navigateTo({ url: '/pages/login/login' })
          reject(res)
        } else {
          // 其他错误 -> 根据后端错误信息轻提示
          uni.showToast({
            icon: 'none',
            title: (res.data as Data<T>).msg || '请求错误',
          })
          reject(res)
        }
      },
      // 响应失败
      fail(err) {
        uni.showToast({
          icon: 'none',
          title: '网络错误,换个网络试试',
        })
        reject(err)
      },
    })
  })
}


http://www.kler.cn/a/299670.html

相关文章:

  • Linux: network: IPv6: ESP: UDP checksum error 一例
  • 你不知道的《回流与重绘》那些事儿
  • 双指针(8)_单调性_四数之和
  • 硅外延片行业分析:未来几年年复合增长率CAGR为5.3%
  • uniapp中scroll-view标签
  • 【Unity3D】如何用MMD4Mecanim插件将pmx格式模型转换为fbx格式模型
  • 分布式通信:多计算平台的任务分配
  • IP包头的总长度字段和UDP包头的长度字段之间的关系
  • 深入掌握Go语言中的正则表达式与字符串处理
  • 概率生成模型(Generative Models)和概率判别模型(Discriminative Models)
  • 什么情况?苹果、谷歌同时败诉
  • 面试—Redis
  • Linux驱动(六):Linux2.6驱动编写之平台设备总线
  • SLMi33x系列SLMi330HCG-DG具有主动保护功能 兼容光耦的单通道带保护功能的隔离驱动器
  • 【mysql】mysql之主从延迟复制测试场景
  • Vue3封装table表格右键菜单功能
  • QQueue调用dequeue闪退解决方法
  • CentOS 7.9安装GCC 7.3.0
  • 四款音频剪辑软件免费使用,你更pick哪一个?
  • 第三天旅游线路预览——从贾登峪到禾木风景区入口