从 Retrofit 源码学习 Java 的动态代理的使用

   2016-11-13 0
核心提示:Retrofit 是当前 Android 最流行的 HTTP 网络库之一了,其使用方式比较特殊,是通过定义一个接口类,通过给接口中方法和方法参数添加注解的方式来定义网络请求接口。这种风格下定义一个网络接口变得很简单。不过 Retrofit 是如何使用一个接口的 Class 创建出

Retrofit 是当前 Android 最流行的 HTTP 网络库之一了,其使用方式比较特殊,是通过定义一个接口类,通过给接口中方法和方法参数添加注解的方式来定义网络请求接口。这种风格下定义一个网络接口变得很简单。不过 Retrofit 是如何使用一个接口的 Class 创建出来实现了该接口的对象呢?最近因为工作原因想封装项目中的网络请求部分,在解决获取泛型嵌套问题的时候,一直没有找到比较理想的方案,所以拜读了 Retrofit 的源码看看这个明星网络库是如何实现这一黑科技的。

Java 的动态代理机制

在 Retrofit 2.0 中,create 方法中是这样定义的:

public<T>Tcreate(finalClass<T> service){
 Utils.validateServiceInterface(service);
if(validateEagerly) {
 eagerlyValidateMethods(service);
 }
return(T) Proxy.newProxyInstance(service.getClassLoader(),newClass<?>[] { service },
newInvocationHandler() {
privatefinalPlatform platform = Platform.get();

@OverridepublicObjectinvoke(Object proxy, Method method, Object... args)
throwsThrowable {
//...
 }
 });
 }

前两段的 Utils.validateServiceInterface(service)eagerlyValidateMethods(service) 从命名可以看出是验证传入接口合法性的,所以跳过。

接下来调用了一个关键方法 Proxy.newProxyInstance() ,这个方法就是创建一个代理实例的方法,该方法需要三个参数,ClassLoader loader, Class<?>[] interfaces, InvocationHandler h,其中 interfaces 放置了我们传入的接口 class,InvocationHandler 也是一个接口,只有一个回调方法,也回调了三个参数:Object proxy, Method method, Object… args:其中 proxy 就是构造出来的代理实例,使用 proxy 调用一个方法时,这里传进来的 mthod 就是该方法,而 args 就是这个参数。

空说无凭,使用代码测试一下:

首先假装已经有个一个 retrofit,来定义一个接口类

interfaceApi{
@POST("http://example.com")
CallAdapter<User> fetchUser();

@POST("http://example.com")
CallAdapter<String> login(@Param("username")String username, @Param("password")String password);
}

跟真的一样。

接下来 retrofit 调用了一个 create 方法返回的 Api 对象就让它能 fetchUser,能 login 了。那么可以想象一下,如果需要实现这些事情,它需要从这个接口定义中获取什么?

首先是 @POST,它让 retrofit 知道了这个请求是使用 post,然后里面的值是一个 PATH(这里为了实现简单直接放上了完整的路径),通过它和构造 retrofit 对象时的 baseUrl 就得到了接口的完整 URL。再后面需要知道定义的参数,每个注解对应的是一个参数,同时注解里标注了这个参数的名字,所以这里获取的应该是所有参数注解值和参数的值。知道这些也就万事俱备了,接下来 retrofit 只要把请求构造出来交给 OkHttp 然后等待返回结果再传给返回值。

为了获得上面所说的信息,在 invoke 方法里尝试以下代码:

@Override
publicObjectinvoke(Object proxy, Method method, Object[] args)throwsThrowable{
// 打印调用的请求方法
 System.out.println("调用的请求方法:"+ method.getName());

// 这里获取了方法的注解,retrofit 就是在这里判断使用的哪种请求方式,以及获取 path
 POST annotation = method.getAnnotation(POST.class);
 String postUrl = annotation.value();
 System.out.println("Url: "+ postUrl);

// 非法判断是个麻烦的事情,这里我们只实现例子中 api 的解析,所以以下代码不进行异常判断。
if(args !=null&& args.length !=0) {
 List<String> params = newArrayList<>();
// 获取所有参数的所有注解。得到的是一个二维数组,前面一个下标代表的是第几个参数,后面下标代表的是这个参数的所有注解。这里我们每个参数只定义了一个注解,所以直接取了[0]
 Annotation[][] annotations = method.getParameterAnnotations();
for(inti =0; i < args.length; i++) {
 Annotation[] paramsAnnotation = annotations[i];
// 取到 Param 注解中的值,这个值在 retrofit 中一般是参数的键,而 args[i] 则是参数的值
 params.add(String.format("%s: %s", ((Param) paramsAnnotation[0]).value(), args[i].toString()));
 }
// 打印所有参数
 System.out.printf("Params: %s\n", params.toString());
 }

// 下面的内容可能更加关键:如何获得返回值类型。
// 我们用了默认的 CallAdapter 来实现

// 首先获得 CallAdapter 里定义的泛型,也就是我们最终需要的数据 class
 Type genericReturnType = method.getGenericReturnType(); // 这里获得的是最外层,也就是 CallAdapter
// 这里偷懒使用了 Retrofit 工具类中的方法来获取 CallAdapter 的泛型。这边为了显示方便又强制转换成了 Class 对象,实际上例如 Gson,直接传入 Type 就可以解析出实体了不需要再转换为 Class
 Class resultClass = (Class) getParameterUpperBound(0, (ParameterizedType) genericReturnType);
 System.out.println("return: type"+ resultClass.getSimpleName());
// 返回调用需要的实体。这里为了篇幅没有再模拟网络请求,而只是调用了 newInstance 来创建一个对象回调出去。
return(CallAdapter<T>) call -> {
if(call !=null) {
try{
//noinspection unchecked
 call.call((T) resultClass.newInstance());
 } catch(InstantiationException | IllegalAccessException e) {
 e.printStackTrace();
 }
 }
 };
}

好了验证一下结果:

Api print = create(Api.class);
print.fetchUser().call(user -> {
 String result = "print.fetchUser().call => "+ user.getClass().getSimpleName();
 System.out.println(result);
});
print.login("user1","passw0rd").call(s -> {
 String result = "print.login(\"user1\", \"passw0rd\").call => "+ s.getClass().getSimpleName();
 System.out.println(result);
});

打印结果:

调用请求方法:fetchUser
Url: http://example.com
return: typeUser
print.fetchUser().call => User
调用请求方法:login
Url: http://example.com
Params: [username: user1, password: passw0rd]
return: typeString
print.login("user1", "passw0rd").call => String

完全符合预期。有了这些参数,Retrofit 就可以统一封装网络请求进行处理后返回给用户定义的方法了。

 
标签: Retrofit Java
反对 0举报 0 评论 0
 

免责声明:本文仅代表作者个人观点,与乐学笔记(本网)无关。其原创性以及文中陈述文字和内容未经本站证实,对本文以及其中全部或者部分内容、文字的真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
    本网站有部分内容均转载自其它媒体,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责,若因作品内容、知识产权、版权和其他问题,请及时提供相关证明等材料并与我们留言联系,本网站将在规定时间内给予删除等相关处理.

  • [Java] Retrofit2.0 如何进行GBK编码
    对Retrofit + OkHttp还不熟悉的人可以点传送门,先看下这两个东西的使用。Retrofit:https://github.com/square/retrofitOkHttp:https://github.com/square/okhttp分析接口文档要求Post请求,字段使用GBK编码我们先按照Retrofit的规范和接口文档来写接口: @PO
    12-23 RetrofitJava
  • Android常用的开源项目及其比较系列-Retrofit进
    上一篇我们谈了谈Androiod开源项目的网络框架, 比较了它们之间的优缺点,原文在这里。今天我们着重谈谈Retrofit框架如何更友好的使用,本着提出问题解决问题的原则,也为大家以后解决问题提供基本思路。目前都有哪些问题?根据官方Demo, 简单使用是这么样的
  • Android MVP架构实践
    Android MVP架构实践
    首先声明一下,没有完美的架构,只要适合自己的项目,那就是最好的架构。本例子是MVP + Retrofit + RxJava结合的例子,但本文的重点在于讲解MVP架构,所以涉及Retrofit和RxJava的部分将直接略过,默认读者已了解这两部分内容,如有需要,请自行查阅相关资料,
  • 打造企业级网络请求框架集合 retrofit+gson+mvp
    打造企业级网络请求框架集合 retrofit+gson+m
    本文是企业级网络框架第二篇主要讲MVP模式和Gson在Retrofit网络请求框架下的使用方式。(已更新为一篇) 对MVP不了解的请看梦之鬼索MVP模式在Android中的设计和实现 http://blog.csdn.net/androidmsky/article/details/52248797 对Retrofit还不了解的情看 打
    10-31 RetrofitGson
  • Retrofit分析之框架设计艺术
    Retrofit分析之框架设计艺术
    Retrofit使用者会觉得接口+注解方式去写网络请求很吊。但当你真正的去看它的源码,会被它独特,漂亮的解耦方式所吸引,整个结构运用了动态代理,策略模式,Builder模式,工厂等设计模式。Retrofit v2.1基于Okhttp3,可以说是对okhttp进行二次封装。先感受一下
    10-07 Retrofit
  • Android Retrofit框架解析
    随着Google对HttpClient的摒弃,和Volley的逐渐没落,OkHttp开始异军突起,而Retrofit则对okHttp进行了强制依赖。Retrofit也是Square公司开发的一款针对Android网络请求的框架,其实质就是对okHttp的封装,使用面向接口的方式进行网络请求,利用动态生成的代理类
    10-04 Retrofit
  • Android网络开源库-Retrofit(五)简易封装
    1.前言Rrtrofit的扩展性很强,如果对retrofit不熟悉的话,是很难应对各种各样的需求的。因此,在这里,做一下简单的封装。主要为了下面三点需求:使用简单加密处理错误处理2.怎样才能简单使用为了简单粗暴,我做了以下工作。使用单例Retrofit引入RxJava在这里
  • Android网络开源库-Retrofit(四)文件相关
    以前写过一些retrofit的相关文章,当时只是自己学习研究的,最近项目,加入了retrofit,因此遇到了一些问题,需要记录一下。1.前言在以前,写过retrofit上传文件相关,但是,需求总是变化的。前面的,介绍了上传进度的监听,但是,那时候是监听单文件进度。虽
  • 急速开发系列——Retrofit 响应数据及异常处理
    今天我们来谈谈客户端对通讯协议的处理,主要分为三部分:约定响应数据格式,响应数据的自动映射以及错误处理三部分。由于数据协议采用json的居多,因此我们在此基础上进行说明。约定响应数据格式协议格式通常来说,你拿到的设计文档中会存在通信协议的说明,
    09-29 RetrofitGson
  • 使用Android API最佳实践
    使用Android API最佳实践
    写在前面 现在,Android应用程序中集成第三方API已十分流行。应用程序都有自己的网络操作和缓存处理机制,但是大部分比较脆弱,没有针对网络糟糕情况进行优化。感谢 Square lnc这家有创新精神的公司,将信用卡商业交易带到手机上。现在有了一系列高质量开源库
    09-20 APIRetrofit
点击排行