Flutter 网络请求框架封装(1)

本文介绍了Dio网络请求框架的使用方法,包括如何通过对象传递查询参数、添加请求和响应拦截器,以及如何处理请求错误,同时提到了FlutterJsonBeanFactory插件用于自动生成JSON实体类。
摘要由CSDN通过智能技术生成

var response = await dio.get(“/test?id=12&name=chen”);
_content = response.data.toString();
}

对于 query 参数,我们可以通过对象来进行传递,上面的代码等同于:

void getRequest() async {
Dio dio = new Dio();
var response = await dio.get(“/test”,data:{“id”:12,“name”:“chen”});
_content = response.data.toString();
}

Post 请求

void postRequest() async {
var dio = new Dio();
var response = await dio.post(url_post, data:{“id”:12,“name”:“wendu”});
_content = response.data.toString();
}

Dio 网络请求框架封装

日志信息拦截

Dio 和 okhttp 一样,都会有一个请求拦截器和响应拦截器,通过拦截器,我们可以在请求之前或响应之后做一些同意的预处理。例如我们发起请求前查看我们请求的参数和头部,响应的时候,我们可以查看返回来的数据。

Dio dio = new Dio();
// 添加拦截器
if (Config.DEBUG) {
dio.interceptors.add(InterceptorsWrapper(
onRequest: (RequestOptions options){
print(“\n================== 请求数据 ========“);
print(“url = ${options.uri.toString()}”);
print(“headers = ${options.headers}”);
print(“params = ${options.data}”);
},
onResponse: (Response response){
print(”\n
响应数据 ========“);
print(“code = ${response.statusCode}”);
print(“data = ${response.data}”);
print(”\n");
},
onError: (DioError e){
print("\n
错误响应数据 ======================”);
print(“type = ${e.type}”);
print(“message = ${e.message}”);
print(“stackTrace = ${e.stackTrace}”);
print(“\n”);
}
));
}

如果我们想要移除拦截器,那么我们可以将其设置为 null

dio.interceptor.request.onSend=null;
dio.interceptor.response.onSuccess=null;
dio.interceptor.response.onError=null;

token 添加

// 头部添加 token 验证
headers[“Authorization”] = “token lskjdlklsjkdklsjd333”;
option.headers = headers;
///超时
option.connectTimeout = 15000;
try {
Response response = await dio.request(url, data: params, options: option);
} on DioError catch (e) {
// 请求错误处理
}

自动生成 dart 的 json 实体类插件 FlutterJsonBeanFactory

在 Android 开发中,有 GsonFormat 这个插件来讲 json 数据自动转化成 Bean;那么在 Flutter 中也有类似的插件可以生产序列化的实体类的插件:FlutterJsonBeanFactory

  • step 1:下载插件 FlutterJsonBeanFactory,安装完成后重启

Setting -> Plugins -> Browse Respositories 中搜索 FlutterJsonBeanFactory

  • step 2:创建实体类,在指定目录下:

New -> dart bean class File from JSON

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

  • step 3:输入实体类名及 json 格式的数据

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

  • step 4:最后生成的实体类:LoginEntity

class LoginEntity {
String easemobpassword;
String username;

LoginEntity({this.easemobpassword, this.username});

LoginEntity.fromJson(Map<String, dynamic> json) {
easemobpassword = json[‘easemobPassword’];
username = json[‘username’];
}

Map<String, dynamic> toJson() {
final Map<String, dynamic> data = new Map<String, dynamic>();
data[‘easemobPassword’] = this.easemobpassword;
data[‘username’] = this.username;
return data;
}
}

请求错误处理

Response response;
try {
response = await dio.request(url, data: params, options: option);
} on DioError catch (e) {
// 请求错误处理
Response errorResponse;
if (e.response != null) {
errorResponse = e.response;
} else {
errorResponse = new Response(statusCode: 666);
}
if (e.type == DioErrorType.CONNECT_TIMEOUT) {
errorResponse.statusCode = Code.NETWORK_TIMEOUT;
}
if (Config.DEBUG) {
print('请求异常: ’ + e.toString());
print('请求异常 url: ’ + url);
}
return new ResultData(Code.errorHandleFunction(errorResponse.statusCode, e.message, noTip), false, errorResponse.statusCode);
}

其中 ResultData 是网络结果处理的实体类

/**

  • 网络结果数据
  • Created by chenjianrun
  • Date: 2018-07-16
    */
    class ResultData {
    var data;
    bool result;
    int code;
    var headers;

ResultData(this.data, this.result, this.code, {this.headers});
}

Code 是处理网络错误的编码,并将错误结果通过 eventbus 发送出去,一般我们可以在 main_pager 中注册监听这个事件。

///网络请求错误编码
class Code {
///网络错误
static const NETWORK_ERROR = -1;

///网络超时
static const NETWORK_TIMEOUT = -2;

///网络返回数据格式化一次
static const NETWORK_JSON_EXCEPTION = -3;

static const SUCCESS = 200;

static final EventBus eventBus = new EventBus();

static errorHandleFunction(code, message, noTip) {
if(noTip) {
return message;
}
eventBus.fire(new HttpErrorEvent(code, message));
return message;
}
}

完成的网络请求类:HttpRequest

import ‘dart:io’;

import ‘package:dio/dio.dart’;
import ‘package:private_tutor/common/SpUtils.dart’;
import ‘package:connectivity/connectivity.dart’;

import ‘dart:collection’;

import ‘package:private_tutor/common/config/Config.dart’;
import ‘package:private_tutor/net/ResultCode.dart’;
import ‘package:private_tutor/net/ResultData.dart’;

///http请求管理类,可单独抽取出来
class HttpRequest {
static String _baseUrl;
static const CONTENT_TYPE_JSON = “application/json”;
static const CONTENT_TYPE_FORM = “application/x-www-form-urlencoded”;
static Map optionParams = {
“timeoutMs”: 15000,
“token”: null,
“authorizationCode”: null,
};

static setBaseUrl(String baseUrl){
_baseUrl = baseUrl;
}

static get(url,param) async{
return await request(_baseUrl+url, param, null, new Options(method:“GET”));
}

static post(url,param) async{
return await request(_baseUrl+url, param, {“Accept”: ‘application/vnd.github.VERSION.full+json’}, new Options(method: ‘POST’));
}

static delete(url,param) async{
return await request(_baseUrl+url, param, null, new Options(method: ‘DELETE’));
}

static put(url,param) async{
return await request(_baseUrl+url, param, null, new Options(method: “PUT”, contentType: ContentType.text));
}

///发起网络请求
///[ url] 请求url
///[ params] 请求参数
///[ header] 外加头
///[ option] 配置
static request(url, params, Map<String, String> header, Options option, {noTip = false}) async {

//没有网络
var connectivityResult = await (new Connectivity().checkConnectivity());
if (connectivityResult == ConnectivityResult.none) {
return new ResultData(Code.errorHandleFunction(Code.NETWORK_ERROR, “”, noTip), false, Code.NETWORK_ERROR);
}

Map<String, String> headers = new HashMap();
if (header != null) {
headers.addAll(header);
}

//授权码
if (optionParams[“authorizationCode”] == null) {
var authorizationCode = await getAuthorization();
if (authorizationCode != null) {
optionParams[“authorizationCode”] = authorizationCode;
}
}

headers[“Authorization”] = optionParams[“authorizationCode”];
// 设置 baseUrl

if (option != null) {
option.headers = headers;
} else{
option = new Options(method: “get”);
option.headers = headers;
}

///超时
option.connectTimeout = 15000;

Dio dio = new Dio();
// 添加拦截器
if (Config.DEBUG) {
dio.interceptors.add(InterceptorsWrapper(
onRequest: (RequestOptions options){
print(“\n================== 请求数据 ========“);
print(“url = ${options.uri.toString()}”);
print(“headers = ${options.headers}”);
print(“params = ${options.data}”);
},
onResponse: (Response response){
print(”\n
响应数据 ========“);
print(“code = ${response.statusCode}”);
print(“data = ${response.data}”);
print(”\n");
},
onError: (DioError e){
print("\n
错误响应数据 ======================”);
print(“type = ${e.type}”);
print(“message = ${e.message}”);
print(“stackTrace = ${e.stackTrace}”);
print(“\n”);
}
));
}

Response response;
try {
response = await dio.request(url, data: params, options: option);
} on DioError catch (e) {
// 请求错误处理
Response errorResponse;
if (e.response != null) {
errorResponse = e.response;
} else {
errorResponse = new Response(statusCode: 666);
}
if (e.type == DioErrorType.CONNECT_TIMEOUT) {
errorResponse.statusCode = Code.NETWORK_TIMEOUT;
}
if (Config.DEBUG) {
print('请求异常: ’ + e.toString());
print('请求异常 url: ’ + url);
}
return new ResultData(Code.errorHandleFunction(errorResponse.statusCode, e.message, noTip), false, errorResponse.statusCode);
}

try {
if (option.contentType != null && option.contentType.primaryType == “text”) {
return new ResultData(response.data, true, Code.SUCCESS);
} else {
var responseJson = response.data;
if (response.statusCode == 201 && responseJson[“token”] != null) {
optionParams[“authorizationCode”] = 'token ’ + responseJson[“token”];
await SpUtils.save(Config.TOKEN_KEY, optionParams[“authorizationCode”]);
}
}
if (response.statusCode == 200 || response.statusCode == 201) {
return ResultData(response.data, true, Code.SUCCESS, headers: response.headers);
}
} catch (e) {
print(e.toString() + url);
return ResultData(response.data, false, response.statusCode, headers: response.headers);
}
return new ResultData(Code.errorHandleFunction(response.statusCode, “”, noTip), false, response.statusCode);
}

///清除授权
static clearAuthorization() {
optionParams[“authorizationCode”] = null;
SpUtils.remove(Config.TOKEN_KEY);
}

///获取授权token
static getAuthorization() async {
String token = await SpUtils.get(Config.TOKEN_KEY);
if (token == null) {
String basic = await SpUtils.get(Config.USER_BASIC_CODE);
if (basic == null) {
//提示输入账号密码
} else {
//通过 basic 去获取token,获取到设置,返回token
return “Basic $basic”;
}
} else {
optionParams[“authorizationCode”] = token;
return token;
}
}
}

使用示例

/// 登录 model
class LoginModel{
// 手机号码登录
static phoneLogin(String phone,String verifyCode) async{
ResultData response = await HttpRequest.post(Address.phoneLogin, {“phoneNum” : phone,“captcha”:verifyCode});
if(response != null && response.result){
PhoneLoginEntity phoneLoginEntity = PhoneLoginEntity.fromJson(json.decode(response.data));
先自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《Android移动开发全套学习资料》送给大家,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频

如果你觉得这些内容对你有帮助,可以扫码领取!!!!

尾声

最后,我再重复一次,如果你想成为一个优秀的 Android 开发人员,请集中精力,对基础和重要的事情做深度研究。

对于很多初中级Android工程师而言,想要提升技能,往往是自己摸索成长,不成体系的学习效果低效漫长且无助。 整理的这些架构技术希望对Android开发的朋友们有所参考以及少走弯路,本文的重点是你有没有收获与成长,其余的都不重要,希望读者们能谨记这一点。

这里,笔者分享一份从架构哲学的层面来剖析的视频及资料分享给大家梳理了多年的架构经验,筹备近6个月最新录制的,相信这份视频能给你带来不一样的启发、收获。

Android进阶学习资料库

一共十个专题,包括了Android进阶所有学习资料,Android进阶视频,Flutter,java基础,kotlin,NDK模块,计算机网络,数据结构与算法,微信小程序,面试题解析,framework源码!

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可免费领取!

与成长,其余的都不重要,希望读者们能谨记这一点。

这里,笔者分享一份从架构哲学的层面来剖析的视频及资料分享给大家梳理了多年的架构经验,筹备近6个月最新录制的,相信这份视频能给你带来不一样的启发、收获。

[外链图片转存中…(img-0JDfs3tx-1711246955103)]

Android进阶学习资料库

一共十个专题,包括了Android进阶所有学习资料,Android进阶视频,Flutter,java基础,kotlin,NDK模块,计算机网络,数据结构与算法,微信小程序,面试题解析,framework源码!
[外链图片转存中…(img-yFqsSd8W-1711246955103)]
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可免费领取!

  • 21
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Flutter网络请求dio封装是一种常用的网络请求框架,它可以帮助我们快速地进行网络请求,同时也可以提高我们的开发效率。在使用dio进行网络请求时,我们可以通过封装来简化代码,提高代码的可读性和可维护性。常见的封装方式有: 1. 封装请求方法:将网络请求的方法封装成一个函数,可以传入参数,方便调用。例如: ``` Future<Response> post(String url, Map<String, dynamic> data) async { try { Response response = await Dio().post(url, data: data); return response; } catch (e) { throw e; } } ``` 2. 封装请求拦截器:可以在请求前或请求后进行一些操作,例如添加请求头、打印请求日志等。例如: ``` class HttpUtil { static Dio dio = Dio(); static Future<Response> get(String url, {Map<String, dynamic> params}) async { dio.interceptors.add(InterceptorsWrapper( onRequest: (options, handler) { // 添加请求头 options.headers['Authorization'] = 'Bearer token'; return handler.next(options); }, onResponse: (response, handler) { // 打印请求日志 print('response: ${response.data}'); return handler.next(response); }, )); try { Response response = await dio.get(url, queryParameters: params); return response; } catch (e) { throw e; } } } ``` 3. 封装错误处理:可以统一处理网络请求的错误,例如网络异常、请求超时等。例如: ``` class HttpUtil { static Dio dio = Dio(); static Future<Response> get(String url, {Map<String, dynamic> params}) async { try { Response response = await dio.get(url, queryParameters: params); return response; } on DioError catch (e) { if (e.type == DioErrorType.CONNECT_TIMEOUT) { throw '请求超时'; } else if (e.type == DioErrorType.RECEIVE_TIMEOUT) { throw '响应超时'; } else if (e.type == DioErrorType.RESPONSE) { throw '请求异常,状态码:${e.response.statusCode}'; } else if (e.type == DioErrorType.CANCEL) { throw '请求取消'; } else { throw '网络异常'; } } } } ``` 通过封装,我们可以让代码更加简洁、易读、易维护,同时也可以提高开发效率。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值