当前位置:   article > 正文

HarmonyOS 鸿蒙应用开发 (七、HTTP网络组件 axios 介绍及封装使用)_鸿蒙 axios

鸿蒙 axios

在HarmonyOS应用开发中,通过HTTP访问网络,可以使用官方提供的@ohos.net.http模块。但是官方提供的直接使用不太好使用,需要封装下才好。推荐使用前端开发中流行的axios网络客户端库,如果是前端开发者,用 axios也会更加顺手。

目录

axios介绍

在HarmonyOS也能用Axios?

axios网络请求库的使用

下载安装

开通权限

简单使用

axios模块封装及使用

客户端封装

封装后使用

官方@ohos/net.http 介绍

官方简易封装

官方http模块封装使用

写在最后

其他资源

axios介绍

Axios 是一个著名的基于 JavaScript 的开源库,用于浏览器和 Node.js 等环境中发送 HTTP 请求。它支持 Promise API,并且可以处理 XMLHttpRequests 和 Fetch API 背后的复杂性,为开发者提供了一种简洁易用的方式来实现 AJAX(Asynchronous JavaScript and XML)请求。

最早浏览器页面在向服务器请求数据,返回的是整个页面的数据,页面会强制刷新一下,这对于用户来讲并不是很友好。我们只想修改页面的部分数据,但是从服务器端返回的却是整个页面。于是出现一种新的技术,异步网络请求Ajax(Asynchronous JavaScript and XML),它能与后台服务器进行少量数据交换,使网页实现异步局部更新。

由于浏览器中原生的XMLHttpRequest API较难使用,于是又有了更多用于实现ajax的javascript框架出现,比如我们熟悉的jQuery、Dojo、YUI等等。而如今一个叫axios的轻量框架逐步脱颖而出,它本质上也是对原生XHR的封装,只不过它是Promise的实现版本,符合最新的ES规范。

主要特性:

  1. 跨平台支持: Axios 可以在浏览器端通过 XMLHttpRequests 发送请求,在 Node.js 中则使用 http/https 模块发送请求。
  2. Promise API: Axios 的所有网络请求方法都返回 Promise 对象,使得异步编程更加简洁和易于处理。
  3. 拦截请求与响应: 提供了请求和响应的全局和实例级别的拦截器,可以在请求发送前或响应返回后进行预处理、错误处理或数据转换等操作。
  4. 取消请求: 支持主动取消已经发出但还未完成的HTTP请求。
  5. 自动转换JSON数据: Axios 自动将来自服务器的 JSON 数据转换为 JavaScript 对象,并且对于POST、PUT等请求体中的JSON数据也会自动序列化成字符串发送。
  6. 配置灵活性: Axios 允许自定义请求头、URL参数、超时时间等多种配置项,适用于不同场景下的API调用需求。
  7. 请求方法多样: 支持所有标准的HTTP方法(GET、POST、PUT、DELETE等),以及对PATCH等非标准方法的良好支持。
  8. 上传下载进度监控: Axios 还支持监听文件上传和下载的进度事件。

在HarmonyOS也能用Axios?

在HarmonyOS中,官方提供了@ohos/net.http 模块进行网络访问。它是官方提供的基础HTTP数据请求能力库,直接提供了对HTTP协议的底层支持,开发者可以通过这个模块发送GET、POST等HTTP请求,并处理响应结果。由于它是系统级别的API,其优点在于性能和兼容性得到保证,适用于基本的HTTP通信需求。

虽然官方提供了@ohos/net.http 模块进行网络访问,但是Axios库可以看作是一种功能更强大和易用的封装,且接口使用上更符合前端开发者的惯用习惯。Axios库 以其强大的功能性和易用性成为现代JavaScript应用中非常流行的HTTP客户端库。

直接使用原始的axios库肯定是不行,在HarmonyOS中的Axios库,模块名字是@ohos/axios。

@ohos/axios第三方库是基于axios库进行适配,使其可以运行在OpenHarmony中的一个发送网络请求库,并且本库沿用axios库现有用法和特性,使其更加适合于鸿蒙项目的开发。

@ohos/axios 模块可以理解为是对官方HTTP API的一个封装或者扩展,它提供了一种更高级别的抽象和便利性,可能包含了更多的功能特性,比如自动转换数据格式、错误处理、拦截器机制以及对于Promise的良好支持等,这些都是为了简化开发流程,提高开发效率。在实际开发应用时,如果需要更丰富和灵活的网络请求管理功能,通常推荐使用 @ohos/axios 这样的封装库。

通过对@ohos/axis源码的查看,发现也确实是使用ohos.net.http模块,对原库v1.3.4版本进行适配,使其可以应用在HarmonyOS上,并沿用其现有用法和特性。

  • http 请求
  • Promise API
  • request 和 response 拦截器
  • 转换 request 和 response 的 data 数据
  • 自动转换 JSON data 数据

ohos/axios 模块源码:

OpenHarmony-SIG/ohos_axios

axios网络请求库的使用

接口列表

接口参数功能
axios(config)config:请求配置发送请求
axios.create(config)config:请求配置创建实例
axios.request(config)config:请求配置发送请求
axios.get(url[, config])url:请求地址
config:请求配置
发送get请求
axios.delete(url[, config])url:请求地址
config:请求配置
发送delete请求
axios.post(url[, data[, config]])url:请求地址
data:发送请求体数据
config:请求配置
发送post请求
axios.put(url[, data[, config]])url:请求地址
data:发送请求体数据
config:请求配置
发送put请求

属性列表

属性描述
axios.defaults['xxx']默认设置 。值为请求配置 config 中的配置项
例如 axios.defaults.headers 获取头部信息
axios.interceptors拦截器。参考 拦截器 的使用

下载安装

  • 方式一:在Terminal窗口中,执行如下命令安装三方包,DevEco Studio会自动在工程的oh-package.json5中自动添加三方包依赖。
ohpm install @ohos/axis

如果提示无ohpm命令,则是环境变量没有配置。 点击配置ohpm命令查看详细配置。

  • 方式二:在工程的oh-package.json5中设置三方包依赖,配置示例如下:
"dependencies": { "@ohos/axis": "^2.1.0"}

开通权限

 需要配置 ohos.permission.INTERNET权限。在工程目录entry\src\main中找到module.json5文件,配置网络请求权限。

  1. {
  2. "module": {
  3. "name": "entry",
  4. "type": "entry",
  5. "description": "$string:module_desc",
  6. "mainElement": "EntryAbility",
  7. "deviceTypes": [
  8. "phone"
  9. ],
  10. "requestPermissions": [
  11. {
  12. "name": "ohos.permission.INTERNET"
  13. }
  14. ]
  15. }
  16. }

简单使用

  1. import axios from '@ohos/axios'
  2. //创建axios的实例
  3. const instance = axios.create({
  4. baseURL: "http://xx.xx.xx.xx", //基路径,要看API帮助文档的特征来确定基路径
  5. timeout: 5000, //请求超时的时间
  6. headers: {
  7. "Content-Type": "application/json"
  8. }
  9. })
  10. //响应拦截器,通过响应拦截器进一步对返回的数据做处理
  11. instance.interceptors.response.use((response) => {
  12. //只返回接口有数据的结果
  13. if (200 === response.status) {
  14. return response.data; //接口返回的数据
  15. }
  16. return Promise.reject(response); //表示请求有错,交给catch来处理结构
  17. }, err => {
  18. return Promise.reject(err)
  19. })
  20. /**
  21. * get请求
  22. * @param params = {} 查询参数
  23. * @returns
  24. */
  25. export function httpGet(url:string, params = {}) {
  26. return instance.get<any>(url, {
  27. params
  28. })
  29. }
  30. /**
  31. * post请求
  32. * @param data = {} 请求体数据
  33. * @returns
  34. */
  35. export function httpPost(url:string, data = {}) {
  36. return instance.post<any>(url, {
  37. data
  38. })
  39. }

axios模块封装及使用

axios模块封装:

  1. //AxiosHttp.ets
  2. import axios, {
  3. AxiosInstance,
  4. AxiosRequestConfig,
  5. AxiosRequestHeaders,
  6. AxiosResponse,
  7. InternalAxiosRequestConfig
  8. } from "@ohos/axios";
  9. import { LogUtils } from '../utils/LogUtils';
  10. /**
  11. * 定义接口响应包装类
  12. */
  13. export interface BaseResponse {
  14. //wanAndroid-API响应体
  15. errorCode: number
  16. errorMsg: string
  17. //拓展xxx-API响应体
  18. }
  19. /**
  20. * 接口实现类包装,例如有其他业务可以再次继承实现xxxResponse
  21. */
  22. export interface ApiResponse<T = any> extends BaseResponse {
  23. //wanAndroid-API响应体
  24. data: T | any;
  25. //拓展xxx-API响应体
  26. }
  27. /**
  28. * 封装后,不支持传入拦截器
  29. * 需要自己定义接口继承 AxiosRequestConfig类型
  30. * 从而支持传入拦截器,但拦截器选项应为可选属性
  31. * 之后请求实例传入的options为继承了AxiosRequestConfig的自定义类型
  32. */
  33. interface InterceptorHooks {
  34. requestInterceptor?: (config: HttpRequestConfig) => Promise<HttpRequestConfig>;
  35. requestInterceptorCatch?: (error: any) => any;
  36. responseInterceptor?: (response: AxiosResponse) => AxiosResponse | Promise<AxiosResponse>;
  37. responseInterceptorCatch?: (error: any) => any;
  38. }
  39. // @ts-ignore
  40. interface HttpRequestConfig extends InternalAxiosRequestConfig {
  41. showLoading?: boolean; //是否展示请求loading
  42. checkResultCode?: boolean; //是否检验响应结果码
  43. checkLoginState?: boolean //校验用户登陆状态
  44. needJumpToLogin?: boolean //是否需要跳转到登陆页面
  45. interceptorHooks?: InterceptorHooks;
  46. headers?: AxiosRequestHeaders
  47. }
  48. /**
  49. * 网络请求构造
  50. * 基于axios框架实现
  51. */
  52. class AxiosHttpRequest {
  53. config: HttpRequestConfig;
  54. interceptorHooks?: InterceptorHooks;
  55. instance: AxiosInstance;
  56. constructor(options: HttpRequestConfig) {
  57. this.config = options;
  58. this.interceptorHooks = options.interceptorHooks;
  59. this.instance = axios.create(options);
  60. this.setupInterceptor()
  61. }
  62. setupInterceptor(): void {
  63. this.instance.interceptors.request.use(
  64. //这里主要是高版本的axios中设置拦截器的时候里面的Config属性必须是InternalAxiosRequestConfig,但是InternalAxiosRequestConfig里面的headers是必传,所以在实现的子类我设置成非必传会报错,加了个忽略注解
  65. // @ts-ignore
  66. this.interceptorHooks?.requestInterceptor,
  67. this.interceptorHooks?.requestInterceptorCatch,
  68. );
  69. this.instance.interceptors.response.use(
  70. this.interceptorHooks?.responseInterceptor,
  71. this.interceptorHooks?.responseInterceptorCatch,
  72. );
  73. }
  74. // 类型参数的作用,T决定AxiosResponse实例中data的类型
  75. request<T = any>(config: HttpRequestConfig): Promise<T> {
  76. return new Promise<T>((resolve, reject) => {
  77. this.instance
  78. .request<any, T>(config)
  79. .then(res => {
  80. resolve(res);
  81. })
  82. .catch((err) => {
  83. LogUtils.error("网络请求Request异常:", err.message)
  84. errorHandler(err)
  85. if (err) {
  86. reject(err);
  87. }
  88. });
  89. });
  90. }
  91. get<T = any>(config: HttpRequestConfig): Promise<T> {
  92. return this.request({ ...config, method: 'GET' });
  93. }
  94. post<T = any>(config: HttpRequestConfig): Promise<T> {
  95. return this.request({ ...config, method: 'POST' });
  96. }
  97. delete<T = any>(config: HttpRequestConfig): Promise<T> {
  98. return this.request({ ...config, method: 'DELETE' });
  99. }
  100. patch<T = any>(config: HttpRequestConfig): Promise<T> {
  101. return this.request({ ...config, method: 'PATCH' });
  102. }
  103. }
  104. export function errorHandler(error: any) {
  105. if (error instanceof AxiosError) {
  106. showToast(error.message)
  107. } else if (error != undefined && error.response != undefined && error.response.status) {
  108. switch (error.response.status) {
  109. // 401: 未登录
  110. // 未登录则跳转登录页面,并携带当前页面的路径
  111. // 在登录成功后返回当前页面,这一步需要在登录页操作。
  112. case 401:
  113. break;
  114. // 403 token过期
  115. // 登录过期对用户进行提示
  116. // 清除本地token和清空vuex中token对象
  117. // 跳转登录页面
  118. case 403:
  119. showToast("登录过期,请重新登录")
  120. // 清除token
  121. // localStorage.removeItem('token');
  122. break;
  123. // 404请求不存在
  124. case 404:
  125. showToast("网络请求不存在")
  126. break;
  127. // 其他错误,直接抛出错误提示
  128. default:
  129. showToast(error.response.data.message)
  130. }
  131. }
  132. }
  133. export default AxiosHttpRequest

客户端封装

  1. //AxiosRequest.ets
  2. import {AxiosHttpRequest,errorHandler} from './AxiosHttp'
  3. import { AxiosError, AxiosRequestHeaders } from '@ohos/axios';
  4. import { LogUtils } from '../utils/LogUtils';
  5. import showToast from '../utils/ToastUtils';
  6. import { hideLoadingDialog, showLoadingDialog } from '../utils/DialogUtils';
  7. import { StorageUtils } from '../utils/StorageUtils';
  8. import { StorageKeys } from '../constants/StorageKeys';
  9. import { JsonUtils } from '../utils/JsonUtils';
  10. import { Router } from '../route/Router';
  11. import { RoutePath } from '../route/RoutePath';
  12. /**
  13. * axios请求客户端创建
  14. */
  15. const axiosClient = new AxiosHttpRequest({
  16. baseURL: "/api",
  17. timeout: 10 * 1000,
  18. checkResultCode: false,
  19. headers: {
  20. 'Content-Type': 'application/json'
  21. } as AxiosRequestHeaders,
  22. interceptorHooks: {
  23. requestInterceptor: async (config) => {
  24. // 在发送请求之前做一些处理,例如打印请求信息
  25. LogUtils.debug('网络请求Request 请求方法:', `${config.method}`);
  26. LogUtils.debug('网络请求Request 请求链接:', `${config.url}`);
  27. LogUtils.debug('网络请求Request Params:', `\n${JsonUtils.stringify(config.params)}`);
  28. LogUtils.debug('网络请求Request Data:', `${JsonUtils.stringify(config.data)}`);
  29. axiosClient.config.showLoading = config.showLoading
  30. if (config.showLoading) {
  31. showLoadingDialog("加载中...")
  32. }
  33. if (config.checkLoginState) {
  34. let hasLogin = await StorageUtils.get(StorageKeys.USER_LOGIN, false)
  35. LogUtils.debug('网络请求Request 登录状态校验>>>', `${hasLogin.toString()}`);
  36. if (hasLogin) {
  37. return config
  38. } else {
  39. if (config.needJumpToLogin) {
  40. Router.push(RoutePath.TestPage)
  41. }
  42. throw new AxiosError("请登录")
  43. }
  44. }
  45. return config;
  46. },
  47. requestInterceptorCatch: (err) => {
  48. LogUtils.error("网络请求RequestError", err.toString())
  49. if (axiosClient.config.showLoading) {
  50. hideLoadingDialog()
  51. }
  52. return err;
  53. },
  54. responseInterceptor: (response) => {
  55. //优先执行自己的请求响应拦截器,在执行通用请求request的
  56. if (axiosClient.config.showLoading) {
  57. hideLoadingDialog()
  58. }
  59. LogUtils.debug('网络请求响应Response:', `\n${JsonUtils.stringify(response.data)}`);
  60. if (response.status === 200) {
  61. // @ts-ignore
  62. const checkResultCode = response.config.checkResultCode
  63. if (checkResultCode && response.data.errorCode != 0) {
  64. showToast(response.data.errorMsg)
  65. return Promise.reject(response)
  66. }
  67. return Promise.resolve(response.data);
  68. } else {
  69. return Promise.reject(response);
  70. }
  71. },
  72. responseInterceptorCatch: (error) => {
  73. if (axiosClient.config.showLoading) {
  74. hideLoadingDialog()
  75. }
  76. LogUtils.error("网络请求响应异常", error.toString())
  77. errorHandler(error);
  78. return Promise.reject(error);
  79. },
  80. }
  81. });
  82. export default axiosClient;

封装后使用

经过封装后,使用变得很简单了。示例如下:

  1. import axiosClient from './AxiosRequest'
  2. let baseUrl = "https://www.wanandroid.com/"
  3. //返回数据结构定义
  4. interface HomeModelIssueList {
  5. releaseTime: number;
  6. type: string;
  7. date: number;
  8. publishTime: number;
  9. count: number;
  10. }
  11. interface HomeModel {
  12. issueList: HomeModelIssueList[];
  13. itemList: HomeModelIssueList[];
  14. nextPageUrl: string;
  15. nextPublishTime: number;
  16. newestIssueType: string;
  17. }
  18. interface BannerDataModelData {
  19. desc: string;
  20. id: number;
  21. imagePath: string;
  22. isVisible: number;
  23. order: number;
  24. title: string;
  25. type: number;
  26. url: string;
  27. }
  28. /**
  29. * 请求首页数据-axios客户端请求
  30. * @param date
  31. * @returns
  32. */
  33. export function getHomeListAxios(date: string = "") {
  34. return axiosClient.get<HomeModel>({
  35. url: baseUrl + "api/v2/feed",
  36. params: { "date": date },
  37. showLoading: true
  38. // headers: { "Accept": "application/json" } as AxiosRequestHeaders
  39. })
  40. }
  41. /**
  42. * 获取分类详情接口
  43. * @param id
  44. * @param start
  45. */
  46. export function getCategoryDetailList(id: number, start: number) {
  47. return axiosClient.get<HomeModel>(
  48. {
  49. url: baseUrl + "api/v4/categories/videoList",
  50. params: {
  51. "id": id,
  52. "start": start,
  53. "udid": CommonConstants.UUID,
  54. deviceModel: CommonConstants.DEVICE_NUM
  55. }
  56. }
  57. )
  58. }
  59. /**
  60. * 获取wanAndroid首页Banner数据,测试校验API data:T泛型数据
  61. * @returns
  62. */
  63. export function getWanAndroidBanner() {
  64. return axiosClient.get<ApiResponse<BannerDataModelData[]>>(
  65. {
  66. url: wanAndroidUrl + "/banner/json",
  67. checkResultCode: true,
  68. showLoading: true
  69. }
  70. )
  71. }

官方@ohos/net.http 介绍

在HarmonyOS(OpenHarmony)中,@ohos/net.http 是官方提供的一个用于进行HTTP通信的基础模块。开发者可以利用这个模块发送和接收HTTP请求与响应,实现应用程序与服务器之间的数据交互。

文档中心--HTTP数据请求

官方简易封装

@ohos/net.http模块直接使用起来不太友好,简易封装如下:

  1. //http.ets
  2. /**
  3. * 定义接口响应包装类
  4. */
  5. import http from '@ohos.net.http';
  6. export interface BaseResponse {
  7. //wanAndroid-API响应体
  8. errorCode: number
  9. errorMsg: string
  10. //拓展xxx-API响应体
  11. }
  12. /**
  13. * 接口实现类包装,例如有其他业务可以再次继承实现xxxResponse
  14. */
  15. export interface ApiResponse<T = any> extends BaseResponse {
  16. //wanAndroid-API响应体
  17. data: T | any;
  18. //拓展xxx-API响应体
  19. }
  20. interface HttpRequestConfig extends http.HttpRequestOptions {
  21. showLoading?: boolean; //是否展示请求loading
  22. checkResultCode?: boolean; //是否检验响应结果码
  23. checkLoginState?: boolean //校验用户登陆状态
  24. needJumpToLogin?: boolean //是否需要跳转到登陆页面
  25. url?: string, //请求网络链接
  26. }
  27. /**
  28. * 网络请求构造器
  29. * 基于鸿蒙默认的http框架实现
  30. */
  31. class HttpBuilder {
  32. httpClient: http.HttpRequest
  33. config: HttpRequestConfig
  34. constructor(options: HttpRequestConfig) {
  35. this.httpClient = http.createHttp()
  36. this.config = options
  37. this.setupInterceptor()
  38. }
  39. /**
  40. * 配置属性拦截器
  41. */
  42. setupInterceptor() {
  43. }
  44. request<T = any>(config: HttpRequestConfig): Promise<T> {
  45. return new Promise<T>((resolve, reject) => {
  46. this.httpClient.request(
  47. config.url,
  48. config,
  49. (error, data) => {
  50. if (!error) {
  51. resolve(data.result as T);
  52. } else {
  53. reject(error)
  54. }
  55. // 当该请求使用完毕时,调用destroy方法主动销毁x
  56. this.httpClient.destroy()
  57. }
  58. )
  59. })
  60. }
  61. get<T = any>(config: HttpRequestConfig): Promise<T> {
  62. return this.request({ ...config, method: http.RequestMethod.GET })
  63. }
  64. post<T = any>(config: HttpRequestConfig): Promise<T> {
  65. return this.request({ ...config, method: http.RequestMethod.POST })
  66. }
  67. }
  68. export default HttpBuilder

官方http模块封装使用

  1. import http from '@ohos.net.http';
  2. import showToast from '../utils/ToastUtils';
  3. import HttpBuilder from './http';
  4. //接口发送超时
  5. const READ_TIMEOUT = 100000
  6. //接口读取超时
  7. const CONNECT_TIMEOUT = 100000
  8. let baseUrl = "https://www.wanandroid.com/"
  9. const httpClient = new HttpBuilder({
  10. readTimeout: READ_TIMEOUT,
  11. connectTimeout: CONNECT_TIMEOUT
  12. })
  13. //返回数据结构定义
  14. interface HomeModelIssueList {
  15. releaseTime: number;
  16. type: string;
  17. date: number;
  18. publishTime: number;
  19. count: number;
  20. }
  21. interface HomeModel {
  22. issueList: HomeModelIssueList[];
  23. itemList: HomeModelIssueList[];
  24. nextPageUrl: string;
  25. nextPublishTime: number;
  26. newestIssueType: string;
  27. }
  28. /**
  29. * 请求数据--系统http请求
  30. * @param date
  31. * @returns
  32. */
  33. export function getHomeList(date: string = "") {
  34. return httpClient.get<HomeModel>({
  35. url: baseUrl + "api/v2/feed",
  36. extraData: { "date": date }
  37. })
  38. }

写在最后

  • 如果你觉得这篇内容对你还蛮有帮助,我想邀请你帮我三个小忙:
  • 点赞,转发,有你们的 『点赞和评论』,才是我创造的动力。
  • 关注博主,同时可以期待后续文章ing
    声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/凡人多烦事01/article/detail/515246
推荐阅读
相关标签