专栏首页JavaQ深入浅出JDK动态代理(一)

深入浅出JDK动态代理(一)

何为代理

代理,即代替主角完成一些额外的事情。例如,明星都有经纪人,明星参演电影之前,经纪人作为明星的代理人和出资方洽谈片酬、排期等,而真正参与拍戏的还是明星本人,明星拍完戏后,由经纪人代理明星去清算片酬等。Java中的代理机制就是在目标方法执行前后执行一些额外的操作,如安全检查、记录日志等,Java中的代理分为静态代理和动态代理。

静态代理

首先看一下静态代理,直接上代码,代码模拟了登录操作。

public interface LoginService { void login(); } public class LoginServiceImpl implements LoginService { @Override public void login() { System.out.println("login"); } } public class LoginServiceProxy implements LoginService { private LoginService loginService; public LoginServiceProxy(LoginService loginService) { this.loginService = loginService; } @Override public void login() { beforeLogin(); loginService.login(); afterLogin(); } private void beforeLogin() { System.out.println("before login"); } private void afterLogin() { System.out.println("after login"); } } public class Client { @Test public void test() { LoginService loginService = new LoginServiceImpl(); LoginService loginServiceProxy = new LoginServiceProxy(loginService); loginServiceProxy.login(); } }

输出结果如下:

before login login after login

上面代码实现的静态代理很容易理解,使用聚合方式,在登录操作前后执行额外的操作。静态代理方式可以看得到具体代理类的代码,且代码由程序员编写,在编译之后会生成相应的class文件。使用静态代理方式的缺点,如果需要对LoginService接口中有N个方法都代理,则需要在代理类中创建N个代理方法,并且需要编写重复的代理操作代码。

概念解释

目标接口,即对目标操作的抽象,如LoginService。

目标类,即目标接口的实现类,如LoginServiceImpl。

目标对象,即目标类的实例。

代理类,即目标类的代理,如LoginServiceProxy。

代理对象,即代理类的实例。

动态代理

动态代理,即在运行时根据目标接口动态生成的代理类。动态代理方式生成的代理类在编译后不会生成实际的class文件,而是在运行时动态生成类字节码,并加载到JVM中使用。下面使用JDK的动态代理机制模拟登录操作,具体代码如下。

public interface LoginService { void login(); } public class LoginServiceImpl implements LoginService { @Override public void login() { System.out.println("login"); } } public class ProxyInvocationHandler implements InvocationHandler { private LoginService loginService; public ProxyInvocationHandler(LoginService loginService) { this.loginService = loginService; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { beforeLogin(); Object invokeResult = method.invoke(loginService, args); afterLogin(); return invokeResult; } private void beforeLogin() { System.out.println("before login"); } private void afterLogin() { System.out.println("after login"); } } public class Client { @Test public void test() { LoginService loginService = new LoginServiceImpl(); ProxyInvocationHandler proxyInvocationHandler = new ProxyInvocationHandler(loginService); LoginService loginServiceProxy = (LoginService) Proxy.newProxyInstance(loginService.getClass().getClassLoader(), loginService.getClass().getInterfaces(), proxyInvocationHandler); loginServiceProxy.login(); createProxyClassFile(); } public static void createProxyClassFile() { String name = "LoginServiceProxy"; byte[] data = ProxyGenerator.generateProxyClass(name, new Class[]{LoginService.class}); try { FileOutputStream out = new FileOutputStream("/Users/" + name + ".class"); out.write(data); out.close(); } catch (Exception e) { e.printStackTrace(); } } }

输出结果如下:

before login login after login

JDK动态代理方式实现代理的步骤如下:

1.编写目标接口;

2.编写目标类实现目标接口,实现目标方法的具体逻辑;

3.编写一个代理处理器类实现InvocationHandler接口,重写invoke方法,用于指定运行时将生成的代理类需要完成的具体操作,包括beforeLogin和afterLogin。代理对象调用任何代理方法时都会调用这个invoke方法;

4.创建代理对象,使用代理对象调用代理方法。

上面的步骤中主要涉及以下两个类:

java.lang.reflect.InvocationHandler和java.lang.reflect.Proxy

InvocationHandler是一个接口,代理类的调用处理器,每个代理对象都具有一个关联的调用处理器,用于指定动态生成的代理类需要完成的具体操作。该接口中有一个invoke方法,代理对象调用任何目标接口的方法时都会调用这个invoke方法,在这个方法中进行目标类的目标方法的调用。

Proxy提供静态方法用于创建动态代理类和代理类实例,同时,使用它提供的方法创建的代理类都是它的子类。这个类中主要关注newProxyInstance方法,该方法用于创建代理类对象,方法声明如下:

public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)

loader参数用于指示使用哪个类加载器加载这个代理类;interfaces表示代理类实现的接口列表;h表示使用哪个调用处理器。

后续文章《深入浅出JDK动态代理(二)》会深入源码分析JDK动态代理生成的代理类是什么样,为什么调用代理类的任何方法时都一定会调用invoke方法,值得期待!

本文分享自微信公众号 - JavaQ(Java-Q),作者:未来先生

原文出处及转载信息见文内详细说明,如有侵权,请联系 yunjia_community@tencent.com 删除。

原始发表时间:2016-10-01

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • 深入理解Spring系列之一:开篇

    Spring经过大神们的构思、编码,日积月累而来,所以,对其代码的理解也不是一朝一夕就能快速完成的。源码学习是枯燥的,需要坚持!坚持!坚持!当然也需要技巧,第一...

    JavaQ
  • 如何去除代码中的多次if而引发的一连串面试问题

    小白:不是,真正的工厂模式有两种:工厂方法和抽象工厂。工厂方法使用继承,首先定义一个抽象父类工厂,然后定义子类工厂,把工厂要创建的对象委托给子工厂类,子工厂类实...

    JavaQ
  • 高并发编程-AQS深入解析

    AbstractQueuedSynchronizer简称AQS,它是java.util.concurrent包下CountDownLatch/FutureTas...

    JavaQ
  • ABP框架学习之——授权(Authorization)

    易兒善
  • 六大原则 动态代理

    代理模式中,中介类与委托类的关系,在编写中介类时就确定了。中介类会控制对委托类对象的访问,。 装饰者模式的装饰关系是在运行时才知道,比如下面的代码。

    平凡的学生族
  • 傀儡政权之代理模式

    代理模式应用非常广泛,特别java领域的Spring框架,可以说把代理模式运用到极致。其中Spring的代理又分JDK动态代理和cglib动态代理。这类不打算深...

    java乐园
  • 看完此文你就学会了代理模式

    一个委托类,一个代理类,实现相同的接口,通常是在代理类中有一个委托类的对象,代理类并不会真正的执行方法,只是在委托类执行方法之前或之后提供一些服务。比如账户验证...

    秃头哥编程
  • 【Java设计模式】之代理模式

    代理模式是Java常见的设计模式之一。所谓代理模式是指客户端并不直接调用实际的对象,而是通过调用代理,来间接的调用实际的对象。 为什么要采用这种间接的形式来调...

    用户5640963
  • Intent传递对象的两种方法

    听着music睡
  • 秒发4000万数据的缓存方案以及消息优先消费

    我发现在mq并发进入消费时并不能保证消息的消费顺序,此时如果同时一万线程对一个生产者一个消费者的一个队列业务互斥进行消费,此时的消费顺序是无序的,同一时刻会...

    疯狂的KK

扫码关注云+社区

领取腾讯云代金券