java中AOP的环绕通知

Wesley13
• 阅读 804

pom.xml

<dependencies>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
            <version>5.0.2.RELEASE</version>
        </dependency>
        <!--解析切入点表达式-->
        <dependency>
            <groupId>org.aspectj</groupId>
            <artifactId>aspectjweaver</artifactId>
            <version>1.8.7</version>
        </dependency>
    </dependencies>

bean.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: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">

    <!--配置spring的IOC,把service对象配置进来-->
    <bean id="accountService" class="com.service.impl.AccountServiceImpl">
    </bean>
    <!--spring中基于XML的AOP配置步骤
    1.把通知Bean也饿交给spring来管理
    2.使用aop:config标签表明开始AOP的配置
    3.使用aop:aspect标签表明开始配置切面
        id属性:是给切面提供一个唯一标识
        ref属性:是指定通知类的bean的Id
    4.在aop:aspect标签的内部使用对应标签来配置通知的类型
            我们现在实例是让pringLog方法在切入点方法执行之前执行,所以是前置通知
            aop:before:表示配置前置通知
                 method属性:用于指定Logger类中哪个方法是前置通知
                 pointcut属性:用于指定切入点表达式,该表达式的含义指的是对业务层中哪些方法增强

          切入点表达式的写法:
              关键字:execution(表达式)
              表达式:
                 访问修饰符  返回值  包名.包名.包名...类名.方法名(参数列表)
               标准的表达式写法:
                   public void com.itheima.service.impl.AccountServiceImpl.saveAccount()
                包名可以使用通配符,表示任意包,但是有几级包,就需要写几个*.
                包名可以使用..表示当前包及其子包
                  * *..AccountServiceImpl.saveAccount()
                 类名和方法名都可以使用*号来实现统配
                   * *..*.*()
                 参数列表:
                     可以直接写数据类型:
                          基本类型直接写名称             int
                          引用类型写包名.类名的方式     java.lang.String
                    可以使用通配符表示任意类型,但是必须有参数
                    可以使用..表示有无参数均可,有参数可以是任意类型
                全通配写法:
                   * *..*.*(..)

                 实际开发中切入点表达的通常写法:
                     且到业务层实现类下的所有方法
                        *  com.service.impg.*.*(..)
    -->
    <!--配置Logger类-->
   <bean id="logger" class="com.utils.Logger"></bean>
   <!--配置AOP-->
    <aop:config>
        <aop:aspect id="logAdvic" ref="logger">
       
            <aop:pointcut id="pt" expression="execution(* com.service.impl.*.*(..))"/>

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

    </aop:config>
</beans>

代码

import org.aspectj.lang.ProceedingJoinPoint;

/**
 * 用于记录日志的工具类,它里面提供了公共的代码
 */
public class Logger {
   /**
     * 用于打印日志:计划让其在切入点方法执行之前执行(切入点方法就是业务层方法)
     * 环绕通知
     *   当我们配置了环绕通知之后,切入点方法没有执行,而通知方法执行了。
     *   分析:通过对比动态代理中的环绕通知代码,发现动态代理的环绕通知有明企鹅的切入点方法调用,而我们的代码中没有
     *  解决:
     *    spring框架为我们提供了一个接口:ProcedingJoinPoint。该接口有一个方法proced(),此方法就相当于明确调用切入点方法。
     *    该接口可以作为环绕通知的方法参数,在程序运行时,spring框架会为我们提供该接口的实现类供我们使用。
     *
     *  spring中的环绕通知:
     *    它是spring框架为我们提供的一种可以在代理中手动控制增强方法何时执行的方式。
     */
 /*   public  void aroundPrintLog(){*/
    public  Object aroundPrintLog(ProceedingJoinPoint pjp){
        Object rtValue=null;
        try {
            System.out.println("环绕通知 Logger类中的aroundPrintLog方法开始记录日志了。。。前置");
            Object [] args=pjp.getArgs(); //得到方法执行所需的参数
            rtValue= pjp.proceed();  //明确调用业务层方法(切入点方法)

            System.out.println("环绕通知 Logger类中的aroundPrintLog方法开始记录日志了。。。后置");
            return rtValue;
        } catch (Throwable throwable) {
            //throwable.printStackTrace();
            System.out.println("环绕通知 Logger类中的aroundPrintLog方法开始记录日志了。。。异常");
            throw new RuntimeException(throwable);
        } finally {
            System.out.println("环绕通知 Logger类中的aroundPrintLog方法开始记录日志了。。。最终");
        }
       /* System.out.println("环绕通知 Logger类中的aroundPrintLog方法开始记录日志了。。。");*/
    }
}
点赞
收藏
评论区
推荐文章
blmius blmius
3年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
待兔 待兔
4个月前
手写Java HashMap源码
HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程HashMap的使用教程22
Jacquelyn38 Jacquelyn38
3年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
Wesley13 Wesley13
3年前
mysql设置时区
mysql设置时区mysql\_query("SETtime\_zone'8:00'")ordie('时区设置失败,请联系管理员!');中国在东8区所以加8方法二:selectcount(user\_id)asdevice,CONVERT\_TZ(FROM\_UNIXTIME(reg\_time),'08:00','0
Wesley13 Wesley13
3年前
00:Java简单了解
浅谈Java之概述Java是SUN(StanfordUniversityNetwork),斯坦福大学网络公司)1995年推出的一门高级编程语言。Java是一种面向Internet的编程语言。随着Java技术在web方面的不断成熟,已经成为Web应用程序的首选开发语言。Java是简单易学,完全面向对象,安全可靠,与平台无关的编程语言。
Stella981 Stella981
3年前
Django中Admin中的一些参数配置
设置在列表中显示的字段,id为django模型默认的主键list_display('id','name','sex','profession','email','qq','phone','status','create_time')设置在列表可编辑字段list_editable
Stella981 Stella981
3年前
Docker 部署SpringBoot项目不香吗?
  公众号改版后文章乱序推荐,希望你可以点击上方“Java进阶架构师”,点击右上角,将我们设为★“星标”!这样才不会错过每日进阶架构文章呀。  !(http://dingyue.ws.126.net/2020/0920/b00fbfc7j00qgy5xy002kd200qo00hsg00it00cj.jpg)  2
Wesley13 Wesley13
3年前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
背景描述Time:20190124T00:08:14.70572408:00User@Host:@Id:Schema:sentrymetaLast_errno:0Killed:0Query_time:0.315758Lock_
Python进阶者 Python进阶者
10个月前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这