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(){} 之后通过 “方法名”,获得切入点的引用

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值