跟我动手搭框架二之AOP实现

举报
西魏陶渊明 发表于 2022/09/25 03:01:01 2022/09/25
【摘要】 代理这里主要用CGLIB代理,主要为实现前置通知,后置通知,环绕通知和异常通知 本篇主要承上启下,承上根据IOC容易实现简单AOP代理, 启下,对将要实现的WEB模块做一个规划 文章中多有代码,会在第三部分WEB容器实现,列出参考文档及GITHUB源码地址 目录 1.编写工具类2.实现AOP3.web实现规划 定义...

代理这里主要用CGLIB代理,主要为实现前置通知,后置通知,环绕通知和异常通知

本篇主要承上启下,承上根据IOC容易实现简单AOP代理, 启下,对将要实现的WEB模块做一个规划

文章中多有代码,会在第三部分WEB容器实现,列出参考文档及GITHUB源码地址

目录

  • 1.编写工具类
  • 2.实现AOP
  • 3.web实现规划

定义接口类并提供抽象空实现

抽象目的: 实现类只需要继承要,实现的方法,即可


  
  1. /**
  2. * @Package: smile.proxy
  3. * @Description: 代理通知
  4. * @author: liuxin
  5. * @date: 2017/10/18 上午10:18
  6. */
  7. public interface ProxyAspect {
  8. /**
  9. * 前置通知
  10. */
  11. void before();
  12. /**
  13. * 后置通知
  14. */
  15. void after();
  16. /**
  17. * 异常通知
  18. */
  19. void throwed();
  20. /**
  21. * 环绕通知
  22. */
  23. void around();
  24. }
  25. /**
  26. * @Package: smile.proxy
  27. * @Description:
  28. * @author: liuxin
  29. * @date: 2017/10/18 上午10:22
  30. */
  31. public abstract class DefaultProxyAspect implements ProxyAspect {
  32. @Override
  33. public void before() {}
  34. @Override
  35. public void after() {}
  36. @Override
  37. public void throwed() {}
  38. @Override
  39. public void around() {}
  40. }

定义CGLIBProxy工具

主要逻辑

  • 实现提供方法级代理,也就是对象不用实例化
  • 对实例进行代理

代码更有说服力,直接看注释


  
  1. /**
  2. * @Package: com.example.proxy
  3. * @Description: JDK自带动态代理,只能代理,拥有接口的,而Cglib代理,是运行在动态生成字节码的工具中
  4. * 根据注解实现
  5. * @author: liuxin
  6. * @date: 17/3/31 上午10:24
  7. */
  8. public class CGLibProxy implements MethodInterceptor {
  9. private ProxyAspect proxyAspect;
  10. private Class cls;
  11. private Object object;
  12. /**
  13. * 预处理
  14. * 当代理逻辑中依赖其他类,需要提前注入时候,仅扩展此类
  15. *
  16. * 扩展逻辑类 proxyAspect 从ioc容器中获取实例
  17. *
  18. * @return
  19. */
  20. public List<String> preProcessing() {
  21. SmileProxyAspect smileProxyAspect = (SmileProxyAspect) cls.getAnnotation(SmileProxyAspect.class);
  22. try {
  23. proxyAspect = smileProxyAspect.proxyAspect().newInstance();
  24. } catch (InstantiationException e) {
  25. e.printStackTrace();
  26. } catch (IllegalAccessException e) {
  27. e.printStackTrace();
  28. }
  29. return Arrays.asList(smileProxyAspect.methods());
  30. }
  31. @Override
  32. public Object intercept(Object obj, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {
  33. Object result = null;
  34. //判断方法是否使用代理
  35. boolean contains = preProcessing().contains(method.getName());
  36. if (contains) {
  37. proxyAspect.around();
  38. proxyAspect.before();
  39. try {
  40. if (object == null) {
  41. result = methodProxy.invokeSuper(obj, args);
  42. } else {
  43. result = methodProxy.invoke(object, args);
  44. }
  45. } catch (Throwable throwable) {
  46. proxyAspect.throwed();
  47. }
  48. proxyAspect.after();
  49. proxyAspect.around();
  50. } else {
  51. result = methodProxy.invokeSuper(obj, args);
  52. }
  53. return result;
  54. }
  55. public static CGLibProxy instance() {
  56. return new CGLibProxy();
  57. }
  58. /**
  59. * 利用泛型
  60. * 获取代理类型
  61. *
  62. * @param cls
  63. * @param <T>
  64. * @return
  65. */
  66. public <T> T getProxy(Class<T> cls) {
  67. this.cls = cls;
  68. return (T) Enhancer.create(cls, this);
  69. }
  70. /**
  71. * 获取代理类
  72. * @param cls
  73. * @return
  74. */
  75. public Object toProxyObject(Class cls) {
  76. this.cls = cls;
  77. return Enhancer.create(cls, this);
  78. }
  79. /**
  80. * 注入代理对象实例
  81. *
  82. * @param obj
  83. * @return
  84. */
  85. public CGLibProxy setProxyObject(Object obj) {
  86. this.object = obj;
  87. return this;
  88. }
  89. /**
  90. * 演示代码中,均使用@SmileProxyAspect注解实现
  91. * 注解中
  92. * proxyAspect 代理切面类,需要包括处理逻辑 要实现DefaultProxyAspect 需要实现的抽象方法
  93. * methods 需要代理的方法名称
  94. *
  95. * @param args
  96. */
  97. public static void main(String[] args) {
  98. /******************************************
  99. * 方法级代理
  100. */
  101. /**
  102. * 实现前置通知和后置通知
  103. */
  104. Jay2 proxy1 = CGLibProxy.instance().getProxy(Jay2.class);
  105. proxy1.dance("芭蕾舞");
  106. System.out.println(proxy1);
  107. /**
  108. * 使用默认通知,只打印日志
  109. */
  110. Jay2 proxy = CGLibProxy.instance().getProxy(Jay2.class);
  111. System.out.println(proxy);
  112. proxy.dance("芭蕾舞");
  113. /******************************************
  114. * 实例代理
  115. */
  116. /**
  117. * 拦截对象
  118. */
  119. Jay2 jay2 = CGLibProxy.instance().setProxyObject(new Jay2("周杰伦")).getProxy(Jay2.class);
  120. jay2.say();
  121. Jay2 jay3 = CGLibProxy.instance().setProxyObject(new Jay2("周杰伦")).getProxy(Jay2.class);
  122. jay3.say();
  123. System.out.println(Jay2.class.isAssignableFrom(jay3.getClass()));
  124. System.out.println(jay3);
  125. Object jay4= CGLibProxy.instance().setProxyObject(new Jay2("周杰伦")).toProxyObject(Jay2.class);
  126. }
  127. }

AOP实现

  • 实现方案

    当IOC容器扫描所有被@SmileComonpent标记的组件时候,会判断是否被@SmileProxyAspect 注解修饰,

    如果有@SmileProxyAspect,则对实例化对象生成代理,注入


  
  1. /**
  2. * 扫描所有被标记的组件
  3. */
  4. public void scanComponent(Class<?> nextCls) {
  5. SmileComponent declaredAnnotation = nextCls.getDeclaredAnnotation(SmileComponent.class);
  6. Object beanInstance = null;
  7. beanInstance = nextCls.newInstance();
  8. //判断是否包括代理注解,如过包括就生成代理对象
  9. SmileProxyAspect smileProxyAspect = (SmileProxyAspect) nextCls.getAnnotation(SmileProxyAspect.class);
  10. if (smileProxyAspect != null) { beanInstance=CGLibProxy.instance().setProxyObject(beanInstance).toProxyObject(nextCls);
  11. ....
  12. ....
  13. registeredBeans.put(beanName, new BeanDefinition(nextCls, beanInstance));
  14. }

3. web规划

在对ioc容器及代理编写完成后,就到重点我们要实现,对HTTP请求的解析和处理. 在此实现,要对项目做一个规划,打一个草稿

项目基于Maven多模块实现

3.1 包名

groupId: org.smileframework.boot

artifactId:

  • org.smileframework.web
  • org.smileframework.tool
  • org.smileframework.data
  • org.smileframework.ioc

3.2TOOLS部分

工欲善其事必先利其器,所以先写我们将会遇到的工具类

  • 生成代理部分 CGLIB代理 org.smileframework.tool.proxy
  • 线程工厂及拒绝策略 org.smileframework.tool.threadpool
  • json及xml转换 org.smileframework.tool.json | .xml
  • 数据流工具类 org.smileframework.tool.io
  • 类加载器器 org.smileframework.tool.clazz
  • 文件读取工具 org.smileframework.tool.io.SmileClassPathResource

3.3 IOC部分


  
  1. 在处理器中扫描注解可以知道项目具有哪些功能
  2. @SmileBootApplication 如果有改扫描器,就到子目录中的使用@SmileComponent注解的都加入到默认的beans容器中
  3. 然后开始实例化,如果发现Class中包括@InsertBean()将从bean容器中的对象,反射进去生成对象.发现方法中用@SmileBean修饰的同样从ioc容器中获取实例,并注入到该对象中,最终保存到IOC容器

3.4 合并WEB上下文

从bean容器中获取到ExtApplicationContext

3.5 绑定Url和处理类


  
  1. @GetMapping 标记get请求方法
  2. @PostMapping 标记post请求方法
  3. @RequestBody 将请求体绑定到方法指定类型
  4. @RequestHander 绑定请求头到被就是的map类型
  5. @RequestParam 指定方法请求参数名
  6. WebDefinition webDefinition = WebContextTools.getWebDefinitionByUrl(dispatchUrl, requestMethod);
  7. 校验使用反射,将该方法的参数名称和请求到的参数做一个简单校验
  8. //测试将POST请求体中数据,反序列化为User
  9. @PostMapping(value = "/smile/test/requestBody", consumes.., produces..)
  10. public String testRequestBody(@RequestBoby UserDto user) {
  11. return user.getName() + "--" + user.getAge();
  12. }
  13. //Method: 获取到的执行方法 parameters:请求参数 headers:请求头
  14. //将以上信息根据consumes定义的解析方法,转换成方法指定类型
  15. Object[] args = ControllerUtils.getArgs(method, parameters,headers);
  16. Object invokeResult = method.invoke(controller, args);
  17. //根据produces定义的返回值类型,通过Netty channle返回给客户端

3.6 AOP 拦截

在扫描时候找到切面类,读取里面的class类型


  
  1. proxyAspect: 代理逻辑,实现 1.前置 2. 后置 3.环绕 4.异常通知
  2. @SmileProxyAspect(proxyAspect = ControllerProxyAspectDemo.class, methods = {"testRequestBody", "testRequestParam"})

获取到这个注解,并获取class类型,从bean容器中,获取这个对象(A),根据Smile的方法把这个A对象,进行代理.

3.7 web容器使用Netty

  • 定义WebApplicationContext 类实现ExtApplicationContext上下文信息获取IOC容器,并扫描ioc获取url绑定handler绑定的信息
  • 创建netty server端

文章来源: springlearn.blog.csdn.net,作者:西魏陶渊明,版权归原作者所有,如需转载,请联系作者。

原文链接:springlearn.blog.csdn.net/article/details/102425344

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。