SpringBoot项目中使用AOP
1.概述
将通用的逻辑用AOP技术实现可以极大的简化程序的编写,例如验签、鉴权等。Spring的声明式事务也是通过AOP技术实现的。
具体的代码参照 示例项目 https://github.com/qihaiyan/springcamp/tree/master/spring-aop
Spring的AOP技术主要有4个核心概念:
Pointcut: 切点,用于定义哪个方法会被拦截,例如
execution(* cn.springcamp.springaop.service.*.*(..))
Advice: 拦截到方法后要执行的动作
Aspect: 切面,把Pointcut和Advice组合在一起形成一个切面
Join Point: 在执行时Pointcut的一个实例
Weaver: 实现AOP的框架,例如 AspectJ 或 Spring AOP
2. 切点定义
常用的Pointcut定义有 execution 和 @annotation 两种。execution 定义对方法无侵入,用于实现比较通用的切面。@annotation 可以作为注解加到特定的方法上,例如Spring的Transaction注解。
execution切点定义应该放在一个公共的类中,集中管理切点定义。
示例:
public class CommonJoinPointConfig { @Pointcut("execution(* cn.springcamp.springaop.service.*.*(..))") public void serviceLayerExecution() {} }
这样在具体的Aspect类中可以通过 CommonJoinPointConfig.serviceLayerExecution()
来引用切点。
public class BeforeAspect { @Before("CommonJoinPointConfig.serviceLayerExecution()") public void before(JoinPoint joinPoint) { System.out.println(" -------------> Before Aspect "); System.out.println(" -------------> before execution of " + joinPoint); } }
当切点需要改变时,只需修改CommonJoinPointConfig类即可,不用修改每个Aspect类。
3. 常用的切面
Before: 在方法执行之前执行Advice,常用于验签、鉴权等。
After: 在方法执行完成后执行,无论是执行成功还是抛出异常.
AfterReturning: 仅在方法执行成功后执行.
AfterThrowing: 仅在方法执抛出异常后执行.
一个简单的Aspect:
@Aspect@Componentpublic class BeforeAspect { @Before("CommonJoinPointConfig.serviceLayerExecution()") public void before(JoinPoint joinPoint) { System.out.println(" -------------> Before Aspect "); System.out.println(" -------------> before execution of " + joinPoint); } }
4. 自定义注解
假设我们想收集特定方法的执行时间,一种比较合理的方式是自定义一个注解,然后在需要收集执行时间的方法上加上这个注解。
首先定义一个注解TrackTime:
@Target({ElementType.METHOD, ElementType.TYPE})@Retention(RetentionPolicy.RUNTIME)public @interface TrackTime { String param() default ""; }
然后再定义一个Aspect类,用于实现注解的行为:
@Aspect@Componentpublic class TrackTimeAspect { @Around("@annotation(trackTime)") public Object around(ProceedingJoinPoint joinPoint, TrackTime trackTime) throws Throwable { Object result = null; long startTime = System.currentTimeMillis(); result = joinPoint.proceed(); long timeTaken = System.currentTimeMillis() - startTime; System.out.println(" -------------> Time Taken by " + joinPoint + " with param[" + trackTime.param() + "] is " + timeTaken); return result; } }
在某个方法上使用这个注解,就可以收集这个方法的执行时间:
@TrackTime(param = "myService")public String runFoo() { System.out.println(" -------------> foo"); return "foo"; }
注意 @TrackTime(param = "myService")
注解是可以传参的。
为了让注解可以传参数,需要在定义注解时指定一个参数String param() default "默认值"
,
同时在Aspect类中,around方法上加上相应的参数,@Around注解中也需要用参数的变量名trackTime,而不能用类名TrackTime。
@Around("@annotation(trackTime)")public Object around(ProceedingJoinPoint joinPoint, TrackTime trackTime)
5.总结
在运行示例项目时,控制台会输出以下内容:
-------------> Before Aspect -------------> before execution of execution(String cn.springcamp.springaop.service.MyService.runFoo()) -------------> foo -------------> Time Taken by execution(String cn.springcamp.springaop.service.MyService.runFoo()) with param[myService] is 8 -------------> After Aspect -------------> after execution of execution(String cn.springcamp.springaop.service.MyService.runFoo()) -------------> AfterReturning Aspect -------------> execution(String cn.springcamp.springaop.service.MyService.runFoo()) returned with value foo
可以看出几种 Aspect 的执行顺序依次为 Before After Around AfterReturning(AfterThrowing)
更多参考内容:http://www.roncoo.com/course/list.html?courseName=spring

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
兼顾稳定和性能,58大数据平台的技术演进与实践
讲师|赵健博编辑|尚剑大家好!我是赵健博,来自58赶集,今天给大家分享一下58大数据这块的经验。我先做个自我介绍,我本科和研究生分别是在北京邮电大学和中国科学院计算技术研究所先后毕业的,之前在百度和360工作,现在是58赶集高级架构师、58大数据平台负责人。我有多年的分布式系统(存储、计算)的实践和研发经验,在我工作的这些年中运营了大大小小的集群,最大单集群也达到了四五千台,在这个过程中做了大量的功能研发、系统优化,当然也淌了大量的坑,今天会给大家介绍一些我认为比较重要的。 接下来我会跟大家分享一下58大数据平台在最近一年半的时间内技术演进的过程。主要内容分为三方面:58大数据平台目前的整体架构是怎么样的;最近一年半的时间内我们面临的问题、挑战以及技术演进过程;以及未来的规划。 首先看一下58大数据平台架构。大的方面来说分为三层:数据基础平台层、数据应用平台层、数据应用层,还有两列监控与报警和平台管理。 数据基础平台层又分为四个子层: 1.接入层,包括了Canal/Sqoop(主要解决数据库数据接入问题)、还有大量的数据采用Flume解决方案; 2.存储层,典型的系统HDFS(文件存储...
- 下一篇
Redis有的值能存有的值不能存、Jedis不好使了?
问题来了 有的值可以存进去 有的值存不进去 是不是redis坏了?是不是Jedis 客户端不好使了。 本地测试一下: 同样的redis.conf 文件(window 和 centos) centos 是线上 windows是本地测试 @Test publicvoidtest03(){ Jedisjedis=JedisUtil.getInstance().getJedis(); jedis.hset(Constants.GOODS_KANGO_AMOUNT,"wxg_1803271041153448","1"); jedis.hincrBy(Constants.GOODS_KANGO_AMOUNT,"wxg_1803271041153449",1L); JedisUtil.returnBrokenResource(jedis); } 结果:redis 看到了这个值。存进去了 为什么 在线上 存不进去呢 ? 解决思路 就是想看下 客户端到底干了什么 ./redis-cli #输入你的密码(如果需要) auth**** #开启对客户端的监控 monitor 于是再次测试(真是逻辑。不是上面的...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- CentOS8,CentOS7,CentOS6编译安装Redis5.0.7
- CentOS7安装Docker,走上虚拟化容器引擎之路
- SpringBoot2编写第一个Controller,响应你的http请求并返回结果
- CentOS8编译安装MySQL8.0.19
- CentOS8安装Docker,最新的服务器搭配容器使用
- Docker使用Oracle官方镜像安装(12C,18C,19C)
- Springboot2将连接池hikari替换为druid,体验最强大的数据库连接池
- SpringBoot2全家桶,快速入门学习开发网站教程
- 2048小游戏-低调大师作品
- CentOS7设置SWAP分区,小内存服务器的救世主