Flutter网络请求dio简易封装

DioManager

///    date   : 2021/7/11
///    desc   : 网络请求类

///成功回调
typedef Success<T> = Function(T? data);

///失败回调,若传空,则系统吐司提示,若不会空,自行处理错误结果
typedef Fail = Function(int? code, String? msg);

class DioManager {
  /// 默认code--返回成功
  static const RESP_OK = 200;

  /// 默认code--40001未登录,40002token失效,40010账号被异地登录,40011token来源异常
  static const List RESP_LOGIN_EXPIRE = [40001, 40002, 40010, 40011];

  String? signKey;

  Dio dio = new Dio();
  static DioManager? _instance;

  static DioManager getInstance() {
    if (_instance == null) {
      _instance = new DioManager();
      _instance!._init();
    }
    return _instance!;
  }

  void _init() {
    dio.options.baseUrl = "https://www.fastmock.site/mock/6d5084df89b4c7a49b28052a0f51c29a/test/";
    dio.options.connectTimeout = 5000;
    dio.options.receiveTimeout = 5000;
    dio.options.responseType = ResponseType.json;
    dio.interceptors.add(HeaderInterceptor()); //请求头拦截器
    dio.interceptors.add(DioLogInterceptor()); //是否开启请求日志
    // dio.interceptors.add(ResponseInterceptor()); //请求结果回调
  }

  void setBaseUrl(String baseUrl) {
    dio.options.baseUrl = baseUrl;
  }

  void setSignKey(String signKey) => this.signKey = signKey;

  void setHeader(Map<String, dynamic>? headers) {
    dio.options.headers = headers;
  }

  ///post form 表单提交操作
  Future postForm<T>(
    String path, {
    SplayTreeMap<String, dynamic>? params,
    required Success<T> success,
    required Fail? fail,
    bool withLoading = true,
  }) async {
    var splayTreeMap = new SplayTreeMap<String, dynamic>();
    if (params != null) {
      splayTreeMap.addAll(params);
    }

    if (withLoading) {
      // ToastUtils.getInstance().showLoading();
    }
    try {
      var response = await dio.post(path, data: FormData.fromMap(splayTreeMap));
      parseResponse(response, success, fail);
    } on DioError catch (e) {
      formatError(e, fail);
    }
  }

  ///post json
  Future postJson<T>(
    String path, {
    SplayTreeMap<String, dynamic>? params,
    required Success<T> success,
    required Fail? fail,
    bool withLoading = true,
  }) async {
    var splayTreeMap = new SplayTreeMap<String, dynamic>();
    if (params != null) {
      splayTreeMap.addAll(params);
    }
    if (withLoading) {
      // ToastUtils.getInstance().showLoading();
    }
    try {
      var response = await dio.post(path, data: splayTreeMap);
      parseResponse(response, success, fail);
      print('你好============response11${response}');

    } on DioError catch (e) {
      formatError(e, fail);
    }
  }

  ///get
  Future get<T>(
    String path, {
    SplayTreeMap<String, dynamic>? params,
    required Success<T> success,
    required Fail? fail,
    bool withLoading = true,
  }) async {
    var splayTreeMap = new SplayTreeMap<String, dynamic>();
    if (params != null) {
      splayTreeMap.addAll(params);
    }

    if (withLoading) {
      // ToastUtils.getInstance().showLoading();
    }
    try {
      var response = await dio.get(path, queryParameters: splayTreeMap);
      parseResponse(response, success, fail);
    } on DioError catch (e) {
      formatError(e, fail);
    }
  }

  void parseResponse<T>(
      Response<dynamic> response, Success<T> success, Fail? fail) {
    // ToastUtils.getInstance().hideLoading();
    if (response.statusCode == 200) {
      Map<String, dynamic> result = response.data;
      if (result["code"] == RESP_OK) {
        success(result["data"]);
        return;
      }
      if (RESP_LOGIN_EXPIRE.contains(result["code"])) {
        //登录过期需要重新登录
        // ToastUtils.getInstance().toastFail(result["msg"]);
        // NativeUtils.sendNativeData(NativeUtils.NATIVE_LOGIN);
        return;
      }
      if (fail != null) {
        // fail(result["code"], result["msg"]);
        fail(result["code"], result["message"]);
      } else {
        // ToastUtils.getInstance().toastFail(result["msg"] ?? '');
      }
    } else {
      if (fail != null) {
        fail(response.statusCode ?? -1, response.statusMessage ?? 'error');
      } else {
        // ToastUtils.getInstance().toastFail(response.statusMessage ?? 'error');
      }
    }
  }

  void formatError(DioError e, Fail? fail) {
    print(e.stackTrace);
    if (fail != null) {
      fail(-1, e.message);
    } else {
      // ToastUtils.getInstance().toastFail(e.message);
    }
  }
}

HeaderInterceptor

///    author : luzi
///    date   : 2021/7/11
///    desc   : 请求头拦截器
class HeaderInterceptor extends Interceptor {
  @override
  Future<void> onRequest(
    RequestOptions options,
    RequestInterceptorHandler handler,
  ) async {
    Map<String, dynamic> headers = options.headers;
    headers['clientId'] = 1; //渠道(1.APP 2.Web)
    if (Platform.isAndroid) {
      headers['systemName'] = 'Android'; //系统名称
    } else if (Platform.isIOS) {
      headers['systemName'] = 'iOS'; //系统名称
    }

    var time = DateTime.now().millisecondsSinceEpoch;
    headers['ts'] = time; //时间戳
    String sign = getParamsStr(options, time);
    headers['sign'] = sign; //参数加密签名
    headers['language'] = Get.locale?.countryCode ?? 'MY'; //语言环境
    headers['Accept'] = 'application/json';
    options.headers = headers;
    handler.next(options);
  }

  getParamsStr(RequestOptions options, int time) {
    StringBuffer stringBuffer = new StringBuffer();
    String? paramsStr;
    var method = options.method;
    if (method == 'GET') {
      if (options.queryParameters.isNotEmpty) {
        String query = options.uri.query;
        var splitArray = query.split('&');
        var splayTreeMap = new SplayTreeMap();
        splitArray.forEach((split) {
          var splitChild = split.split('=');
          splayTreeMap[splitChild[0]] = Uri.encodeQueryComponent(splitChild[1]);
        });
        paramsStr = json.encode(splayTreeMap);
        if (paramsStr.isNotEmpty) {
          stringBuffer.write('$paramsStr&');
        } else {
          stringBuffer.write('{}&');
        }
      } else {
        stringBuffer.write('{}&');
      }
    } else if (method == 'POST') {
      var data = options.data;
      if (data is FormData) {
        if (data.fields.isNotEmpty) {
          String replace = data.fields
              .join(",")
              .replaceAll('MapEntry', '')
              .replaceAll('(', '')
              .replaceAll(')', '');
          paramsStr = replace;
          if (paramsStr.isNotEmpty) {
            stringBuffer.write('{$paramsStr}&');
          } else {
            stringBuffer.write('{}&');
          }
        } else {
          stringBuffer.write('{}&');
        }
      } else {
        paramsStr = jsonEncode(data);
        if (paramsStr.isNotEmpty) {
          stringBuffer.write('$paramsStr&');
        } else {
          stringBuffer.write('{}&');
        }
      }
    }
    // NativeUtils.sendNativeData(NativeUtils.NATIVE_LOG, paramsStr.toString());
    // print(paramsStr.toString());
    stringBuffer.write('ts=$time&');
    stringBuffer.write('tmpsign=${DioManager.getInstance().signKey}');
    String toString = stringBuffer.toString().replaceAll("\"", "");
    // NativeUtils.sendNativeData(NativeUtils.NATIVE_LOG, toString);
    // print(toString);
    // Get.log.printInfo(info: '拼接后:$toString');
    var content = new Utf8Encoder().convert(toString);
    var digest = md5.convert(content);
    String md5Encode = hex.encode(digest.bytes).toUpperCase();
    Get.log.printInfo(info: 'MD5:$md5Encode');
    return md5Encode;
  }
}

DioLogInterceptor

///    date   : 2021/7/11
///    desc   : 日志拦截器
class DioLogInterceptor extends Interceptor {
  /// Print request [Options]
  final bool request;

  /// Print request header [Options.headers]
  final bool requestHeader;

  /// Print request data [Options.data]
  final bool requestBody;

  /// Print [Response.data]
  final bool responseBody;

  /// Print [Response.headers]
  final bool responseHeader;

  /// Print error message
  final bool error;

  /// InitialTab count to logPrint json response
  static const int initialTab = 1;

  /// 1 tab length
  static const String tabStep = '    ';

  /// Print compact json response
  final bool compact;

  /// Width size per logPrint
  final int maxWidth;

  /// Log printer; defaults logPrint log to console.
  /// In flutter, you'd better use debugPrint.
  /// you can also write log in a file.
  void Function(Object object) logPrint;

  DioLogInterceptor({
    this.request = true,
    this.requestHeader = true,
    this.requestBody = true,
    this.responseHeader = false,
    this.responseBody = true,
    this.error = true,
    this.maxWidth = 90,
    this.compact = true,
    this.logPrint = print,
  });

  @override
  void onRequest(
      RequestOptions options, RequestInterceptorHandler handler) async {
    if (request) {
      _printRequestHeader(options);
    }
    if (requestHeader) {
      _printMapAsTable(options.queryParameters, header: 'Query Parameters');
      final requestHeaders = Map();
      requestHeaders.addAll(options.headers);
      requestHeaders['contentType'] = options.contentType?.toString();
      requestHeaders['responseType'] = options.responseType.toString();
      requestHeaders['followRedirects'] = options.followRedirects;
      requestHeaders['connectTimeout'] = options.connectTimeout;
      requestHeaders['receiveTimeout'] = options.receiveTimeout;
      _printMapAsTable(requestHeaders, header: 'Headers');
      _printMapAsTable(options.extra, header: 'Extras');
    }
    if (requestBody && options.method != 'GET') {
      final data = options.data;
      if (data != null) {
        if (data is Map) _printMapAsTable(options.data, header: 'Body');
        if (data is FormData) {
          final formDataMap = Map()
            ..addEntries(data.fields)
            ..addEntries(data.files);
          _printMapAsTable(formDataMap, header: 'Form data | ${data.boundary}');
        } else
          _printBlock(data.toString());
      }
    }
    handler.next(options);
  }

  @override
  void onError(
    DioError err,
    ErrorInterceptorHandler handler,
  ) async {
    if (error) {
      if (err.type == DioErrorType.response) {
        final uri = err.response?.requestOptions.uri;
        _printBoxed(
            header:
                'DioError ║ Status: ${err.response?.statusCode} ${err.response?.statusMessage}',
            text: uri.toString());
        if (err.response != null && err.response?.data != null) {
          logPrint('╔ ${err.type.toString()}');
          if (err.response != null) {
            _printResponse(err.response!);
          }
        }
        _printLine('╚');
        logPrint('');
      } else
        _printBoxed(header: 'DioError ║ ${err.type}', text: err.message);
    }
    handler.next(err);
  }

  @override
  void onResponse(
    Response response,
    ResponseInterceptorHandler handler,
  ) async {
    if (responseHeader) {
      final responseHeaders = Map<String, String>();
      response.headers
          .forEach((k, list) => responseHeaders[k] = list.toString());
      _printMapAsTable(responseHeaders, header: 'Headers');
    }

    if (responseBody) {
      _printResponseHeader(response);
      logPrint('╔ Body');
      logPrint('║');
      _printResponse(response);
      logPrint('║');
      _printLine('╚');
    }
    handler.next(response);
  }

  void _printBoxed({String? header, String? text}) {
    logPrint('');
    logPrint('╔╣ $header');
    logPrint('║  $text');
    _printLine('╚');
  }

  void _printResponse(Response response) {
    if (response.data != null) {
      if (response.data is Map)
        _printPrettyMap(response.data);
      else if (response.data is List) {
        logPrint('║${_indent()}[');
        _printList(response.data);
        logPrint('║${_indent()}[');
      } else
        _printBlock(response.data.toString());
    }
  }

  void _printResponseHeader(Response? response) {
    final uri = response?.requestOptions.uri;
    final method = response?.requestOptions.method;
    //添加以下代码,防止多接口请求时 response的header和body被分开打印
    String header =
        'Response ║ $method ║ Status: ${response?.statusCode} ${response?.statusMessage}';
    logPrint('╔╣ $header' + ('═' * 70));
    logPrint('║  ${uri.toString()}');
    logPrint('║  ');
  }

  void _printRequestHeader(RequestOptions? options) {
    final uri = options?.uri;
    final method = options?.method;
    _printBoxed(header: 'Request ║ $method ', text: uri.toString());
  }

  void _printLine([String pre = '', String suf = '╝']) =>
      logPrint('$pre${'═' * maxWidth}$suf');

  void _printKV(String key, Object? v) {
    final pre = '╟ $key: ';
    final msg = v.toString();

    if (pre.length + msg.length > maxWidth) {
      logPrint(pre);
      _printBlock(msg);
    } else
      logPrint('$pre$msg');
  }

  void _printBlock(String msg) {
    int lines = (msg.length / maxWidth).ceil();
    for (int i = 0; i < lines; ++i) {
      logPrint((i >= 0 ? '║ ' : '') +
          msg.substring(i * maxWidth,
              math.min<int>(i * maxWidth + maxWidth, msg.length)));
    }
  }

  String _indent([int tabCount = initialTab]) => tabStep * tabCount;

  void _printPrettyMap(Map data,
      {int tabs = initialTab, bool isListItem = false, bool isLast = false}) {
    final bool isRoot = tabs == initialTab;
    final initialIndent = _indent(tabs);
    tabs++;

    if (isRoot || isListItem) logPrint('║$initialIndent{');

    data.keys.toList().asMap().forEach((index, key) {
      final isLast = index == data.length - 1;
      var value = data[key];
//      key = '\"$key\"';
      if (value is String) value = '\"$value\"';
      if (value is Map) {
        if (compact && _canFlattenMap(value))
          logPrint('║${_indent(tabs)} $key: $value${!isLast ? ',' : ''}');
        else {
          logPrint('║${_indent(tabs)} $key: {');
          _printPrettyMap(value, tabs: tabs);
        }
      } else if (value is List) {
        if (compact && _canFlattenList(value))
          logPrint('║${_indent(tabs)} $key: ${value.toString()}');
        else {
          logPrint('║${_indent(tabs)} $key: [');
          _printList(value, tabs: tabs);
          logPrint('║${_indent(tabs)} ]${isLast ? '' : ','}');
        }
      } else {
        final msg = value.toString().replaceAll('\n', '');
        final indent = _indent(tabs);
        final linWidth = maxWidth - indent.length;
        if (msg.length + indent.length > linWidth) {
          int lines = (msg.length / linWidth).ceil();
          for (int i = 0; i < lines; ++i) {
            logPrint(
                '║${_indent(tabs)} ${msg.substring(i * linWidth, math.min<int>(i * linWidth + linWidth, msg.length))}');
          }
        } else
          logPrint('║${_indent(tabs)} $key: $msg${!isLast ? ',' : ''}');
      }
    });

    logPrint('║$initialIndent}${isListItem && !isLast ? ',' : ''}');
  }

  void _printList(List list, {int tabs = initialTab}) {
    list.asMap().forEach((i, e) {
      final isLast = i == list.length - 1;
      if (e is Map) {
        if (compact && _canFlattenMap(e))
          logPrint('║${_indent(tabs)}  $e${!isLast ? ',' : ''}');
        else
          _printPrettyMap(e, tabs: tabs + 1, isListItem: true, isLast: isLast);
      } else
        logPrint('║${_indent(tabs + 2)} $e${isLast ? '' : ','}');
    });
  }

  bool _canFlattenMap(Map map) {
    return map.values.where((val) => val is Map || val is List).isEmpty &&
        map.toString().length < maxWidth;
  }

  bool _canFlattenList(List list) {
    return (list.length < 10 && list.toString().length < maxWidth);
  }

  void _printMapAsTable(Map? map, {String? header}) {
    if (map == null || map.isEmpty) return;
    logPrint('╔ $header ');
    map.forEach((key, value) => _printKV(key, value));
    _printLine('╚');
  }
}

OperatingCenterEngine

class OperatingCenterEngine {

  static Future queryAlertInfo(Success<Map<String, dynamic>> success,
      {Fail? fail}) async {
    await DioManager.getInstance().get('warning-rules/getConfiguration',
        params: null, success: success, fail: fail);
  }


  static Future updateAlertInfo(
      String alertAmount, Success<Map<String, dynamic>> success,
      {Fail? fail}) async {
    SplayTreeMap<String, dynamic> map = new SplayTreeMap<String, dynamic>();
    map['alertAmount'] = alertAmount; 
    await DioManager.getInstance().postJson(
        'warning-rules/alertValueConfiguration',
        params: map,
        success: success,
        fail: fail);
  }


  static Future queryBindBankInfo(Success<Map<String, dynamic>> success,
      {Fail? fail}) async {
    await DioManager.getInstance().postJson('bank-card/cashDetail',
        params: null, success: success, fail: fail);
  }


  static Future addExample(
      SplayTreeMap<String, dynamic> map, Success<Map<String, dynamic>> success,
      {Fail? fail}) async {
    await DioManager.getInstance()
        .postJson('bank-card/save', params: map, success: success, fail: fail);
  }

  static Future updateExample(
      SplayTreeMap<String, dynamic> map, Success<Map<String, dynamic>> success,
      {Fail? fail}) async {
    await DioManager.getInstance().postJson('bank-card/update',
        params: map, success: success, fail: fail);
  }


  static Future getExampleList(Success<List<dynamic>> success,
      {Fail? fail}) async {
    SplayTreeMap<String, dynamic> map = new SplayTreeMap();
    map['dictCode'] = 'CASH_BANK';
    await DioManager.getInstance()
        .get('dict/listDictValue', params: map, success: success, fail: fail);
  }


  static Future getImageCodeNew( SplayTreeMap<String, dynamic> map,Success<Map<String, dynamic>> success,
      {Fail? fail}) async {
    // SplayTreeMap<String, dynamic> map = new SplayTreeMap();
    // map['ceshi'] = '123456999999';
    await DioManager.getInstance()
        .postJson('test', params: map, success: success, fail: fail);

  }
}

实际调用

   SplayTreeMap<String, dynamic> map = new SplayTreeMap();
          map['username'] ="你好" ;
          map['loongwind'] ="123456" ;
          OperatingCenterEngine.getImageCodeNew(map,(data) =>{
          print('你好============123456${data!['age']}'),
          });

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容