react native 基于 fetch 简单封装自己的网络请求

导语

在项目开发中,常用的工具类,我们都会专门封装起来进行统一处理,方便以后统一调用、修改。网络请求类算是用到最多的公共类之一了,本文就简单来对 react native 的 fetch 进行简单的封装,并不会讲述太多细节,你可以使用你喜欢的搜索引擎去搜索 fetch api 关键字以了解更多信息。

参考

  • RN 中文网的简介

fetch api

  • method: 默认为 GET 请求
  • headers:请求头 用来添加一些设备信息和用户信息
  • body:请求体 请求参数

基本功能

  • 支持 GET POST 两种请求 默认为 POST 请求
  • 支持自定义请求头、传入请求参数
  • 支持自定义请求超时时间
  • 支持请求中断(防止重复发送同一个请求)
  • 便于扩展

接下来先看一下我们的网络请求调用方式
第一步:创建,继承封装好的 fetch 类 重写一下 requestUrl 方法

class XXXRequest extends BaseRequest{
  requestUrl() {
     return 'api/xxx.xxx';
}

第二步:使用

//创建一个请求对象,参数为(请求体,请求方法)
XXXRequest request = new XXXRequest();
// timeout 超时时间 单位为 s
// 显示网络请求加载动画
request.timeout(10).showLoadingView().start(
     // 成功的回调
     (success)=>{
     },
     // 失败的的回调
    (error)=>{
   });
// 中断本次请求
request&&request.setCancled(true);

接下来直接上代码 我会在代码里面进行详细的讲解

/**
* Created by jzz on 2017/8/25.
*/

import  {
   Platform,
   BackAndroid,
   Alert,
}from 'react-native'

// 是否开启 debug 模式
const DEBUG_MODE = true;
// 默认超时时间 20s
const TIMEOUT = 20;
export default class BaseRequest {
   // 构造
   constructor(body, method, mode) {
       // 是否终止请求 默认false
       this.isCancled = false;
       // 请求失败是否打印后台message 默认false
       this.isShowMessage = false;
       if (body == null) {
           body = {};
       }
       // 默认的一些参数比如 版本号 token 
       Object.assign(body, {
           // version: version
           // access_token: testToken
       });
       // 当没有指定请求方法的时候默认post
       if (method == null) {
           method = 'POST';
       }
       if (mode == null) {
           mode = 'cors';
       }
       this.method = method;
       this.body = body;
       this.mode = mode;
       this.headers = {
           'Accept': 'application/json',
           'Content-Type': 'application/json',
           // 在这里可以添加你需要添加的请求头参数
           // 如 用户 token、手机设备号等等
       };
       // 扩展 网络请求耗时
       this.requestTime = this.getDefaultTestingTime();
      // 默认超时时间
      // 想要超时时间生效 必须修改 fetch 的源码 在发请求之前加上如下代码 (路径为node_modules下的fetch.js的 self.fetch 方法里面)
      // if(init&&init!==null&&typeof init.timeout!=='undefined'){
      //     xhr.timeout=init.timeout;
      // }
       this.requestTimeout = TIMEOUT;
   }

   /**
    * 请覆盖此方法
    */
   requestUrl() {
       throw ({message: 'function requestUrl must be overrided!'});
   }

   /**
    * 开启加载动画
    * 如果需要加载动画可以在这里面去出路
    * 这里在用 redux 去控制显示加载
    * 可以在这里面添加调用加载动画的代码 
    * @returns {BaseRequest}
    */
   showLoadingView() {
       this.isShowing = true;
       // 没有删除自己的 redux
       //store.dispatch(DialogAction.showLoading(true))
       return this;
   }

   /**
    * 关闭dialog
    * @returns {BaseRequest}
    */
   dismissLoadingView() {
       this.isShowing = false;
       // store.dispatch(DialogAction.showLoading(false))
       return this;
   }

   /**
    * 设置超时时间 如果不设置默认20s
    */
   timeout(timeout) {
       this.requestTimeout = typeof timeout === 'number' ? (timeout > 0 ? timeout : TIMEOUT) : TIMEOUT ;
       return this;
   }

   /**
    * 开始请求
    * @param successCallBack 成功后的回调
    * @param failCallBack 失败后的回调
    * @returns {BaseRequest}
    */
   start(successCallBack, failCallBack) {
       //接口响应时间测试
       if (this.requestTime) {
           this.startTime = new Date().getTime();
       }
       //监听android返回键,android
       let url = this.requestUrl();
       if (this.isShowing) {
           BackAndroid.addEventListener(url, () => {
               this.dismissLoadingView();
               BackAndroid.removeEventListener(url);
               return false;
           });
       }
       this._doPost(successCallBack, failCallBack);
       return this;
   }

   /**
    * 开始请求
    * @param successCallBack 成功后的回调
    * @param failCallBack 失败后的回调
    * @returns {BaseRequest}
    */
   async _doPost(successCallBack, failCallBack) {
       try {
           let url = this.getBaseUrl() + this.requestUrl();
           if ('GET' === this.method) {
               let str = this.toQueryString(this.body);
               if (str && str.length > 0) url += '?' + str;
           }
           this.showLog('requestUrl==>' + url);
           this.showLog(this.body);
           let response = await fetch(url, {
               headers: this.headers,
               method: this.method,
               mode: this.mode,
               body: this.method == 'GET' ? null : JSON.stringify(this.body),
               timeout: this.requestTimeout * 1000,
               // credentials: 'include',
           });
           let responseJson = await response.json();
           // console.log('---> BaseRequest start body' + JSON.stringify(this.body));
           this.showLog('response==>' + responseJson);
           this.showLog(responseJson);
           if (this.isShowing) {
               this.dismissLoadingView();
           }
           if (responseJson && !this.isCancled) {
               this.handleResponse(responseJson, successCallBack, failCallBack);
           } else {
               if (failCallBack && !this.isCancled) failCallBack('请求失败');
           }
       } catch (erro) {
           if (this.isShowing) {
               this.dismissLoadingView();
           }
           this.showLog('erro==>');
           this.showLog(erro);
           if (failCallBack && !this.isCancled) failCallBack(erro);
       }
       return this;
   }

   /**
    * 处理response
    * @param responseJson
    * @param successCallBack
    */
   handleResponse(responseJson, successCallBack, failCallBack) {
       if (this.requestTime) {
           try {
               console.log('********requestUrl: ' + this.requestUrl());
               console.log('********This request consumes time: ' + (new Date().getTime() - this.startTime) + 'ms');
           } catch (e) {
           }
       }
       if (this.isShowing) {
           this.dismissLoadingView();
       }
       // 我们项目里面 code 为200 或者 '200' 作为网络请求成功的标示 其余为失败 大家可以根据自己项目的实际情况做更改
       if ('200' == responseJson.code || parseInt(responseJson.code) === 200) {
           if (successCallBack) successCallBack(responseJson);
       } else if (responseJson.message && responseJson.message.length > 0 && this.isShowMessage) {
           // 可以在这里利用 code 处理错误信息  例如:token失效 调用登陆
           // Alert(responseJson.message);
           if (failCallBack) failCallBack(responseJson);
       } else {
           if (failCallBack) failCallBack(responseJson);
       }
   }

   /**
    * 请求是否已取消
    * @returns {*|boolean}
    */
   isCancel() {
       return this.isCancled;
   }

   /**
    * 是否取消请求
    * @param cancle
    */
   setCancled(cancle) {
       this.isCancled = cancle;
       if (this.isShowing) {
           this.dismissLoadingView();
       }
   }

   /**
    * 请求失败后是否显示后台message
    * @param show
    * @returns {BaseRequest}
    */
   setShowMessage(show) {
       this.isShowMessage = show;
       return this;
   }

   /**
    * 用于对对象编码以便进行传输
    * @param obj 对象参数
    * @returns {string} 返回字符串
    */
   toQueryString(obj) {
       let str = '';
       if (obj) {
           let keys = [];
           for (let key in obj) {
               keys.push(key);
           }
           keys.forEach((key, index) => {
               str += key + '=' + obj[key];
               if (index !== keys.length - 1) {
                   str += '&';
               }
           });
       }
       return str;
   }

   /**
    * 打印后台message
    * @param str
    */
   alertWithStr(str) {
       TimerMiXin.setTimeout(() => {
           if (Platform.OS == 'ios') {
               Alert.alert(
                   str,
                   null,
                   [
                       {text: '确定'}
                   ]
               );
           }
           if (Platform.OS == 'android') {
               // 弹窗  可以自己写
           }
       }, 1000);
   }

   /**
    * 打印log信息
    * @param log
    */
   showLog(log) {
       if (DEBUG_MODE) {
           console.log(log);
       }
   }

   /**
    * 返回baseurl
    */
   getBaseUrl() {
       return baseUrl;
   }

   /**
    * 是否打印接口请求时间
    * @returns {boolean}
    */
   getDefaultTestingTime() {
       return false;
   }
}

如需扩展,添加一些功能,可以参考showLoadingView()。
如有错误,欢迎指正。等有时间我会写个 demo 上传到github。

你可能感兴趣的:(react native 基于 fetch 简单封装自己的网络请求)