java动态代理

来源:互联网 发布:位面淘宝txt下载 编辑:程序博客网 时间:2024/05/29 15:15


       都知道Retrofit是通过动态代理来生成代理对象作为网络请求的发起者。


       今天就来看下动态代理是怎么操作的。或者说是怎么让一个貌似接口的对象调用它的抽象方法呢?


先来看代码:


       public static void main(String[] args) { Factory factory = new Factory(); Bird bird = factory.create(Bird.class); bird.fly();//输出结果为 flying...}interface Bird { void fly();}

这里代码通过一个Factory实例调用create方法,传入一个接口的class对象就可以返回一个接口的实例,可以调用接口中的方法fly()。


       而在我们的静态代码中并没有一个类去实现了这个Bird接口(完整代码可以看下方)。那么这个对象到底是从哪里来的呢?


完整代码如下:


       public class DynamicProxy { interface Bird { void fly(); } public static void main(String[] args) { Factory factory = new Factory(); Bird bird = factory.create(Bird.class); bird.fly();//输出结果为flying... } static class Factory implements InvocationHandler { public <T> T create(Class<T> target) { return (T) Proxy.newProxyInstance(target.getClassLoader(), new Class[]{target}, this); } public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("flying..."); return null; } }}


       在调用bird.fly()时,输出结果为flying...,很明显,代码中就如同开始所说的,并不存在一个实现了Bird接口的子类,而bird又实实在在调用了fly()方法。唯一的可能就是bird是接口的实例(或者说实现接口的子类的对象)。这里看起来似乎就有些诡异了。


       当然编程没有魔法,这里只是利用到了Java的动态代理,通过Proxy.newProxyInstance()方法生成实现了指定接口的子类,然后返回了这个动态生成类的实例对象。


       这个子类在调用接口中的方法时,其实调用的是InvocationHandler的invoke()方法。在此方法中会有对应参数的回调,可以根据这些参数做出合适的拦截/增强等操作。

       都知道Retrofit是通过动态代理来生成代理对象作为网络请求的发起者。


       今天就来看下动态代理是怎么操作的。或者说是怎么让一个貌似接口的对象调用它的抽象方法呢?


先来看代码:


       public static void main(String[] args) { Factory factory = new Factory(); Bird bird = factory.create(Bird.class); bird.fly();//输出结果为 flying...}interface Bird { void fly();}

这里代码通过一个Factory实例调用create方法,传入一个接口的class对象就可以返回一个接口的实例,可以调用接口中的方法fly()。


       而在我们的静态代码中并没有一个类去实现了这个Bird接口(完整代码可以看下方)。那么这个对象到底是从哪里来的呢?


完整代码如下:


       public class DynamicProxy { interface Bird { void fly(); } public static void main(String[] args) { Factory factory = new Factory(); Bird bird = factory.create(Bird.class); bird.fly();//输出结果为flying... } static class Factory implements InvocationHandler { public <T> T create(Class<T> target) { return (T) Proxy.newProxyInstance(target.getClassLoader(), new Class[]{target}, this); } public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { System.out.println("flying..."); return null; } }}


       在调用bird.fly()时,输出结果为flying...,很明显,代码中就如同开始所说的,并不存在一个实现了Bird接口的子类,而bird又实实在在调用了fly()方法。唯一的可能就是bird是接口的实例(或者说实现接口的子类的对象)。这里看起来似乎就有些诡异了。


       当然编程没有魔法,这里只是利用到了Java的动态代理,通过Proxy.newProxyInstance()方法生成实现了指定接口的子类,然后返回了这个动态生成类的实例对象。


       这个子类在调用接口中的方法时,其实调用的是InvocationHandler的invoke()方法。在此方法中会有对应参数的回调,可以根据这些参数做出合适的拦截/增强等操作。

原创粉丝点击