AOP实现原理之JDK动态代理

男娘i 2022-02-09 14:39 418阅读 0赞

JDK动态代理

JDK动态代理是Spring AOP底层实现的方式之一。

JDK动态代理 对“装饰者”设计模式 简化。使用前提:必须有接口。

JDK动态代理实现

首先我们需要有target目标类,也就是我们需要接口+实现类

首先我们创建一个UserService的接口

  1. public interface UserService {
  2. public void addUser();
  3. public void updateUser();
  4. public void deleteUser();
  5. }

然后还有实现类UserServiceImpl

  1. public class UserServiceImpl implements UserService{
  2. @Override
  3. public void addUser() {
  4. System.out.println("addUser()");
  5. }
  6. @Override
  7. public void updateUser() {
  8. System.out.println("updateUser()");
  9. }
  10. @Override
  11. public void deleteUser() {
  12. System.out.println("deleteUser()");
  13. }
  14. }

然后我们需要切面类:用于存Advice通知 的MyAspect类,也就是存放我们的增强方法

  1. public class MyAspect {
  2. public void before(){
  3. System.out.println("----before()----");
  4. }
  5. public void after(){
  6. System.out.println("----after()----");
  7. }
  8. }

然后我们还需要工厂类MyBeanFactory ,这个类的主要的工作就是将我们的Advice通知(增强的代码)跟我们的target目标类结合起来生成proxy代理对象。

  1. import java.lang.reflect.InvocationHandler;
  2. import java.lang.reflect.Method;
  3. import java.lang.reflect.Proxy;
  4. public class MyBeanFactory {
  5. public static UserService createService(){
  6. //1 目标类
  7. final UserService userService = new UserServiceImpl();
  8. //2切面类
  9. final MyAspect myAspect = new MyAspect();
  10. /* 3 代理类:将目标类(切入点)和 切面类(通知) 结合 --> 切面
  11. * Proxy.newProxyInstance
  12. * 参数1:loader ,类加载器,动态代理类 运行时创建,任何类都需要类加载器将其加载到内存。
  13. * 一般情况:当前类.class.getClassLoader();
  14. * 目标类实例.getClass().get...
  15. * 参数2:Class[] interfaces 代理类需要实现的所有接口
  16. * 方式1:目标类实例.getClass().getInterfaces() ;注意:只能获得自己接口,不能获得父元素接口
  17. * 方式2:new Class[]{UserService.class}
  18. * 例如:jdbc 驱动 --> DriverManager 获得接口 Connection
  19. * 参数3:InvocationHandler 处理类,接口,必须进行实现类,一般采用匿名内部
  20. * 提供 invoke 方法,代理类的每一个方法执行时,都将调用一次invoke
  21. * 参数31:Object proxy :代理对象
  22. * 参数32:Method method : 代理对象当前执行的方法的描述对象(反射)
  23. * 执行方法名:method.getName()
  24. * 执行方法:method.invoke(对象,实际参数)
  25. * 参数33:Object[] args :方法实际参数
  26. *
  27. */
  28. UserService proxService = (UserService)Proxy.newProxyInstance(
  29. MyBeanFactory.class.getClassLoader(),
  30. userService.getClass().getInterfaces(),
  31. new InvocationHandler() {
  32. @Override
  33. public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
  34. //前执行
  35. myAspect.before();
  36. //执行目标类的方法
  37. Object obj = method.invoke(userService, args);
  38. //后执行
  39. myAspect.after();
  40. return obj;
  41. }
  42. });
  43. return proxService;
  44. }
  45. }

最后是测试

  1. public class TestProxy {
  2. @Test
  3. public void demo01(){
  4. UserService userService = MyBeanFactory.createService();
  5. userService.addUser();
  6. userService.updateUser();
  7. userService.deleteUser();
  8. }
  9. }

在这里插入图片描述

在这里插入图片描述

发表评论

表情:
评论列表 (有 0 条评论,418人围观)

还没有评论,来说两句吧...

相关阅读

    相关 JDK动态代理实现原理

    之前虽然会用JDK的动态代理,但是有些问题却一直没有搞明白。比如说:InvocationHandler的invoke方法是由谁来调用的,代理对象是怎么生成的,直到前几个星期才把