Spring学习笔记六
什么是AspectJ
AspectJ是一个面向切面的框架,它扩展了Java 语言。AspectJ定义了AOP语法,也可以说AspectJ是一个基于Java语言的AOP框架。通常我们在使用Spring AOP的时候,都会AspectJ的相关jar包。
<!-- https://mvnrepository.com/artifact/org.aspectj/aspectjweaver -->
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.9.2</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.springframework/spring-aop -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aop</artifactId>
<version>5.1.5.RELEASE</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.springframework/spring-aspects -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aspects</artifactId>
<version>5.1.5.RELEASE</version>
</dependency>
在spring2.0后,spring新增了对AspectJ切点表达式的支持;AspectJ1.5新增注解功能,通过JDK1.5的注解技术,能直接在类中定义切面;新版本的spring框架,也建议使用AspectJ来实现AOP。所以说在spring AOP的核心包Spring-aop里面也有对AspectJ的支持。
切点表达式
<!-- 切入点表达式 -->
<aop:pointcut expression="execution(* com.rzjt.aop.*.*(..))" id="myPointCut"/>
表达式的意思是 返回值任意,包名为com.rzjt.aop 下的任意类名中的任意方法名,参数任意。
execution 是AspectJ框架定义的一个切入点方法,语法形式:
execution(修饰符 返回值 包.类.方法名(参数) throws异常)
修饰符:一般省略,public 公共方法,* 任意
返回值:不能省略,void 返回没有值,string 返回字符串,*任意
包:可以省略,com.rzjt.aop 固定包,com.rzjt.*.service 表示rzjt下面任意子包下的service包,coml.rzjt.one…表示one包下的所有子包,包括自己,*任意
类:可以省略,UserServiceImpl 指定类,*Impl 以Impl结尾,User*以User开头,*任意
方法名:不能省略, 与类的方式一致
参数:()表示无参,(int)表示一个整型的参数,两个点表示任意类型任意数量的参数
throws:可省略,一般不写
如果切入点表达式有多个不同目录,可以通过逻辑运算符表示关系
如
<aop:pointcut expression="execution(* com.rzjt.*Service1.*(..)) ||
execution(* com.rzjt.*Service2.*(..))" id="myPointCut"/>
AspectJ 通知类型
AspectJ 通知类型,定义了类型名 以及方法格式。类型如下:
before:前置通知(应用:各种校验)
在方法执行前执行,如果通知抛出异常,阻止方法运行
afterReturning:后置通知(应用:常规数据处理)
方法正常返回后执行,如果方法中抛出异常,通知无法执行
必须在方法执行后才执行,所以可以获得方法的返回值。
around:环绕通知(应用:十分强大,可以做任何事情)
方法执行前后分别执行,可以阻止方法的执行
必须手动执行目标方法
afterThrowing:抛出异常通知(应用:包装异常信息)
方法抛出异常后执行,如果方法没有抛出异常,无法执行
after:最终通知(应用:清理现场)
方法执行完毕后执行,无论方法中是否出现异常
在程序中
try{
//前置:before
//手动执行目标方法
//后置:afterRetruning
} catch(){
//抛出异常 afterThrowing
} finally{
//最终 after
}
AOP具体实例
- 创建接口
public interface UserService{
//添加user
public int addUser();
//删除user
public int deleteUser();
}
- 创建实现类
public class UserServiceImpl implements UserService{
@Override
public int addUser() {
System.out.println("增加User");
return 1;
}
@Override
public int deleteUser() {
System.out.println("删除User");
return 2;
}
}
- 创建切面类
public class MyAspect{
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
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:aop="http://www.springframework.org/schema/aop"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd">
<!--1、 创建目标类 -->
<bean id="userService" class="com.rzjt.aop.UserServiceImpl"></bean>
<!--2、创建切面类(通知) -->
<bean id="myAspect" class="com.rzjt.aop.MyAspect"></bean>
<!--3、aop编程
3.1 导入命名空间
3.2 使用 <aop:config>进行配置
proxy-target-class="true" 声明时使用cglib代理
如果不声明,Spring 会自动选择cglib代理还是JDK动态代理
<aop:pointcut> 切入点 ,从目标对象获得具体方法
<aop:advisor> 特殊的切面,只有一个通知 和 一个切入点
advice-ref 通知引用
pointcut-ref 切入点引用
3.3 切入点表达式
execution(* com.ys.aop.*.*(..))
选择方法 返回值任意 包 类名任意 方法名任意 参数任意
-->
<aop:config>
<aop:aspect ref="myAspect">
<!-- 切入点表达式 -->
<aop:pointcut expression="execution(* com.rzjt.aop.*.*(..))" id="myPointCut"/>
<!-- 3.1 前置通知
<aop:before method="" pointcut="" pointcut-ref=""/>
method : 通知,及方法名
pointcut :切入点表达式,此表达式只能当前通知使用。
pointcut-ref : 切入点引用,可以与其他通知共享切入点。
通知方法格式:public void myBefore(JoinPoint joinPoint){
参数1:org.aspectj.lang.JoinPoint 用于描述连接点(目标方法),获得目标方法名等
-->
<aop:before method="myBefore" pointcut-ref="myPointCut"/>
<!-- 3.2后置通知 ,目标方法后执行,获得返回值
<aop:after-returning method="" pointcut-ref="" returning=""/>
returning 通知方法第二个参数的名称
通知方法格式:public void myAfterReturning(JoinPoint joinPoint,Object ret){
参数1:连接点描述
参数2:类型Object,参数名 returning="ret" 配置的
-->
<aop:after-returning method="myAfterReturning" pointcut-ref="myPointCut" returning="ret" />
<!-- 3.3 最终通知 -->
<aop:after method="myAfter" pointcut-ref="myPointCut"/>
</aop:aspect>
</aop:config>
</beans>
- 测试
public class App
{
public static void main( String[] args )
{
ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
UserService userService = (UserService) context.getBean("userService");
userService.addUser();
}
}
- 结果
前置通知:addUser
增加User
后置通知addUser,--->1
最终通知