【Spring进阶系列丨第九篇】基于XML的面向切面编程(AOP)详解

这篇具有很好参考价值的文章主要介绍了【Spring进阶系列丨第九篇】基于XML的面向切面编程(AOP)详解。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

【Spring进阶系列丨第九篇】基于XML的面向切面编程(AOP)详解,Spring进阶系列,spring,xml,java,spring boot

一、基于XML的AOP

1.1、打印日志案例

1.1.1、beans.xml中添加aop的约束

<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
        https://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context
        https://www.springframework.org/schema/context/spring-context.xsd
        http://www.springframework.org/schema/aop
        https://www.springframework.org/schema/aop/spring-aop.xsd">
</beans>

1.1.2、定义Bean

package cn.bdqn.domain;
public class User {

}
package cn.bdqn.service;
public interface UserService {

    // 保存用户
    public void save(User user);

    // 根据id查询用户
    public User queryById(Integer id);

    // 查询全部用户
    public List<User> queryAll();
}
package cn.bdqn.service;
public class UserServiceImpl implements UserService{

    // 保存用户
    public void save(User user){

    }

    // 根据id查询用户
    public User queryById(Integer id){
        return new User();
    }

    // 查询全部用户
    public List<User> queryAll(){
        return new ArrayList<User>();
    }
}

1.2、定义记录日志的类【切面】

package cn.bdqn.advice;

// 定义记录日志的类,这个类就封装了我们所有的公共的代码
public class Logger {

    //  该方法的作用是在切入点方法执行之前执行
    public void beforePrintLog(){
        System.out.println("开始打印日志啦");
    }
}

1.3、导入AOP的依赖

<dependency>
    <groupId>org.aspectj</groupId>
    <artifactId>aspectjweaver</artifactId>
    <version>1.9.4</version>
</dependency>

1.4、主配置文件中配置AOP

<beans>
  	<!--  1、注册UserServiceImpl这个Bean  -->
    <bean id="userService" class="cn.bdqn.service.UserServiceImpl"/>

    <!--  2、以下操作都是Spring基于XML的AOP配置步骤
       2.1 把通知/增强Bean也需要注册到Spring容器中
       2.2 使用<aop:config/>标签来去声明开始AOP的配置了
       2.3 使用<aop:aspect/>标签来去表示开始配置切面了
        可以想一下:既然要配置切面,那切面就是切入点和通知的结合,所以肯定需要配置切入点和通知这两部分
              id属性:是给切面提供一个唯一标识
              ref属性:是指定通知类bean的Id。
       2.4 在<aop:aspect/>标签的内部使用对应标签来配置通知的类型
              前置通知/后置通知/异常通知/最终通知
              需求:beforePrintLog方法在切入点方法执行之前之前:所以是前置通知
              前置通知:<aop:before/>
                  method属性:用于指定Logger类中哪个方法是前置通知
                  pointcut属性:用于指定切入点表达式,该表达式的含义指的是对业务层中哪些方法增强

          3、切入点表达式的写法:
                关键字:execution(表达式)
                表达式:
                    访问修饰符  方法返回值  包名1.包名2...类名.方法名(参数列表)
                需求:
                    我现在就想对UserServiceImpl类中的queryAll方法进行拦截
                    public java.util.List cn.bdqn.service.UserServiceImpl.queryAll()
    -->

    <!--  2.1 把通知/增强Bean也需要注册到Spring容器中  -->
    <bean id="logger" class="cn.bdqn.advice.Logger"/>
    <!--  2.2 使用此标签来去声明开始AOP的配置了-->
    <aop:config>
        <!--配置切面 -->
        <aop:aspect id="loggerAdvice" ref="logger">
            <!-- 配置通知的类型,并且建立增强方法和切入点方法的关联-->
            <aop:before method="beforePrintLog" 
                        pointcut="execution(public java.util.List cn.bdqn.service.UserServiceImpl.queryAll())"/>
        </aop:aspect>
    </aop:config>
</beans>

1.5、测试

【Spring进阶系列丨第九篇】基于XML的面向切面编程(AOP)详解,Spring进阶系列,spring,xml,java,spring boot

@Test
public void testUserServiceImpl() throws Exception{

   	ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");
    UserService userService = (UserService) ac.getBean("userService");

    userService.queryAll();
}

1.6、切入点表达式

​ 问题:我们上面的案例经过测试发现确实在调用业务方法之前增加了日志功能,但是问题是仅仅能针对某一个业务方法进行增强,而我们的业务方法又有可能有很多,所以显然一个一个的去配置很麻烦,如何更加灵活的去配置呢?这个就需要使用到切入点表达式

​ 语法:execution(表达式)

访问修饰符  方法返回值  包名1.包名2...类名.方法名(参数列表)

1.6.1、访问修饰符可以省略

// 完整写法
public java.util.List cn.bdqn.service.UserServiceImpl.queryAll())

// 标准写法
java.util.List cn.bdqn.service.UserServiceImpl.queryAll())

1.6.2、返回值可以使用通配符,表示任意返回值

* cn.bdqn.service.UserServiceImpl.queryAll())

1.6.3、包名可以使用通配符表示任意包。有几级包,就几个*

* *.*.*.UserServiceImpl.queryAll())

但是对于包来说,连续的写3个*,显然也是麻烦的,那么可以使用“…”表示当前包及其子包。

// 表示的是任意包下的只要有UserServiceImpl类都会对queryAll方法进行增强
* *..UserServiceImpl.queryAll())

1.6.4、类名也可以用*

* *..*.queryAll()

1.6.5、方法也可以用*

* *..*.*()

1.6.6、参数列表

写法1、可以直接写数据类型:
             基本类型直接写名称           
                  int、double
             引用类型写包名.类名的方式   
                  java.lang.String、java.util.List
写法2、可以使用通配符表示任意类型
			 前提是必须要有参数。

写法3、使用..
			 可以使用..表示有无参数均可,如果有参数则表示的可以是任意类型	

1.6.7、全通配符写法

 * *..*.*(..)

1.6.8、使用最多的写法

​ 实际中的写法:切到业务层实现类下的所有方法。即:

* com.bdqn.service.impl.*.*(..)

1.7、通知类型的使用

1.7.1、在日志类中新增通知方法

// 定义记录日志的类,这个类就封装了我们所有的公共的代码
public class Logger {

    //  该方法的作用是在切入点方法执行之前执行
    public void beforePrintLog(){
        System.out.println("前置通知(beforePrintLog):开始打印日志啦");
    }

    //  该方法的作用是在切入点方法执行之后执行
    public void afterReturningPrintLog(){
        System.out.println("后置通知(afterReturningPrintLog):业务方法执行完了,日志打印");
    }

    //  该方法的作用是在切入点方法执行出错后执行
    public void afterThrowingPrintLog(){
        System.out.println("异常通知(afterThrowingPrintLog):业务方法出现异常了,日志打印");
    }

    //  该方法的作用是在切入点方法执行之后不管有没有错误,都最终要执行
    public void afterPrintLog(){
        System.out.println("最终通知(afterPrintLog):业务方法不管有没有异常了,日志打印");
    }
}

1.7.2、配置AOP

<beans>
	<!--  2.1 把通知/增强Bean也需要注册到Spring容器中  -->
    <bean id="logger" class="cn.bdqn.advice.Logger"/>
    <!--  2.2 使用此标签来去声明开始AOP的配置了-->
    <aop:config>
        <!--配置切面 -->
        <aop:aspect id="loggerAdvice" ref="logger">
          
            <!-- 配置前置通知:在切入点方法执行之前执行-->
            <aop:before method="beforePrintLog" 
                        pointcut="execution(* cn.bdqn.service.UserServiceImpl.queryAll())"/>
          
            <!-- 后置通知:在切入点方法正常执行之后值。它和异常通知永远只能执行一个-->
            <aop:after-returning method="afterReturningPrintLog"  pointcut="execution(* cn.bdqn.service.UserServiceImpl.queryAll())"/>
          
            <!--配置异常通知:在切入点方法执行产生异常之后执行。它和后置通知永远只能执行一个-->
            <aop:after-throwing method="afterThrowingPrintLog"  
                                pointcut="execution(* cn.bdqn.service.UserServiceImpl.queryAll())"/>
          
            <!--配置最终通知:无论切入点方法是否正常执行它都会在其后面执行-->
            <aop:after method="afterPrintLog"
                       pointcut="execution(* cn.bdqn.service.UserServiceImpl.queryAll())"/>
            
        </aop:aspect>
    </aop:config>
</beans>

1.7.3、测试

@Test
public void testUserServiceImpl() throws Exception{

        ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");
        UserService userService = (UserService) ac.getBean("userService");

        userService.queryAll();
}
/***
	前置通知(beforePrintLog):开始打印日志啦
    查询全部用户执行啦
    后置通知(afterReturningPrintLog):业务方法执行完了,日志打印
    最终通知(afterPrintLog):业务方法不管有没有异常了,日志打印
**/

1.8、切入点表达式改进

​ 通过11.7可以发现,我们在配置文件中配置了四种通知类型,其中的pointcut配置的是切入点表达式,发现是一模一样的,那么有没有一种改进写法呢?可以将表达式抽取出来,将来可以引用。

1.8.1、方式一

<beans>
	<!--  1、注册UserServiceImpl这个Bean  -->
    <bean id="userService" class="cn.bdqn.service.UserServiceImpl"/>

    <!--  2、以下操作都是Spring基于XML的AOP配置步骤-->

    <!--  2.1 把通知/增强Bean也需要注册到Spring容器中  -->
    <bean id="logger" class="cn.bdqn.advice.Logger"/>
    <!--  2.2 使用此标签来去声明开始AOP的配置了-->
    <aop:config>
        <!--配置切面 -->
        <aop:aspect id="loggerAdvice" ref="logger">

            <!--
                配置切入点表达式
                    id属性用于指定切入点表达式的唯一标识。
                    expression属性用于指定表达式内容
                此标签写在aop:aspect标签内部只能当前切面使用。
           -->
            <aop:pointcut id="loggerPt" 
                          expression="execution(* 																					cn.bdqn.service.UserServiceImpl.queryAll())"/>
            
            <!-- 配置前置通知:在切入点方法执行之前执行-->
            <aop:before method="beforePrintLog" pointcut-ref="loggerPt"/>
            <!-- 后置通知:在切入点方法正常执行之后值。它和异常通知永远只能执行一个-->
            <aop:after-returning method="afterReturningPrintLog" pointcut-ref="loggerPt"/>
            <!--配置异常通知:在切入点方法执行产生异常之后执行。它和后置通知永远只能执行一个-->
            <aop:after-throwing method="afterThrowingPrintLog" pointcut-ref="loggerPt"/>
            <!--配置最终通知:无论切入点方法是否正常执行它都会在其后面执行-->
            <aop:after method="afterPrintLog" pointcut-ref="loggerPt"/>
                
        </aop:aspect>
    </aop:config>
</beans>

1.8.2、方式二

​ 对于方式一,我们将aop:pointcut标签写在了aop:aspect里面,这样的话这切入点表达式只能被当前的切面使用,而如果其他切面想使用就使用不到了,所以我们可以把这个切入点表示再定义到外面。

<beans>
	<bean id="userService" class="cn.bdqn.service.UserServiceImpl"/>

    <!--  2、以下操作都是Spring基于XML的AOP配置步骤-->

    <!--  2.1 把通知/增强Bean也需要注册到Spring容器中  -->
    <bean id="logger" class="cn.bdqn.advice.Logger"/>
    <!--  2.2 使用此标签来去声明开始AOP的配置了-->
    <aop:config>

        <!--
                配置切入点表达式
                    id属性用于指定切入点表达式的唯一标识。
                    expression属性用于指定表达式内容
                此标签写在aop:aspect标签外面,那么所有的切面都可以使用。
          -->
        <aop:pointcut id="loggerPt" 
                      expression="execution(* cn.bdqn.service.UserServiceImpl.queryAll())"/>

        <!--配置切面 -->
        <aop:aspect id="loggerAdvice" ref="logger">

            <!-- 配置前置通知:在切入点方法执行之前执行-->
            <aop:before method="beforePrintLog" pointcut-ref="loggerPt"/>
            <!-- 后置通知:在切入点方法正常执行之后值。它和异常通知永远只能执行一个-->
            <aop:after-returning method="afterReturningPrintLog" pointcut-ref="loggerPt"/>
            <!--配置异常通知:在切入点方法执行产生异常之后执行。它和后置通知永远只能执行一个-->
            <aop:after-throwing method="afterThrowingPrintLog" pointcut-ref="loggerPt"/>
            <!--配置最终通知:无论切入点方法是否正常执行它都会在其后面执行-->
            <aop:after method="afterPrintLog" pointcut-ref="loggerPt"/>
                
        </aop:aspect>
    </aop:config>
</beans>

1.9、环绕通知

1.9.1、在日志记录类中新增环绕通知

public class Logger {
    // 环绕通知
    public void aroundPrintLog(){
        System.out.println("环绕通知....aroundPrintLog.....");
    }
}

1.9.2、AOP配置环绕通知

<beans>
   <!--  1、注册UserServiceImpl这个Bean  -->
    <bean id="userService" class="cn.bdqn.service.UserServiceImpl"/>

    <!--  2、以下操作都是Spring基于XML的AOP配置步骤-->
    <!--  2.1 把通知/增强Bean也需要注册到Spring容器中  -->
    <bean id="logger" class="cn.bdqn.advice.Logger"/>
    <!--  2.2 使用此标签来去声明开始AOP的配置了-->
    <aop:config>

        <!--    配置切入点表达式    -->
        <aop:pointcut id="loggerPt" 
                      expression="execution(* cn.bdqn.service.UserServiceImpl.queryAll())"/>

        <!--配置切面 -->
        <aop:aspect id="loggerAdvice" ref="logger">

            <!-- 环绕通知-->
            <aop:around method="aroundPrintLog" pointcut-ref="loggerPt"/>
                
        </aop:aspect>
    </aop:config>
</beans>

1.9.3、测试1

@Test
public void testUserServiceImpl() throws Exception{

        ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");
        UserService userService = (UserService) ac.getBean("userService");

        userService.queryAll();
}
/**
	环绕通知....aroundPrintLog.....
	发现:仅仅打印了环绕通知的代码。当我们配置了环绕通知之后,切入点方法没有执行,而通知方法执行了
*/

1.9.4、解决

​ Spring框架为我们提供了一个接口:ProceedingJoinPoint。该接口有一个方法proceed(),此方法就相当于明确调用切入点方法。该接口可以作为环绕通知的方法参数,在程序执行时,spring框架会为我们提供该接口的实现类供我们使用。

public class Logger {
    // 环绕通知
    public Object aroundPrintLog(ProceedingJoinPoint pjp){
        Object result = null;
        try{
            Object[] args = pjp.getArgs();
            System.out.println(pjp.getSignature().getName());
            System.out.println("前置");
            result = pjp.proceed(args);
            System.out.println("后置");
            return result;
        }catch (Throwable t){
            System.out.println("异常");
            throw new RuntimeException(t);
        }finally {
            System.out.println("最终");
        }
    }
}
/**
	环绕通知:它是spring框架为我们提供的一种可以在代码中手动控制增强方法何时执行的方式。
*/

好书推荐

【Spring进阶系列丨第九篇】基于XML的面向切面编程(AOP)详解,Spring进阶系列,spring,xml,java,spring boot

《深入浅出Spring Boot 3.x》

【Spring进阶系列丨第九篇】基于XML的面向切面编程(AOP)详解,Spring进阶系列,spring,xml,java,spring boot

作者简介

杨开振——长期从事Java开发工作,拥有近十年的Java开发经验,目前就职于一家互联网金融公司,担任互联网软件开发职位。
IT技术的狂热爱好者,热衷于Java互联网方向的软件技术开发与研究。熟练掌握Java基础、软件开发设计模式和数据库相关知识,对Spring、MyBatis等主流Java开源框架有深入研究。

购书链接:点此进入

【Spring进阶系列丨第九篇】基于XML的面向切面编程(AOP)详解,Spring进阶系列,spring,xml,java,spring boot文章来源地址https://www.toymoban.com/news/detail-854980.html

到了这里,关于【Spring进阶系列丨第九篇】基于XML的面向切面编程(AOP)详解的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【Spring进阶系列丨第六篇】Spring的Bean管理(基于注解)

    回顾一下 基于xml配置的Spring对Bean的管理 ,对Bean的完整管理如下所示: 分析可以发现:我们对Bean的管理就四个方面,分别是: 用于创建对象的 用于注入数据的 用于改变Bean的作用域的 和Bean的生命周期相关的 其实对于注解来说,也是包括了这四个方面,换句话说, 使用注

    2024年02月03日
    浏览(46)
  • 【Spring进阶系列丨第一篇】初识Spring开发

    小伙伴们大家好,我是陈橘又青,今天起 《Spring进阶系列》 开始更新。本专栏将涵盖Spring框架的核心概念、配置管理、Web开发、AOP、Boot、Security、Data、Integration和Batch等多个主题。通过理论讲解和实际案例的剖析,帮助读者深入理解Spring框架的原理和应用技巧,提升开发人员

    2024年02月05日
    浏览(44)
  • 【Spring进阶系列丨第五篇】详解Spring中的依赖注入

    全称 Dependency Injection(DI) 与IoC的关系 IoC和DI其实说的是一个意思,可以这么说: IoC是一种思想,DI是对这种思想的一种具体实现 依赖关系的管理 以后都交给spring来维护,在当前类需要用到其他类的对象,由spring为我们提供,我们只需要在配置文件中说明。 依赖关系的维护

    2024年02月04日
    浏览(54)
  • 【Spring进阶系列丨第七篇】Spring框架新注解分类及详解

    1.1.1、定义一个类 1.1.2、使用Configuration注解修饰类 1.1.3、作用 ​ 使用Configuration注解修饰的类表示的是:当前类是一个配置类。该类的作用和beans.xml是一样的,换句话说,该 注解所修饰的类就是用来代替beans.xml文件的。 1.2.1、定义bean 1.2.2、在主配置类中注册bean ​ 在以前,

    2024年04月10日
    浏览(92)
  • 【Spring进阶系列丨第三篇】Spring核心技术之 IoC 与 DI 实战案例

    在上一篇文章中,我们学习了IoC与DI的相关概念与原理,现在让我们 以HelloWorld为例,编写一个程序,让创建对象的工作由Spring帮助我们创建。 一同感受一下Spring框架带给我们开发的便捷性。 这种做法是以前最常用的做法,HelloWorld这个类的对象是我们程序员自己去创建并为属

    2024年02月05日
    浏览(49)
  • 【Spring进阶系列丨第二篇】Spring中的两大核心技术IoC(控制反转)与DI(依赖注入)

    我们都知道Spring 框架主要的优势是在 简化开发 和 框架整合 上,至于如何实现就是我们要学习Spring 框架的主要内容,今天我们就来一起学习Spring中的两大核心技术IoC(控制反转)与DI(依赖注入)。 以经典的三层架构MVC作为案例,以前我们都是这么干的,看如下代码: 按照

    2024年02月05日
    浏览(66)
  • ESP32系列--第九篇 ADC的使用

            本篇主要介绍ESP32的ADC功能,ESP32有两个ADC模块,分别为ADC1/ADC2,每个ESP32系列具有的通道数不一样,详情请看下表。         在WiFi在使用时,ADC2的使用受到一些限制,实际应用场景中一般只使用ADC1即可。 ADC的IO引脚分配 ESP32系列 (下表来自ESP-IDF开发文档) GPIO

    2024年02月05日
    浏览(55)
  • 干翻Dubbo系列第九篇:Dubbo体系中序列化详解

    文章目录 文章说明 一:序列化概念 1:概念 2:Dubbo中序列化方式 二:Kyro序列化方案 1:引入依赖 2:XML的配置方式 3:Boot的方式 4:Consumer端调用 三:FST序列化方式使用 1:引入依赖 2:XML的配置方式 3:SpringBoot的配置方式 4: Consumer端调⽤ 序列化是RPC的时候,将需要传输的

    2024年02月13日
    浏览(69)
  • JavaScript系列从入门到精通系列第九篇:JavaScript中赋值运算符和关系运算符以及Unicode编码介绍

    文章目录 一:赋值运算符 1:= 2:+= 3:-= 4:*= 5:/= 6:%= 二:关系运算符  1:数值类型关系运算 (一): (二):= (三): (四):= 2:其他类型关系运算 三:Unicode编码表         =右侧的值可以赋值给左侧的变量。         上边这两个写法是一样的。                 

    2024年02月08日
    浏览(57)
  • SpringSecurity6从入门到上天系列第九篇:SpringSecurity当中的默认用户的生成、存储、认证过程的源码级别分析

    😉😉 欢迎加入我们的学习交流群呀! ✅✅1:这是 孙哥suns 给大家的福利! ✨✨2: 我们免费分享Netty、Dubbo、k8s、Mybatis、Spring等等很多应用和源码级别的高质量视频和笔记资料,你想学的我们这里都有 ! 🥭🥭3:QQ群: 583783824   📚📚  工作VX: BigTreeJava 拉你进VX群,免

    2024年02月03日
    浏览(41)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包