从动态代理角度看Retrofit

这是我在csdn发表的一篇技术博文:https://blog.csdn.net/sinat_23092639/article/details/102237404

1.Retrofit简介

retrofit(Retrofit官方)已经诞生好几年了,从诞生开始一直都是Android应用开发最流行的网络请求框架,准确来说,是网络请求框架一个巧妙的包装。

正如官网所说,retrofit最大的特点,在于可以用一个Java interface通过注解去表示一个Http请求。
1.比如定义一个GET请求的Java interface:

public interface GitHubService {
  @GET("users/{user}/repos")
  Call<List<Repo>> listRepos(@Path("user") String user);
}
    

2.然后 通过Retrofit创建一个GitHubService实例:

Retrofit retrofit = new Retrofit.Builder()
    .baseUrl("https://api.github.com/")
    .build();

GitHubService service = retrofit.create(GitHubService.class);

3.再获取GitHubService实例对应的Call,通过Call就可以发起Http请求了:

Call<List<Repo>> repos = service.listRepos("octocat");
 call.enqueue(new Callback<List<Repo>>()
        {
            @Override
            public void onResponse(Call<List<Repo>> call, Response<List<Repo>> response)
            {
                Log.e(TAG, "normalGet:" + response.body() + "");
            }

            @Override
            public void onFailure(Call<List<Repo>> call, Throwable t)
            {

            }
        });

Retrofit底层默认是用OkHttp,正如前面所说它只是一个包装,我们之所以使用Retrofit的目的是为了简化和优化代码的调用,优化开发效率。

2.动态代理机制分析

关于Retrofit的使用不是本文的重点,本文重点谈谈Retrofit中的动态代理机制。

动态代理可以说是Retrofit中的核心机制,其使用体现在上面第2步,即生成一个我们定义的请求Java interface的请求实例,将原来一个普通的Java Interface转化为一个可以进行Http请求的对象

要了解Retrofit中动态代理机制的运用,还是让我们从动态代理本身讲起吧。

建议先看下官网关于动态代理的基本说明,毕竟是权威资料而且详细:
Oracle动态代理说明
我们先来看一个最简单的动态代理例子,感受下通过动态代理将一个普通Java接口转化为一个代理对象:

一.首先创建一个Java Interface,对应上文的Retrofit请求接口GitHubService :

public interface Book {
    String read(String s);
}

二.通过Proxy生成Book 的代理对象的Class对象

Class bookProxyClass = Proxy.getProxyClass(Book.class.getClassLoader(),Book.class);

我们知道,一个Java Interface是不可以直接创建一个对象的,所以动态代理所做的是在运行时生成一个实现了该Interface的类的Class对象。

Proxy的getProxyClass方法注释已经解释清楚了:
Returns the {@code java.lang.Class} object for a proxy class given a class loader and an array of interfaces. The proxy class will be defined by the specified class loader and will implement all of the supplied interfaces. If any of the given interfaces is non-public, the proxy class will be non-public. If a proxy class for the same permutation of interfaces has already been defined by the
class loader, then the existing proxy class will be returned; otherwise,a proxy class for those interfaces will be generated dynamically and defined by the class loader.

简单来说,就是在运行时生成一个代理Class二进制流,并通过传入的ClassLoader去加载成一个代理Class对象,该Class实现了传入的第二个参数对应的Interface。

具体怎么生成Class二进制流就不在这里讲了,这里看下生成的Class具体啥样子:

通过添加一行代码:

System.getProperties().put("sun.misc.ProxyGenerator.saveGeneratedFiles","true");

可以将运行时生成的动态代理Class文件保存下来(关于Class文件,详细可以看我这篇博文:从字节码角度仔细剖析一个HelloWorld程序

用Idea打开该Class文件(自动进行了反编译):

public final class $Proxy0 extends Proxy implements Book {
    private static Method m1;
    private static Method m3;
    private static Method m2;
    private static Method m0;

    public $Proxy0(InvocationHandler var1) throws  {
        super(var1);
    }

    public final boolean equals(Object var1) throws  {
        try {
            return (Boolean)super.h.invoke(this, m1, new Object[]{var1});
        } catch (RuntimeException | Error var3) {
            throw var3;
        } catch (Throwable var4) {
            throw new UndeclaredThrowableException(var4);
        }
    }

    public final String read(String var1) throws  {
        try {
            return (String)super.h.invoke(this, m3, new Object[]{var1});
        } catch (RuntimeException | Error var3) {
            throw var3;
        } catch (Throwable var4) {
            throw new UndeclaredThrowableException(var4);
        }
    }

    public final String toString() throws  {
        try {
            return (String)super.h.invoke(this, m2, (Object[])null);
        } catch (RuntimeException | Error var2) {
            throw var2;
        } catch (Throwable var3) {
            throw new UndeclaredThrowableException(var3);
        }
    }

    public final int hashCode() throws  {
        try {
            return (Integer)super.h.invoke(this, m0, (Object[])null);
        } catch (RuntimeException | Error var2) {
            throw var2;
        } catch (Throwable var3) {
            throw new UndeclaredThrowableException(var3);
        }
    }

    static {
        try {
            m1 = Class.forName("java.lang.Object").getMethod("equals", Class.forName("java.lang.Object"));
            m3 = Class.forName("proxy.Book").getMethod("read", Class.forName("java.lang.String"));
            m2 = Class.forName("java.lang.Object").getMethod("toString");
            m0 = Class.forName("java.lang.Object").getMethod("hashCode");
        } catch (NoSuchMethodException var2) {
            throw new NoSuchMethodError(var2.getMessage());
        } catch (ClassNotFoundException var3) {
            throw new NoClassDefFoundError(var3.getMessage());
        }
    }
}

代码很简单,从该代码可以提取以下几个要点:

1.代理类Proxy0继承Proxy,实现了我们的Book接口。
2.用4个Method引用保存equals、hashCode、toString以及Book接口的read方法的Method对象
3.在equals、hashCode、toString以及Book接口的read方法的实现都为调用super.h的invoke方法并传入2中对应的Method对象和参数。
4.通过查看Proxy的代码可知super.h为一个InvocationHandler 引用。而该InvocationHandler 引用就是从Proxy0的构造方法中传入。

所以说白了,该代理类就是构造方法传入的InvocationHandler 对象的代理,无论调用什么方法,都会调用InvocationHandler 对象的invoke方法并传入方法对应的Method对象和参数。

三:通过反射从Class对象创建对应的具体Book代理的实例对象:

有了代理类的Class对象,就可以通过反射创建对象了。

Constructor constructor = bookProxyClass.getConstructor(InvocationHandler.class);
//注意到上文的代理类只有一个有参构造方法,参数为InvocationHandler类型
Book bookProxy  = (Book) constructor.newInstance(new InvocationHandler() {
                @Override
                public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                    String s = (String) args[0];
                    return "i am proxy of " + s;
                }
            });

这里终于得到了通过Book转换得到的对象bookProxy。

四.调用Book代理的实例对象的代理方法(即Book接口的方法)

 String a = bookProxy.read("红楼梦");
 System.out.println(a);

控制台打印出结果:
i am proxy of 红楼梦

正是三传入的InvocationHandler对象invoke方法指定的逻辑。

以上2,3步其实可以合成一步:

 Book bookProxy  = Proxy.newProxyInstance(Book.class.getClassLoader(),new Class[]{Book.class} ,
                new InvocationHandler() {
                    @Override
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                        String s = (String) args[0];
                        return "i am proxy of " + s;
                    }
                });

这就是我们经常见到的动态代理的代码,但是为了解释清楚内在原理流程,所以我拆为了2步。

那么动态代理有什么作用呢?
《深入理解Java虚拟机》一书中给出的答案是:


在这里插入图片描述

结合Retrofit来讲,就是在具体的Http请求还未知的情况下,就确定了Http的请求代码。

3.Retrofit中的动态代理

Retrofit源码因为复杂就不在这里详细叙述了,只要明白了动态代理原理,它的流程就很容易明白。

就拿上文的Retrofit例子来说吧,在执行

GitHubService service = retrofit.create(GitHubService.class);

这一行的时候,虚拟机内部生成了代理类的Class二进制流,并且加载到虚拟机中形成代理类的Class对象,再反射得到代理类对象,而该代理类即实现了GitHubService,并且持有了InvocationHandler的引用。

当调用

Call<List<Repo>> repos = service.listRepos("octocat");

就会调用代理对象的listRepos方法,通过上文的分析,它会调用
该InvocationHandler引用的对象的invoke方法,并传入GitHubService的listRepos的Method对象。

InvocationHandler引用的对象的invoke方法会通过该Method对象,得到方法的注解以及参数,得到Http请求的链接、请求方法、请求路径、参数等请求信息,构建一个OkHttp的请求并执行。

4.动手实现一个迷你Retrofit

所谓纸上得来终觉浅,绝知此事要躬行。上文的叙述恐怕可能还是让人听得云里雾里,那么接下来,我就来重点围绕动态代理实现一个迷你的Retrofit。这个迷你Retrofit唯一的功能,就是通过Get请求请求我的csdn首页数据,主要是为了将Retrofit动态代理相关的的主要流程阐述清楚~

1.创建一个Get注解:

@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface Get {
    String value();
}

2.创建Call接口和CallBack回调接口:

这是代理对象请求方法返回的对象,T表示请求的数据类型。和Retrofit的Call对应。

public interface Call<T> {
    void enqueue(CallBack<T> callBack);
}

这是请求的回调接口,T表示请求的数据类型,和Retrofit的CallBack对应。

public  interface  CallBack<T> {
    void onResponse(T response);
    void onFail(Exception e);
}

3.GetBaiduService接口添加Get注解和Path:
这里为了简单就不添加参数了。

public interface GetBaiduService {
    @Get("sinat_23092639")
    Call<String> read();
}

这里为了简单就直接将path写在注解上了。

4.根据Retrofit源码创建MiniRetrofit:

public class MiniRetrofit {

    String mBaseUrl;

    public MiniRetrofit(String baseUrl) {
        mBaseUrl = baseUrl;
    }

    public <T> T create(final Class<T> service) {
        return (T) Proxy.newProxyInstance(service.getClassLoader(), new Class<?>[]{service},
                new InvocationHandler() {
                    @Override
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                        //如果是equals、toString、hashCode方法则直接调用原来的方法
                        if (method.getDeclaringClass() == Object.class) {
                            return method.invoke(this, args);
                        }

                        Call<?> call = null;
                        //获取被代理的接口的方法注解
                        Annotation[] annotations = method.getDeclaredAnnotations();
                        for (Annotation annotation : annotations) {
                            //如果找到对应的Get注解,则将Get注解中的url和mBaseUrl拼接一起
                            if (annotation instanceof Get) {
                                String path = ((Get) annotation).value();
                                String url = mBaseUrl + "/" + path;
                                System.out.println("url:" + url);
                                //通过OkHttp请求url,并将结果回调到外部
                                final OkHttpClient okHttpClient = new OkHttpClient().newBuilder().build();
                                final Request request = new Request.Builder()
                                        .url(url)
                                        .build();
                                call = new Call<Object>() {
                                    @Override
                                    public void enqueue(final CallBack<Object> callBack) {
                                        okHttpClient.newCall(request).enqueue(new Callback() {
                                            @Override
                                            public void onFailure(@NotNull okhttp3.Call call, @NotNull IOException e) {
                                                callBack.onFail(e);
                                            }

                                            @Override
                                            public void onResponse(@NotNull okhttp3.Call call, @NotNull Response response) throws IOException {
                                                callBack.onResponse(response.body().string());
                                            }
                                        });

                                    }
                                };

                            }
                        }

                        return call;
                    }
                });
    }

    static class Builder {
        String baseUrl;

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

        public MiniRetrofit build() {
            return new MiniRetrofit(baseUrl);
        }
    }
}

5.最后添加最外层的请求调用代码:

MiniRetrofit miniRetrofit = new MiniRetrofit.Builder().baseUrl("https://blog.csdn.net").build();
        GetBaiduService getBaiduService = miniRetrofit.create(GetBaiduService.class);
        Call<String> call = getBaiduService.read();
        call.enqueue(new CallBack<String>() {
            @Override
            public void onResponse(String response) {
                System.out.println("success:"+response);
            }

            @Override
            public void onFail(Exception e) {
                System.out.println(e.toString());
            }
        });

执行程序:


在这里插入图片描述

成功请求到了我的csdn首页数据。

动态代理还有很多用处,比如我之前写过的通过反射实现的仿ButterKnife功能Demo
一文里就讲过通过动态代理巧妙解决控件绑定点击监听回调方法的方案。总的来说,它提供了很大的灵活性,让我们在运行期根据具体情况做具体的应变。

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

推荐阅读更多精彩内容