Spring学习笔记七
xml 的方式实现AOP
接口 UserService
public interface UserService{
//添加user
public int addUser();
//删除user
public int deleteUser();
}
实现类UserServiceImpl
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;
}
}
切面类,也就是说通知类 MyAspect
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 myAfterThrowing(JoinPoint joinpPoint, Throwable e){
System.out.println("抛出异常通知:"+e.getMessage());
}
public void myAfter(){
System.out.println("最终通知");
}
}
AOP配置文件 applicationContext.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
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
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.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();
userService.deleteUser();
}
}
结果
前置通知:addUser
增加User
后置通知addUser,--->1
最终通知
前置通知:deleteUser
删除User
后置通知deleteUser,--->2
最终通知
注解实现AOP
注解配置 bean
前面的xml配置:
<!--1、创建目标类 -->
<bean id="userService" class="com.rzjt.aop.UserServiceImpl"></bean>
<!--2、创建切面类(通知) -->
<bean id="myAspect" class="com.rzjt.aop.MyAspect"></bean>
注解配置:
目标类:
@Service("userService")
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;
}
}
切面类:
@Component("myAspect")
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 myAfterThrowing(JoinPoint joinpPoint, Throwable e){
System.out.println("抛出异常通知:"+e.getMessage());
}
public void myAfter(){
System.out.println("最终通知");
}
}
配置扫描注解识别
上面配置的注解,Spring 如何才能识别这些类上面的注解呢?我们必须告诉它。
在applicationContext.xml 文件中添加的配置:
<!-- 配置扫描注解类
base-package:表示含有注解类的包名。
如果扫描多个包,则下面的代码书写多行,改变 base-package 里面的内容即可!
-->
<context:component-scan base-package="com.rzjt.aop"></context:component-scan>
注解配置AOP
xml配置:
在这里插入代码片
对应的注解配置,在切面类上添加@Aspect注解:
@Component("myAspect")
@Aspect
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 myAfterThrowing(JoinPoint joinpPoint, Throwable e){
System.out.println("抛出异常通知:"+e.getMessage());
}
public void myAfter(){
System.out.println("最终通知");
}
}
如何让Spring认识我们所配置的AOP注解呢?光有前面的类注解扫描是不够的,这里我们要配置AOP注解的识别。
在applicationContext中增加配置
<!--2、确定 aop 注解生效 -->
<aop:aspectj-autoproxy></aop:aspectj-autoproxy>
注解配置前置通知
xml配置前置通知:
<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"/>
注解方式:
@Before("execution(* com.rzjt.aop.*.*(..))")
public void myBefore(JoinPoint joinPoint){
System.out.println("前置通知:"+joinPoint.getSignature().getName());
}
注解配置后置通知
xml配置后置通知
<!-- 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" />
后置通知有returning=ret 配置,这是用来获得目标方法的返回值的。
注解配置如下
@AfterReturning(value = "execution(* com.rzjt.aop.*.*(..))",returning = "ret")
public void myAfterReturning(JoinPoint joinPoint,Object ret){
System.out.println("后置通知"+joinPoint.getSignature().getName()+",--->"+ret);
}
测试
public class App
{
public static void main( String[] args )
{
ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext_Annotation.xml");
UserService userService = (UserService) context.getBean("userService");
userService.addUser();
userService.deleteUser();
}
}
结果
前置通知:addUser
增加User
后置通知addUser,--->1
前置通知:deleteUser
删除User
后置通知deleteUser,--->2
注解改进
我们可以看前置通知和后置通知的注解配置
@Before("execution(* com.rzjt.aop.*.*(..))")
public void myBefore(JoinPoint joinPoint){
System.out.println("前置通知:"+joinPoint.getSignature().getName());
}
@AfterReturning(value = "execution(* com.rzjt.aop.*.*(..))",returning = "ret")
public void myAfterReturning(JoinPoint joinPoint,Object ret){
System.out.println("后置通知"+joinPoint.getSignature().getName()+",--->"+ret);
}
切入点表达式,很显然是重复的,而且如果我们有多个通知方法,那就得在每个方法名上都写这样的注解,而且如果包名复杂,也容易写错。
改进的解决办法:
在切点面MyAspect中新增一个切入点方法 myPointCut(),然后在这个方法上添加@Pointcut注解
@Pointcut("execution(* com.rzjt.aop.*.*(..))")
public void myPointCut(){
}
那么前置通知和后置通知,我们可以进行如下写配置:
@Before("myPointCut()")
public void myBefore(JoinPoint joinPoint){
System.out.println("前置通知:"+joinPoint.getSignature().getName());
}
@AfterReturning(value = "myPointCut()",returning = "ret")
public void myAfterReturning(JoinPoint joinPoint,Object ret){
System.out.println("后置通知"+joinPoint.getSignature().getName()+",--->"+ret);
}
总结
@Aspect 声明切面,修改切面类,从而获得通知。
@Before 前置通知
@AfterReturning 后置通知
@Around 环绕通知
@AfterThrowing 抛出异常通知
@After 最终通知
@PoinCut 修饰方法 private void xxx(){} 之后通过 “方法名”,获得切入点的引用