zoukankan      html  css  js  c++  java
  • 基于注解的AOP之环绕通知日志记录案例

    1、 创建maven的jar工程,导入依赖

    <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>

    2、 编写业务层的接口

    public interface IAccountService {
    
        /**
         * 模拟保存账户
         */
       void saveAccount();
    }

    3、编写业务层接口的实现类:

    @Service("accountService")
    public class AccountServiceImpl implements IAccountService{
        @Override
        public void saveAccount() {
            System.out.println("执行了保存");
           // int i=1/0;
        }
    }

    要想注解生效,必须扫描注解所在的包。

    4、 创建一个具有公共代码的类

    @Component  //要把通知Bean交给spring来管理,否则通知Bean可能不生效
    @Aspect//表示当前类是一个切面类
    @ComponentScan(basePackages = "com.itheima")  //配置spring创建容器时要扫描的包(必须的)
    @EnableAspectJAutoProxy  // spring开启注解AOP的支持(必须的,否则切面相关注解不起作用,从而起不到增强的作用)
    public class Logger {
        @Pointcut("execution(* com.itheima.service.impl.*.*(..))")
        private void pt1(){}/**
         * 环绕通知
         * 问题:
         *      当我们配置了环绕通知之后,切入点方法没有执行,而通知方法执行了。
         * 分析:
         *      通过对比动态代理中的环绕通知代码,发现动态代理的环绕通知有明确的切入点方法调用,而我们的代码中没有。
         * 解决:
         *      Spring框架为我们提供了一个接口:ProceedingJoinPoint。该接口有一个方法proceed(),此方法就相当于明确调用切入点方法。
         *      该接口可以作为环绕通知的方法参数,在程序执行时,spring框架会为我们提供该接口的实现类供我们使用。
         *
         * spring中的环绕通知:
         *      它是spring框架为我们提供的一种可以在代码中手动控制增强方法何时执行的方式。
         */
        @Around("pt1()")
        public Object aroundPringLog(ProceedingJoinPoint pjp){
            Object rtValue = null;
            try{
                Object[] args = pjp.getArgs();//得到方法执行所需的参数
    
                System.out.println("Logger类中的aroundPringLog方法开始记录日志了。。。前置");
    
                rtValue = pjp.proceed(args);//明确调用业务层方法(切入点方法)
    
                System.out.println("Logger类中的aroundPringLog方法开始记录日志了。。。后置");
    
                return rtValue;
            }catch (Throwable t){
                System.out.println("Logger类中的aroundPringLog方法开始记录日志了。。。异常");
                throw new RuntimeException(t);
            }finally {
                System.out.println("Logger类中的aroundPringLog方法开始记录日志了。。。最终");
            }
        }
    }

    5、测试

    public class AOPTest {
        public static void main(String[] args) {
            //1.获取容器
    //        ApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");
            ApplicationContext ac = new AnnotationConfigApplicationContext(Logger.class);
            //2.获取对象
            IAccountService as = (IAccountService)ac.getBean("accountService");
            //3.执行方法
            as.saveAccount();
        }
    }

    结果:

  • 相关阅读:
    Java并发之ThreadPoolExecutor
    Java并发之同步工具类
    em和i , b和Strong 的区别
    OS应用架构谈(二):View层的组织和调用方案(中)
    iOS应用架构谈(二):View层的组织和调用方案(上)
    java(List或Array数组)求交集、并集、差集, 泛型工具类
    AES/DES 可逆性加密算法 -- java工具类
    用xtrabackup实现mysql的主从复制 阿里云rds到自己创建mysql
    java 生成二维码工具
    XDU 1022 (数论筛法+前缀和)
  • 原文地址:https://www.cnblogs.com/zwh0910/p/14628887.html
Copyright © 2011-2022 走看看