SpringCloud开发学习总结—— 结合注解的AOP示例
Posted king-brook
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了SpringCloud开发学习总结—— 结合注解的AOP示例相关的知识,希望对你有一定的参考价值。
面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
我们现在做的一些非业务,如:日志、事务、安全等都会写在业务代码中(也即是说,这些非业务类横切于业务类),但这些代码往往是重复,复制——粘贴式的代码会给程序的维护带来不便,AOP就实现了把这些业务需求与系统需求分开来做。这种解决的方式也称代理机制。
先来了解一下AOP的相关概念,《Spring参考手册》中定义了以下几个AOP的重要概念,结合下面示例分析如下:
- 切面(Aspect):官方的抽象定义为“一个关注点的模块化,这个关注点可能会横切多个对象”,在本例中,“切面”就是类TestAspect所关注的具体行为,例如,AServiceImpl.barA()的调用就是切面TestAspect所关注的行为之一。“切面”在ApplicationContext中<aop:aspect>来配置。
- 连接点(Joinpoint) :程序执行过程中的某一行为,例如,UserService.get的调用或者UserService.delete抛出异常等行为。
- 通知(Advice) :“切面”对于某个“连接点”所产生的动作,例如,TestAspect中对com.spring.service包下所有类的方法进行日志记录的动作就是一个Advice。其中,一个“切面”可以包含多个“Advice”,例如ServiceAspect。
- 切入点(Pointcut) :匹配连接点的断言,在AOP中通知和一个切入点表达式关联。例如,TestAspect中的所有通知所关注的连接点,都由切入点表达式execution(* com.spring.service.*.*(..))来决定。
- 目标对象(Target Object) :被一个或者多个切面所通知的对象。例如,AServcieImpl和BServiceImpl,当然在实际运行时,Spring AOP采用代理实现,实际AOP操作的是TargetObject的代理对象。
- AOP代理(AOP Proxy) :在Spring AOP中有两种代理方式,JDK动态代理和CGLIB代理。默认情况下,TargetObject实现了接口时,则采用JDK动态代理,例如,AServiceImpl;反之,采用CGLIB代理,例如,BServiceImpl。强制使用CGLIB代理需要将 <aop:config>的 proxy-target-class属性设为true。
通知(Advice)类型:
- 前置通知(Before advice):在某连接点(JoinPoint)之前执行的通知,但这个通知不能阻止连接点前的执行。ApplicationContext中在<aop:aspect>里面使用<aop:before>元素进行声明。例如,TestAspect中的doBefore方法。
- 后置通知(After advice):当某连接点退出的时候执行的通知(不论是正常返回还是异常退出)。ApplicationContext中在<aop:aspect>里面使用<aop:after>元素进行声明。例如,ServiceAspect中的returnAfter方法,所以Teser中调用UserService.delete抛出异常时,returnAfter方法仍然执行。
- 返回后通知(After return advice):在某连接点正常完成后执行的通知,不包括抛出异常的情况。ApplicationContext中在<aop:aspect>里面使用<after-returning>元素进行声明。
- 环绕通知(Around advice):包围一个连接点的通知,类似Web中Servlet规范中的Filter的doFilter方法。可以在方法的调用前后完成自定义的行为,也可以选择不执行。ApplicationContext中在<aop:aspect>里面使用<aop:around>元素进行声明。例如,ServiceAspect中的around方法。
- 抛出异常后通知(After throwing advice):在方法抛出异常退出时执行的通知。ApplicationContext中在<aop:aspect>里面使用<aop:after-throwing>元素进行声明。例如,ServiceAspect中的returnThrow方法。
接下来通过一下例子来演示SpringCloud+aop
- 在pom.xml文件中引入(starter中默认添加了@EnableAspectJAutoProxy)
1 <!--引用AOP注解功能开始--> 2 <dependency> 3 <groupId>org.springframework.boot</groupId> 4 <artifactId>spring-boot-starter-aop</artifactId> 5 </dependency> 6 <!--引用AOP注解功能结束-->
- 自定义一个注解,用于注解式AOP
public enum LogType { INFO, WARN, ERROR }
1 /** 2 * 系统日志记录 3 * 4 * @author cjg 5 * 6 */ 7 @Target({ ElementType.METHOD }) 8 @Retention(RetentionPolicy.RUNTIME) 9 @Documented 10 public @interface ServiceLog { 11 /** 12 * 操作类型,新增用户?删除用户 ?调用xx服务?使用接口?... 13 * 14 * @return 15 */ 16 public String operation(); 17 18 /** 19 * 日志级别 20 * 21 * @return 22 */ 23 public LogType level() default LogType.INFO; 24 25 }
- 使用@Aspect注解将一个java类定义为切面类
1 @Component 2 @Aspect 3 public class LogAspect { 4 5 private static final Logger log = LoggerFactory.getLogger(LogAspect.class); 6 7 /** 8 * 切入点 9 */ 10 @Pointcut("@annotation(com.didispace.web.aspect.ServiceLog) ") 11 public void entryPoint() { 12 // 无需内容 13 } 14 15 @Before("entryPoint()") 16 public void before(JoinPoint joinPoint) { 17 18 log.info("=====================开始执行前置通知=================="); 19 try { 20 String targetName = joinPoint.getTarget().getClass().getName(); 21 String methodName = joinPoint.getSignature().getName(); 22 Object[] arguments = joinPoint.getArgs(); 23 Class<?> targetClass = Class.forName(targetName); 24 Method[] methods = targetClass.getMethods(); 25 String operation = ""; 26 for (Method method : methods) { 27 if (method.getName().equals(methodName)) { 28 Class<?>[] clazzs = method.getParameterTypes(); 29 if (clazzs.length == arguments.length) { 30 operation = method.getAnnotation(ServiceLog.class).operation();// 操作人 31 break; 32 } 33 } 34 } 35 StringBuilder paramsBuf = new StringBuilder(); 36 for (Object arg : arguments) { 37 paramsBuf.append(arg); 38 paramsBuf.append("&"); 39 } 40 41 // *========控制台输出=========*// 42 log.info("[X用户]执行了[" + operation + "],类:" + targetName + ",方法名:" + methodName + ",参数:" 43 + paramsBuf.toString()); 44 log.info("=====================执行前置通知结束=================="); 45 } catch (Throwable e) { 46 log.info("around " + joinPoint + " with exception : " + e.getMessage()); 47 } 48 49 } 50 51 /** 52 * 环绕通知处理处理 53 * 54 * @param joinPoint 55 * @throws Throwable 56 */ 57 @Around("entryPoint()") 58 public Object around(ProceedingJoinPoint point) throws Throwable { 59 // 先执行业务,注意:业务这样写业务发生异常不会拦截日志。 60 Object result = point.proceed(); 61 try { 62 handleAround(point);// 处理日志 63 } catch (Exception e) { 64 log.error("日志记录异常", e); 65 } 66 return result; 67 } 68 69 /** 70 * around日志记录 71 * 72 * @param point 73 * @throws SecurityException 74 * @throws NoSuchMethodException 75 */ 76 public void handleAround(ProceedingJoinPoint point) throws Exception { 77 Signature sig = point.getSignature(); 78 MethodSignature msig = null; 79 if (!(sig instanceof MethodSignature)) { 80 throw new IllegalArgumentException("该注解只能用于方法"); 81 } 82 msig = (MethodSignature) sig; 83 Object target = point.getTarget(); 84 Method currentMethod = target.getClass().getMethod(msig.getName(), msig.getParameterTypes()); 85 // 方法名称 86 String methodName = currentMethod.getName(); 87 // 获取注解对象 88 ServiceLog aLog = currentMethod.getAnnotation(ServiceLog.class); 89 // 类名 90 String className = point.getTarget().getClass().getName(); 91 // 方法的参数 92 Object[] params = point.getArgs(); 93 94 StringBuilder paramsBuf = new StringBuilder(); 95 for (Object arg : params) { 96 paramsBuf.append(arg); 97 paramsBuf.append("&"); 98 } 99 // 处理log。。。。 100 log.info("[X用户]执行了[" + aLog.operation() + "],类:" + className + ",方法名:" + methodName + ",参数:" 101 + paramsBuf.toString()); 102 103 } 104 105 @AfterThrowing(pointcut = "entryPoint()", throwing = "e") 106 public void doAfterThrowing(JoinPoint joinPoint, Throwable e) { 107 // 通过request获取登陆用户信息 108 // HttpServletRequest request = ((ServletRequestAttributes) 109 // RequestContextHolder.getRequestAttributes()).getRequest(); 110 try { 111 String targetName = joinPoint.getTarget().getClass().getName(); 112 String className = joinPoint.getTarget().getClass().getName(); 113 String methodName = joinPoint.getSignature().getName(); 114 Object[] arguments = joinPoint.getArgs(); 115 Class<?> targetClass = Class.forName(targetName); 116 Method[] methods = targetClass.getMethods(); 117 String operation = ""; 118 for (Method method : methods) { 119 if (method.getName().equals(methodName)) { 120 Class<?>[] clazzs = method.getParameterTypes(); 121 if (clazzs.length == arguments.length) { 122 operation = method.getAnnotation(ServiceLog.class).operation(); 123 break; 124 } 125 } 126 } 127 128 StringBuilder paramsBuf = new StringBuilder(); 129 for (Object arg : arguments) { 130 paramsBuf.append(arg); 131 paramsBuf.append("&"); 132 } 133 134 log.info("异常方法:" + className + "." + methodName + "();参数:" + paramsBuf.toString() + ",处理了:" + operation); 135 log.info("异常信息:" + e.getMessage()); 136 } catch (Exception ex) { 137 log.error("异常信息:{}", ex.getMessage()); 138 } 139 }
- 写AOP测试功能
1 public interface ILogService { 2 3 public boolean insert(Map<String, Object> params, String id); 4 5 public boolean update(String name, String id); 6 7 public boolean delete(String id); 8 9 public boolean doError(String id); 10 }
1 @Service 2 public class TestServiceImp implements ILogService { 3 4 @ServiceLog(operation = "新增用户信息测试操作。。。。。") 5 @Override 6 public boolean insert(Map<String, Object> params, String id) { 7 return false; 8 } 9 10 @ServiceLog(operation = "更新用户信息操作....") 11 @Override 12 public boolean update(String name, String id) { 13 return false; 14 } 15 16 @ServiceLog(operation = "删除操作。。。。") 17 @Override 18 public boolean delete(String id) { 19 return false; 20 } 21 22 @ServiceLog(operation = "异常操作测试", level = LogType.ERROR) 23 @Override 24 public boolean doError(String id) { 25 try { 26 @SuppressWarnings("unused") 27 int i = 1 / 0; 28 } catch (Exception e) { 29 throw new RuntimeException(e.getMessage()); 30 } 31 return false; 32 } 33 34 }
- 编写SpringBoot测试类,并展示结果
1 @RunWith(SpringRunner.class) 2 @SpringBootTest 3 public class DemoSpringbootAopLogApplicationTests { 4 5 @Autowired 6 ILogService logService; 7 8 @Test 9 public void contextLoads() { 10 Map<String, Object> params = new HashMap<>(); 11 params.put("key1", "v1"); 12 params.put("key2", "v2"); 13 14 logService.insert(params, "111"); 15 logService.update("king", "kang"); 16 logService.delete("111"); 17 logService.doError("111"); 18 } 19 20 }
至此,SpringCloud+AOP搭建成功!
项目完整代码见https://github.com/Adosker/hello
以上是关于SpringCloud开发学习总结—— 结合注解的AOP示例的主要内容,如果未能解决你的问题,请参考以下文章
SpringCloud学习系列之二 ----- 服务消费者(Feign)和负载均衡(Ribbon)