Spring AspectJ 实现AOP的方法你了解吗

网友投稿 301 2022-11-06

Spring AspectJ 实现AOP的方法你了解吗

目录1、什么是 AspectJ?2、切入点表达式AOP 切入点表达式支持多种形式的定义规则:2、Aspect 通知类型3、AOP具体实例①、创建接口②、创建实现类③、创建切面类(包含各种通知)  ④、创建spring配置文件applicationContext.xml⑤、测试  4、测试异常通知5、测试环绕通知总结

1、什么是 AspectJ?

AspectJ是一个面向切面的框架,它扩展了java语言。AspectJ定义了AOP语法,也可以说 AspectJ 是一个基于 Java 语言的 AOP 框架。通常我们在使用 Spring AOP 的时候,都会导入 AspectJ 的相关 jar 包。

在 spring2.0以后,spring新增了对AspectJ 切点表达式的支持;Aspect1.5新增注解功能,通过 JDK5的注解技术,能直接在类中定义切面;新版本的 spring 框架,也都建议使用 AspectJ 来实现 AOP。所以说在 spring AOP 的核心包 Spring-aop-3.2.jar 里面也有对 AspectJ 的支持。

2、切入点表达式

上一篇博客中,我们在spring配置文件中配置如下:

那么它表达的意思是 返回值任意,包名为 com.ys.aop 下的任意类名中的任意方法名,参数任意。那么这到底是什么意思呢?

首先 execuhttp://tion 是 AspectJ 框架定义的一个切入点函数,其语法形式如下:

execution(modifiers-pattern? ref-type-pattern declaring-type-pattern? name-pattern(param-pattern) throws-pattern?)

           类修饰符 返回值 方法所在的包 方法名 方法抛出的异常

简单点来说就是:

语法:execution(修饰符 返回值 包.类.方法名(参数) throws异常)

具体解释我们用下面一张思维导图来看:

注意:如果切入点表达式有多个不同目录呢? 可以通过 || 来表示或的关系。

表示匹配 com.ys包下的,以 Service1结尾或者以Service2结尾的类的任意方法。

AOP 切入点表达式支持多种形式的定义规则:

1、execution:匹配方法的执行(常用)

execution(public *.*(..))

2.within:匹配包或子包中的方法(了解)

within(com.ys.aop..*)

3.this:匹配实现接口的代理对象中的方法(了解)

this(com.ys.aop.user.UserDAO)

4.target:匹配实现接口的目标对象中的方法(了解)

target(com.ys.aop.user.UserDAO)

5.args:匹配参数格式符合标准的方法(了解)

args(int,int)

6.bean(id) 对指定的bean所有的方法(了解)

bean('userServiceId')

2、Aspect 通知类型

Aspect 通知类型,定义了类型名称以及方法格式。类型如下:

before:前置通知(应用:各种校验)

在方法执行前执行,如果通知抛出异常,阻止方法运行

afterReturning:后置通知(应用:常规数据处理)

方法正常返回后执行,如果方法中抛出异常,通知无法执行

必须在方法执行后才执行,所以可以获得方法的返回值。

around:环绕通知(应用:十分强大,可以做任何事情)

方法执行前后分别执行,可以阻止方法的执行

必须手动执行目标方法

afterThrowing:抛出异常通知(应用:包装异常信息)

方法抛出异常后执行,如果方法没有抛出异常,无法执行

after:最终通知(应用:清理现场)

方法执行完毕后执行,无论方法中是否出现异常

这里最重要的是around,环绕通知,它可以代替上面的任意通知。

在程序中表示的意思如下:

try{

//前置:before

//手动执行目标方法

//后置:afterRetruning

} catch(){

//抛出异常 afterThrowing

} finally{

//最终 after

}

对应的 jar 包如下:

我们可以查看源码:

3、AOP具体实例

①、创建接口

package com.ys.aop;

public interface UserService {

//添加 user

public void addUser();

//删除 user

public void deleteUser();

}

②、创建实现类

package com.ys.aop;

public class UserServiceImpl implements UserService{

@Override

public void addUser() {

System.out.println("增加 User");

}

@Override

public void delUHyxbeteUser() {

System.out.println("删除 User");

}

}

③、创建切面类(包含各种通知)

package com.ys.aop;

import org.aspectj.lang.JoinPoint;

public class MyAspect {

/**

* JoinPoint 能获取目标方法的一些基本信息

* @param joinPoint

*/

public void myBefore(JoinPoint joinPoint){

System.out.println("前置通知 : " + joinPoint.getSignature().getName());

}

public void myAfterReturning(JoinPoint joinPoint,Object ret){

System.out.println("后置通知 : " + joinPoint.getSignature().getName() + " , -->" + ret);

}

public void myAfter(){

System.out.println("最终通知");

}

}

④、创建spring配置文件applicationContext.xml

我们首先测试前置通知、后置通知、最终通知

xmlns:xsi="http://w3.org/2001/XMLSchema-instance"

xmlns:aop="http://springframework.org/schema/aop"

xsi:schemaLocation="http://springframework.org/schema/beans

http://springframework.org/schema/beans/spring-beans.xsd

http://springframework.org/schema/aop

http://springframework.org/schema/aop/spring-aop.xsd">

xmlns:xsi="http://w3.org/2001/XMLSchema-instance"

xmlns:aop="http://springframework.org/schema/aop"

xsi:schemaLocation="http://springframework.org/schema/beans

http://springframework.org/schema/beans/spring-beans.xsd

http://springframework.org/schema/aop

http://springframework.org/schema/aop/spring-aop.xsd">

⑤、测试

@Test

public void testAop(){

ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");

UserService useService = (UserService) context.getBean("userService");

useService.addUser();

}

控制台打印:

注意,后置通知的返回值为 null,是因为我们的目标方法 addUser() 没有返回值。如果有返回值,这里就是addUser() 的返回值。

4、测试异常通知

目标接口保持不变,目标类我们手动引入异常:

public void addUser() {

int i = 1/0;//显然这里会抛出除数不能为 0

System.out.println("增加 User");

}

接着配置切面:MyAspect.java

public void myAfterThrowing(JoinPoint joinPoint,Throwable e){

System.out.println("抛出异常通知 : " + e.getMessage());

}public void myAfterThrowing(JoinPoint joinPoint,Throwable e){

System.out.println("抛出异常通知 : " + e.getMessage());

}

接着在 applicationContext.xml 中配置如下:

测试:

@Test

public void testAop(){

String str = "com/ys/execption/applicationContext.xml";

ApplicationContext context = new ClassPathXmlApplicationContext(str);

UserService useService = (UserService) context.getBean("userService");

useService.addUser();

}

控制台打印:

5、测试环绕通知

目标接口和目标类保持不变,切面MyAspect 修改如下:

public class MyAspect {

public Object myAround(ProceedingJoinPoint joinPoint) throws Throwable{

System.out.println("前置通知");

//手动执行目标方法

Object obj = joinPoint.proceed();

System.out.println("后置通知");

return obj;

}

}

applicationContext.xml 配置如下:

测试:

@Test

public void testAop(){

String str = "com/ys/around/applicationContext.xml";

ApplicationContext context = new ClassPathXmlApplicationContext(str);

UserService useService = (UserService) context.getBean("userService");

useService.addUser();

}

印结果:

那么至此,通过 xml 配置的方式我们讲解了Spring AOP 的配置。下一章将通过注解的方式来实现。

总结

本篇文章就到这里了,希望能够给你带来帮助,也希望您能够多多关注我们的更多内容!

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:Git和Github简单教程,如何使用 GitHub?
下一篇:如何使用perf top探究性能
相关文章

 发表评论

暂时没有评论,来抢沙发吧~