Android基于Retrofit2.0 +RxJava 封装的超好用的RetrofitClient工具类(六)
2016-07-19 17:44
801 查看
RetrofitClient
基于Retrofit2.0封装的RetrofitClient. 已加入RxJava避免重复创建Retrofit实列.
调用方便简洁.
无需重复设置属性的步骤.
可固定配置 Host 也可动态配置Url、请求头、参数等.
支持文件下载和上传.
可支持泛型扩展的ApiService
支持RxJava
支持缓存机制
支持统一错误结果处理
使用原生的Retrofit请求网络,熟悉的朋友必定了解,在某个ApiServie方法多时 Retrofit设置就显得有点累赘,今天给大家带来对Retrofit的基本封装。这次对Retrofit进阶篇,本次封装已加入RxJava,请在阅读下文前请先了解RXJAVA和本人写的Retrofit系列文章,
友情导读:
Retrofit 2.0
超能实践,完美支持Https传输
Retrofit2.0
完美同步Cookie实现免登录
Retrofit 2.0 超能实践(三),轻松实现文件/图片上传
基于Retrofit2.0 封装的超好用的RetrofitClient工具类
玩转IOC,教你徒手实现自定义的Retrofit框架
基本步骤:
构建Retrofit的接口service.
构建基础拦截器 Interceptor.
构建Cookie管理工具CookieManger.
构建 单列RetrofitClient客户端.
RetrofitClient的使用.
ApiService
请求网络的API接口类,这里你可以增加你需要的请求接口,也可复用已经实现的几个方法。/** * Created by Tamic on 2016-07-08. */ public interface ApiService { public static final String Base_URL = "http://ip.taobao.com/"; /** *普通写法 */ @GET("service/getIpInfo.php/") Observable<ResponseBody> getData(@Query("ip") String ip); @GET("{url}") Observable<ResponseBody> executeGet( @Path("url") String url, @QueryMap Map<String, String> maps); @POST("{url}") Observable<ResponseBody> executePost( @Path("url") String url, @QueryMap Map<String, String> maps); @Multipart @POST("{url}") Observable<ResponseBody> upLoadFile( @Path("url") String url, @Part("image\\\\"; filename=\\"image.jpg") RequestBody avatar); @POST("{url}") Observable<ResponseBody> uploadFiles( @Path("url") String url, @Path("headers") Map<String, String> headers, @Part("filename") String description, @PartMap() Map<String, RequestBody> maps); @Streaming @GET Observable<ResponseBody> downloadFile(@Url String fileUrl); }
上面新增了几个常用的请求方法
第一个只是普通写法的列子, url ,请求头,参数都是写死的。 不建议这么做
第二,三个分别是Get 和POST请求,method Url,headers, body参数都可以动态外部传入。
四和 五是单文件/图片和多文件/图片上传
最后是文件下载
如果你觉得麻烦 可以用T代替,技术不到位的悠着点哈
@GET() <T> Observable<ResponseBody> executeGet( @Url String url, @QueryMap Map<String, T> maps);
构建基础拦截器
用来设置基础header,这里是通过MAP键值对来构建,将heder加入到Request中。/** * BaseInterceptor,use set okhttp call header * Created by Tamic on 2016-06-30. */ public class BaseInterceptor implements Interceptor{ private Map<String, String> headers; public BaseInterceptor(Map<String, String> headers) { this.headers = headers; } @Override public Response intercept(Chain chain) throws IOException { Request.Builder builder = chain.request() .newBuilder(); if (headers != null && headers.size() > 0) { Set<String> keys = headers.keySet(); for (String headerKey : keys) { builder.addHeader(headerKey, headers.get(headerKey)).build(); } } return chain.proceed(builder.build()); } }
构建Cookie管理者
用来管理cookie, 储存cookie的store这里不再重复说明,具体列子请见:构建RetrofitClient客户端.
今天重要的环节来了,RetrofitClient主要负责创建具体Retrofit,和调度分发请求。设置格式工厂。添加cookie同步,构建OkHttpClient,添加BaseUrl,对加密证书https我没做加入,希望读者参考我的本系列文章自行加入,因为我不喜欢升伸手党。/** * RetrofitClient * Created by Tamic on 2016-06-15. */ public class RetrofitClient { private static final int DEFAULT_TIMEOUT = 5; private ApiService apiService; private OkHttpClient okHttpClient; public static String baseUrl = ApiService.Base_URL; private static Context mContext; private static RetrofitClient sNewInstance; private static class SingletonHolder { private static RetrofitClient INSTANCE = new RetrofitClient( mContext); } public static RetrofitClient getInstance(Context context) { if (context != null) { Log.v("RetrofitClient", DevUtil.isDebug() + ""); mContext = context; } return SingletonHolder.INSTANCE; } public static RetrofitClient getInstance(Context context, String url) { if (context != null) { mContext = context; } sNewInstance = new RetrofitClient(context, url); return sNewInstance; } private RetrofitClient(Context context) { this(context, null); } private RetrofitClient(Context context, String url) { if (TextUtils.isEmpty(url)) { url = baseUrl; } okHttpClient = new OkHttpClient.Builder() .addNetworkInterceptor( new HttpLoggingInterceptor().setLevel(HttpLoggingInterceptor.Level.HEADERS)) .cookieJar(new NovateCookieManger(context)) .addInterceptor(new BaseInterceptor(mContext)) .connectTimeout(DEFAULT_TIMEOUT, TimeUnit.SECONDS) .build(); Retrofit retrofit = new Retrofit.Builder() .client(okHttpClient) .addConverterFactory(GsonConverterFactory.create()) .addCallAdapterFactory(RxJavaCallAdapterFactory.create()) .baseUrl(url) .build(); apiService = retrofit.create(ApiService.class); } public void getData(Subscriber<ResponseBody> subscriber, String ip) { apiService.getData(ip) .subscribeOn(Schedulers.io()) .unsubscribeOn(Schedulers.io()) .observeOn(AndroidSchedulers.mainThread()) .subscribe(subscriber); } public void get(String url, Map headers, Map parameters, Subscriber<ResponseBody> subscriber) { apiService.executeGet(url, headers, parameters) .subscribeOn(Schedulers.io()) .unsubscribeOn(Schedulers.io()) .observeOn(AndroidSchedulers.mainThread()) .subscribe(subscriber); } public void post(String url, Map headers, Map parameters, Subscriber<ResponseBody> subscriber) { apiService.executePost(url, headers, parameters) .subscribeOn(Schedulers.io()) .unsubscribeOn(Schedulers.io()) .observeOn(AndroidSchedulers.mainThread()) .subscribe(subscriber); }
}
大家发现上面的指定生产线程和消费线程的步骤有点麻烦,每个api都得进行指定线程,那么可以利用rxJava的转换器写一个
Transformer
Observable.Transformer schedulersTransformer() { return new Observable.Transformer() { @Override public Object call(Object observable) { return ((Observable) observable).subscribeOn(Schedulers.io()) .unsubscribeOn(Schedulers.io()) .observeOn(AndroidSchedulers.mainThread()); } }; }
那么api可以这样优化了:
public Subscription getData(Subscriber<IpResult> subscriber, String ip) { return apiService.getData(ip) .compose(schedulersTransformer()) .subscribe(subscriber); }
调用 RetrofitClient
RetrofitClient.getInstance(MainActivity.this).createBaseApi().getData(new BaseSubscriber<IpResult>(MainActivity.this) { @Override public void onError(ResponeThrowable e) { Log.e("Lyk", e.code + " "+ e.message); Toast.makeText(MainActivity.this, e.message, Toast.LENGTH_LONG).show(); } @Override public void onNext(IpResult responseBody) { Toast.makeText(MainActivity.this, responseBody.toString(), Toast.LENGTH_LONG).show(); } }, "21.22.11.33");
代码很简洁,在用到的地方获取单列直接调用你需要的方法,在RxSubscriber回调中处理你的业务逻辑即可,无需考虑是否在主线程,其他调用方法同上。
很多时候BaseApiService无法满足需求时,Retrofit增加了扩展接口
create来创建你的API,接着调用execute就可以和RxJava关联
//create you APiService MyApiService service = RetrofitClient.getInstance(MainActivity.this).create(MyApiService.class); // execute and add observable RetrofitClient.getInstance(MainActivity.this, "http://lbs.sougu.net.cn/").execute( service.getSougu(), new BaseSubscriber<SouguBean>(MainActivity.this) { @Override public void onError(ResponeThrowable e) { Log.e("Tamic", e.getMessage()); Toast.makeText(MainActivity.this, e.getMessage(), Toast.LENGTH_LONG).show(); } @Override public void onNext(SouguBean souguBean) { Toast.makeText(MainActivity.this, souguBean.toString(), Toast.LENGTH_LONG).show(); } });
总结
本次封装只对retrofit进行了简单封装,很多场景和需求还是存在缺陷,这种单列模式已不符合目前流行的Builder模式,本人已开始进行下一步的封装工作,在这里提前进行下预告:笔者已进行新的框架开发novate,估计下个月就能和大家见面,敬请继续关注!
系列导读:
Retrofit 2.0(一) 超能实践,完美支持Https传输
Retrofit2.0(二) 完美同步Cookie实现免登录
Retrofit 2.0 超能实践(三),轻松实现文件/图片上传
Retrofit 2.0 超能实践(四),完成大文件断点下载
基于Retrofit2.0+RxJava 封装的超好用的RetrofitClient工具类(六)
玩转IOC,教你徒手实现自定义的Retrofit框架(七)
Retrofit,Okhttp对每个Request统一动态添加header和参数(五)
Rxjava和Retrofit 需要掌握的几个实用技巧,缓存问题和统一对有无网络处理问题(八)
Novate:对Retrofit2.0的又一次完美改进加强!(九)
源码 GitHub :https://github.com/NeglectedByBoss/RetrofitClient
简书 :Tamic_码小白
原文地址:
http://www.jianshu.com/p/29c2a9ac5abf
相关文章推荐
- Android跳转系统相机或相册获取图片
- android检测版本更新
- Android PopupWindow水平偏移
- android mCameraInfo.orientation、setDisplayOrientation
- Android中Cursor类的概念和用法
- android应用开发SharedPreferences存储数据的使用方法
- Android RocooFix 热修复框架[1]
- 状态栏(Android)
- java,android定时器(Timer)的使用
- android图片压缩
- Android6.0以上 上传图片时 需要进行权限申请
- Android 玩转IOC,Retfotit源码解析,教你徒手实现自定义的Retrofit框架
- Android TV开发笔记
- Android 玩转IOC,Retfotit源码解析,教你徒手实现自定义的Retrofit框架
- android控件—MuAutoCompleteTextView
- Android Service 服务(一)—— Service
- Android内存泄漏02
- Android launcher 开发笔记(一) 从脸蛋开始
- Android App监听软键盘按键的三种方式与改变软键盘右下角确定键样式
- Android M权限管理机制:Runtime Permission简介