Taro-04-网络请求

一、Taro 为什么不直接用 fetch/axios,而是封装了一层跨端 API

同一套代码要运行在:

  • 微信小程序
  • H5
  • React Native
  • 支付宝小程序
  • 抖音小程序

所以 Taro 提供统一 API:

复制代码
Taro.request()

底层自动适配:

复制代码
// H5
window.fetch / XMLHttpRequest

// 微信小程序
wx.request

// 支付宝小程序
my.request
二、请求/响应拦截器(同一拦截器里写)

Taro 用的是洋葱模型:一个拦截器函数里,既可以在发请求前改参数,也可以在 chain.proceed() 返回的 Promise 上处理响应。

TypeScript 复制代码
Taro.addInterceptor(chain => {
  const { requestParams } = chain;
  const { url } = requestParams;

  // 请求发出前
  return chain.proceed({
    ...requestParams,
    header: {
      ...requestParams.header,
      Authorization: getToken(),
    },
  }).then(res => {
    // 响应回来后
    if (res.statusCode === 401) {
      // 跳转登录等
    }
    if (res.data?.code !== 0) {
      Taro.showToast({ title: res.data?.message ?? '请求失败', icon: 'none' });
      return Promise.reject(res);
    }
    return res;
  });
});
三、企业项目

通常封装:

1、网络请求基础地址
TypeScript 复制代码
const serverUrl = process.env.NODE_ENV === 'development' ? '/proxy' : '';
2、错误管理器
TypeScript 复制代码
const errorMessageManager = {
  messages: new Map<string, number>(),
  timeout: 3000,
  showErrorOnce(key: string, msg: string) {
    const now = Date.now();
    const lastShown = this.messages.get(key);
    if (!lastShown || now - lastShown > this.timeout) {
      Taro.showToast({ title: msg, icon: 'none', duration: 2000 });
      this.messages.set(key, now);
    }
  },
};
3、ts类型定义
TypeScript 复制代码
type RequestConfig = Taro.request.Option & {
  ignoreError?: boolean;
  params?: Record<string, unknown>;
};
// 定义一个新类型 HttpExtraConfig,它等于 RequestConfig 去掉 url、method、data 这三个属性之后剩下的部分。
// 2. Omit<T, K>
// TypeScript 内置工具类型,含义是:

// Omit<某个类型, 要排除的键>
// 第一个参数 T:源类型
// 第二个参数 K:要从 T 里排除的属性名
type HttpExtraConfig = Omit<RequestConfig, 'url' | 'method' | 'data'>;
4、封装请求前的处理(类似于axios的请求拦截)
TypeScript 复制代码
function patchRequestParams(params: Taro.request.Option): Taro.request.Option {
  return {
    ...params,
    url: `${serverUrl}${params.url}`,
    header: {
      ...params.header,
      Authorization: getToken(),
    },
  };
}
5、封装响应后的处理(响应拦截)
TypeScript 复制代码
function handleResponse(
  res: Taro.request.SuccessCallbackResult,
  params: Taro.request.Option & { ignoreError?: boolean },
) {
  const { statusCode } = res;
  const ignoreError = (params as any).ignoreError;
  if (statusCode === 401) {
    errorMessageManager.showErrorOnce('auth-401', '登录过期,请重新登录!');
    setTimeout(() => Taro.reLaunch({ url: '/pages/login/index' }), 500);
    throw res;
  }
  if (!ignoreError && (statusCode === 400 || statusCode >= 500)) {
    const msg = (res.data as { message?: string })?.message || '请求失败';
    errorMessageManager.showErrorOnce(`error-${statusCode}`, msg);
    throw res;
  }
  return res;
}
6、注册全局拦截器(需在 app 启动时调用一次)
TypeScript 复制代码
export function setupRequestInterceptors() {
  Taro.addInterceptor((chain) => {
    const params = patchRequestParams(chain.requestParams);
    return chain.proceed(params).then((res) => handleResponse(res, params));
  });
}
7、封装 get 请求
TypeScript 复制代码
export const get = (url: string, params = {}, config: HttpExtraConfig = {}) =>
  Taro.request({
    ...config,
    url,
    method: 'GET',
    data: params,
  }).then((res) => (config.responseType ? res : res.data));
8、封装 post 请求
TypeScript 复制代码
export const post = (
  url: string,
  data?: unknown,
  config: HttpExtraConfig = {},
) =>
  Taro.request({
    ...config,
    url,
    method: 'POST',
    data,
  }).then((res) => (config.responseType ? res : res.data));
9、封装 put 请求
TypeScript 复制代码
export const put = (
  url: string,
  data?: unknown,
  config: HttpExtraConfig = {},
) =>
  Taro.request({
    ...config,
    url,
    method: 'PUT',
    data,
  }).then((res) => res.data);
10、封装 del 请求
TypeScript 复制代码
export const del = (url: string, params = {}, config: HttpExtraConfig = {}) =>
  Taro.request({
    ...config,
    url,
    method: 'DELETE',
    data: params,
  }).then((res) => res.data);
相关推荐
丨我是张先生丨11 分钟前
日语单词 Web Page
前端·css·css3
禅思院2 小时前
AI对话前端从入门到崩溃:一个长对话引发的五层优化战争【引子】
前端·面试·架构
TrisighT2 小时前
Electron 鸿蒙 PC 上点外链唤醒应用,我试了 6 种写法只有 1 种能跑
前端·electron·harmonyos
2501_930707782 小时前
如何将HTML文件转换为纯文本(详细步骤指南)
前端·html
天才熊猫君3 小时前
配置与数据分离:一种可视化搭建的属性编辑方案
前端·javascript
林希_Rachel_傻希希3 小时前
web性能之相关路径——AI总结
前端·javascript·面试
不好听6133 小时前
从零搭建一个 RAG 语义搜索系统 —— DEMO的初始阶段
javascript·面试·llm
何时梦醒3 小时前
上下文工程(Context Engineering):AI 应用开发的新范式 —— 从理论到实战全解析
javascript
竹林8184 小时前
用 wagmi v2 踩坑两天,我终于搞懂了多链钱包切换在 DeFi 前端中的正确姿势
前端·javascript
用户2136610035724 小时前
Vue项目搜索功能与面包屑导航
前端·javascript