浅谈Retrofit封装-让框架更加简洁易用

本文详细介绍了如何对Retrofit进行封装,以实现更简洁易用的网络请求接口。包括异步请求回调、参数配置、链式调用、支持RxJava、取消请求、下载大文件等功能,并提供了完整的示例项目链接,旨在帮助开发者提高代码可维护性和可读性。
摘要由CSDN通过智能技术生成

尊重他人的劳动成果,转载请标明出处:http://blog.csdn.net/gengqiquan/article/details/52329259, 本文出自:【gengqiquan的博客】

不知不觉在在这家公司快三个月了,刚接手项目的时候是真的被里面的代码恶心到了,网络请求用的原生的httpURLconnection。这本来什么,关键是根本就没有进行任何封装。activity里面充斥着大量的handler,要找个oncreated()函数先得把类拉到一半,那种感觉岂止酸爽。由于项目周期紧张。没时间去大规模的重构,新框架只能在新功能里写。采用了retrofit,初期做了简单的封装,断断续续一段时间的优化整理。现在分享出来给大家。

为了方便大家,会形成依赖库。引入和调用方式请看文章底部的github文档
本次封装需要做到的

1支持异步请求统一回调
2参数可配置
3链式调用
4支持基本get和post封装
5支持rxjava返回
6支持一个接口Service 对应一种请求类型,而不是每个API都需要去定义一个接口方法
7支持取消单个请求
8支持请求打标签,退出页面时取消当前页面所有请求
9支持动态配置缓存,缓存与具体的请求接口绑定,
10支持追加统一参数,登录信息token、设备号等
11支持接口版本号配置
12流式下载文件,取消下载请求

下面我们来看下封装过程,博客底部会贴出示例项目地址
retrofit独树一帜的把请求采用了接口,方法和注解参数(parameter annotations)来声明式定义一个请求应该如何被创建的方式。
像这样

public interface GitHub {
   
      @GET("/repos/{owner}/{repo}/contributors")
        List<Contributor> contributors(@Path("owner")
        String owner, @Path("repo")
        String repo);

然后去实例化并且调用请求

  GitHub github = restAdapter.create(GitHub.class);
        // Fetch and print a list of the contributors to this library.
        List<Contributor> contributors = github.contributors("square",
                "retrofit");

先不谈retrofit到底做了多少优化、性能上有多少提升。光是这样的调用方式我就受不了。我特么得写多少个 像上面的GitHub 一样的Service,就算是把url注解方法都写在一个里面,那也得多少行?一个项目六七十行应该是没什么问题的了。嗯。反正我会看疯了的。而且这样的一种调用方式是直接面向框架层的,以后万一我们换框架了怎么办?代码挨个找出来全换一边?你疯不疯?
那我们有没有什么办法改变他?很简单,我们在框架外面再套一层通用框架,作为框架的设计者,我们应该让调用者知道怎么调用就可以了,而不应该让调用者去考虑底层实现的细节。
好在retrofit提供了Url 参数替换注解@Url String url,通过这个注解我们可以动态的设置请求的url。
下面列出一些简单的参数注解

@Url 替换url
@QueryMap  替换url中查询参数
@Header  替换header
@FieldMap 替换post请求body中参数
@FormUrlEncoded post请求需要加的方法注解
@POST() 标示该方法为post请求
@GET() 标示该方法为get请求

了解了这些注解这样我们就可以将我们项目的请求变成几个基本的方法,由于我的项目的服务端返回的基本格式不是固定的,而GsonConverterFactory,对于retrofit的返回类型要求只能是: 具体对象、jsonObject、jsonArray这三种 ,为了避免每个请求写两个方法,这里我自己写了个返回类型为String的解析器,参考 Retrofit自定义Converter之StringConverterFactory

这样我们的返回类型都是String

public interface RetrofitHttpService {


    @GET()
    Call<String> get(@Url String url, @QueryMap Map<String, String> params, @Header("Cache-Time") String time);
    @FormUrlEncoded

    @POST()
    Call<String> post(@Url String url, @FieldMap Map<String, String> params, @Header("Cache-Time") String time);


}

如果你项目的返回请求外层有固定的格式可以把 Call<JsonObject>替换成Call<Model>,这里的model就是你的基础数据返回类型。
如果你要使用rxjava的话需要额外两个注解方法


    @GET()
    Observable<String> Obget(@Url String url, @QueryMap Map<String, String> params, @Header("Cache-Time") String time);

    @FormUrlEncoded
    @POST()
    Observable<String> Obpost(@Url String url, @FieldMap Map<String, String> params, @Header("Cache-Time") String time);

构建网络数据请求类,这里为了方便不同的项目使用,用一个builder类来构建单例

public class HttpUtil {
    private static volatile HttpUtil mInstance;
    private static volatile RetrofitHttpService mService;
    private Context mAppliactionContext;
    private static String mVersionApi;
    private ParamsInterceptor mParamsInterceptor;

    //构造函数私有,不允许外部调用
    private HttpUtil(RetrofitHttpService mService, Context mAppliactionContext, String mVersionApi, ParamsInterceptor mParamsInterceptor) {
        this.mService = mService;
        this.mAppliactionContext = mAppliactionContext;
        this.mVersionApi = mVersionApi;
        this.mParamsInterceptor = mParamsInterceptor;
    }

    public static class SingletonBuilder {
        private Context appliactionContext;
        private String baseUrl;
        private List<String> servers = new ArrayList<>();
        private String versionApi;
        private ParamsInterceptor paramsInterceptor;
        private List<Converter.Factory> converterFactories = new ArrayList<>();
        private List<CallAdapter.Factory> adapterFactories = new ArrayList<>();
        OkHttpClient client;

        public SingletonBuilder(Context context) {
            try {
  //防止传入的是activity的上下文
                Activity activity = (Activity) context;
                appliactionContext = context.getApplicationContext();
            } catch (Exception e) {
                e.printStackTrace();
                appliactionContext = context;
            }
        }

        public SingletonBuilder client(OkHttpClient client) {
            this.client = client;
            return this;
        }

        public SingletonBuilder versionApi(String versionApi) {
            this.versionApi = versionApi;
            return this;
        }

        public SingletonBuilder paramsInterceptor(ParamsInterceptor interceptor) {
            this.paramsInterceptor = interceptor;
            return this;
        }

        public SingletonBuilder baseUrl(String baseUrl) {
            this.baseUrl = baseUrl;
            return this;
        }

        public SingletonBuilder addServerUrl(String ipUrl) {
            this.servers.add(ipUrl);
            return this;
        }

        public SingletonBuilder serverUrls(List<String> servers) {
            this.servers = servers;
            return this;
        }

        public SingletonBuilder addConverterFactory(Converter.Factory factory) {
            this.converterFactories.add(factory);
            return this;
        }

        public SingletonBuilder addCallFactory(CallAdapter.Factory factory) {
            this.adapterFactories.add(factory);
            return this;
        }

        public HttpUtil build() {
            if (checkNULL(this.baseUrl)) {
                throw new NullPointerException("BASE_URL can not be null");
            }
            if (converterFactories.size() == 0) {
                converterFactories.add(StringConverterFactory.create())
  • 12
    点赞
  • 31
    收藏
    觉得还不错? 一键收藏
  • 14
    评论
评论 14
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值