Flutter+retrofit +dio+rxdart网络请求封装

pubspec.yaml:

dependencies:
rxdart: ^0.23.1
retrofit: '>=3.0.0 <4.0.0'
logger: any #for logging purpose

dev_dependencies:
retrofit_generator: any
build_runner: '>1.3.0 <4.0.0'//用于生成.g文件
json_serializable: '>4.4.0'//生成实体

json_serializable代码在线生成网址
运行:flutter packages pub run build_runner build 会生成 .g文件
报错 flutter packages get
part 'RestClient.g.dart'; //必须配置,否则无法生成.g文件

BaseError

abstract class BaseError {
  final int? code;
  final String? message;

  BaseError({this.code, this.message});
}

class NeedLogin implements BaseError {
  @override
  int get code => 401;

  @override
  String get message => "请先登录";
}

class NeedAuth implements BaseError {
  @override
  int get code => 403;

  @override
  String get message => "非法访问,请使用正确的token";
}

class UserNotExist implements BaseError {
  @override
  int get code => 408;

  @override
  String get message => "用户不存在";
}

class UserNameEmpty implements BaseError {
  @override
  int get code => 405;

  @override
  String get message => "用户名不能为空";
}

class PwdNotMatch implements BaseError {
  @override
  int get code => 409;

  @override
  String get message => "用户密码不正确";
}

class PwdEmpty implements BaseError {
  @override
  int get code => 406;

  @override
  String get message => "用户密码不能为空";
}

class OtherError implements BaseError {
  final int? statusCode;
  final String? statusMessage;

  OtherError({this.statusCode, this.statusMessage});

  @override
  int? get code => statusCode;

  @override
  String? get message => statusMessage;
}

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;
  }
}

BaseDio

class BaseDio {
  BaseDio._(); // 把构造方法私有化

  static BaseDio? _instance;

  static BaseDio? getInstance() {  // 通过 getInstance 获取实例
    _instance ??= BaseDio._();

    return _instance;
  }

  Dio getDio() {
    final Dio dio = Dio();
    dio.options = BaseOptions(receiveTimeout: 55000, connectTimeout: 66000); // 设置超时时间等 ...
    dio.interceptors.add(HeaderInterceptor()); // 添加拦截器,如 token之类,需要全局使用的参数
    dio.interceptors.add(PrettyDioLogger(  // 添加日志格式化工具类
      requestHeader: true,
      requestBody: true,
      responseBody: true,
      responseHeader: false,
      compact: false,
    ));

    return dio;
  }

  BaseError getDioError(Object obj) {  // 这里封装了一个 BaseError 类,会根据后端返回的code返回不同的错误类
    switch (obj.runtimeType) {
      case DioError:
         if ((obj as DioError).type == DioErrorType.response) {
          final response = (obj as DioError).response;
          if (response!.statusCode == 401) {
            return NeedLogin();
          } else if (response!.statusCode == 403) {
            return NeedAuth();
          } else if (response.statusCode == 408) {
            return UserNotExist();
          } else if (response.statusCode == 409) {
            return PwdNotMatch();
          } else if (response.statusCode == 405) {
            return UserNameEmpty();
          } else if (response.statusCode == 406) {
            return PwdEmpty();
          } else {
            return OtherError(
              statusCode: response.statusCode,
              statusMessage: response.statusMessage,
            );
          }
        }
    }

    return OtherError();
  }
}

RestApi

part 'RestClient.g.dart'; //必须配置,否则无法生成.g文件

@RestApi(baseUrl: "https://uat.../")
abstract class RestClient {
  factory RestClient({Dio? dio, String? baseUrl}) {
    dio ??= BaseDio.getInstance()!.getDio();
    return _RestClient(dio, baseUrl: baseUrl);
  }

  @POST("user/getImgCode.do")
  // Future<Task> getTasks(@Body()Task task);
  Future<Entity> getTasks(@Body() loginparms task);
}

//
// @JsonSerializable()
// class Task {
//   String? imgToken;
//   String? img;
//
//   Task({this.imgToken, this.img});
//
//   factory Task.fromJson(Map<String, dynamic> json) => _$TaskFromJson(json);
//   Map<String, dynamic> toJson() => _$TaskToJson(this);
// }
@JsonSerializable()
class Entity extends Object {
  @JsonKey(name: 'data')
  Data data;

  Entity(
    this.data,
  );

  factory Entity.fromJson(Map<String, dynamic> srcJson) =>
      _$EntityFromJson(srcJson);

  Map<String, dynamic> toJson() => _$EntityToJson(this);
}

@JsonSerializable()
class Data extends Object {
  @JsonKey(name: 'imgToken')
  String imgToken;

  @JsonKey(name: 'img')
  String img;

  Data(
    this.imgToken,
    this.img,
  );

  factory Data.fromJson(Map<String, dynamic> srcJson) =>
      _$DataFromJson(srcJson);

  Map<String, dynamic> toJson() => _$DataToJson(this);
}

@JsonSerializable()
class loginparms extends Object {

  @JsonKey(name: 'parameter')
  Parameter parameter;

  @JsonKey(name: 'versionCode')
  String versionCode;

  loginparms(this.parameter,this.versionCode,);

  factory loginparms.fromJson(Map<String, dynamic> srcJson) => _$loginparmsFromJson(srcJson);

  Map<String, dynamic> toJson() => _$loginparmsToJson(this);

}


@JsonSerializable()
class Parameter extends Object {

  Parameter();

  factory Parameter.fromJson(Map<String, dynamic> srcJson) => _$ParameterFromJson(srcJson);

  Map<String, dynamic> toJson() => _$ParameterToJson(this);

调用

Future<void> clickLike() async {
    try {
      print('你好============123456789--0}');
  
      Parameter parameter=Parameter();
      loginparms loginparm= loginparms(parameter,"");
      Entity data = await RestClient().getTasks(loginparm); // 使用非常简单一句代码即可
      // ImageCodeBean data = await RestClient().getTasks(map); // 使用非常简单一句代码即可
      print('你好============123456789--5${data.data.imgToken}');
    } catch (e) {
      error: BaseDio.getInstance()!.getDioError(e);
      //拿到BaseError对应自定义message错误信息
      print('你好============123456789--1${BaseDio.getInstance()!.getDioError(e).message}');

    }
  }

可参照:https://www.psvmc.cn/article/2020-05-12-flutter-start-07-http-json.html

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

推荐阅读更多精彩内容