文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

SpringBoot2 基础案例(11):配置AOP切面编程,解决日志记录业务

2023-06-02 12:53

关注

本文源码:GitHub·点这里 || GitEE·点这里

一、AOP切面编程

1、什么是AOP编程

在软件业,AOP为Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。AOP是OOP(面向对象编程)的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。

2、AOP编程特点

1)AOP采取横向抽取机制,取代了传统纵向继承体系重复性代码2)经典应用:事务管理、性能监视、安全检查、缓存 、日志等3)aop底层将采用代理机制进行实现4)接口 + 实现类 :spring采用 jdk 的动态代理Proxy5)实现类:spring 采用 cglib字节码增强

3、AOP中术语和图解

1)target:目标类   需要被代理的类。例如:UserService2)Joinpoint:连接点   所谓连接点是指那些可能被拦截到的方法。例如:所有的方法3)PointCut:切入点   已经被增强的连接点。例如:addUser()4)advice:通知/增强   增强代码。例如:after、before5)Weaving:织入   指把增强advice应用到目标对象target来创建新的代理对象proxy的过程.6)proxy 代理类7) Aspect(切面): 是切入点pointcut和通知advice的结合    一个线是一个特殊的面。    一个切入点和一个通知,组成成一个特殊的面。

SpringBoot2 基础案例(11):配置AOP切面编程,解决日志记录业务

二、与SpringBoot2.0整合

1、核心依赖

<!-- AOP依赖 --><dependency>    <groupId>org.springframework.boot</groupId>    <artifactId>spring-boot-starter-aop</artifactId></dependency>

2、编写日志记录注解

package com.boot.aop.config;import java.lang.annotation.*;@Target(ElementType.METHOD)@Retention(RetentionPolicy.RUNTIME)@Documentedpublic @interface LogFilter {    String value() default "" ;}

3、编写日志记录的切面代码

这里分为两种情况处理,一种正常的请求日志,和系统异常的错误日志。
核心注解两个。@Aspect和@Component。

package com.boot.aop.config;import com.alibaba.fastjson.JSONObject;import org.aspectj.lang.ProceedingJoinPoint;import org.aspectj.lang.annotation.Around;import org.aspectj.lang.annotation.Aspect;import org.aspectj.lang.annotation.Pointcut;import org.aspectj.lang.reflect.MethodSignature;import org.slf4j.Logger;import org.slf4j.LoggerFactory;import org.springframework.stereotype.Component;import org.springframework.web.context.request.RequestContextHolder;import org.springframework.web.context.request.ServletRequestAttributes;import javax.servlet.http.HttpServletRequest;import java.lang.reflect.Method;@Aspect@Componentpublic class LogAspect {    private static final Logger LOGGER = LoggerFactory.getLogger(LogAspect.class) ;    @Pointcut("@annotation(com.boot.aop.config.LogFilter)")    public void logPointCut (){    }    @Around("logPointCut()")    public Object around (ProceedingJoinPoint point) throws Throwable {        Object result = null ;        try{            // 执行方法            result = point.proceed();            // 保存请求日志            saveRequestLog(point);        } catch (Exception e){            // 保存异常日志            saveExceptionLog(point,e.getMessage());        }        return result;    }    private void saveExceptionLog (ProceedingJoinPoint point,String exeMsg){        LOGGER.info("捕获异常:"+exeMsg);        HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getRequest();        LOGGER.info("请求路径:"+request.getRequestURL());        MethodSignature signature = (MethodSignature) point.getSignature();        Method method = signature.getMethod();        LOGGER.info("请求方法:"+method.getName());        // 获取方法上LogFilter注解        LogFilter logFilter = method.getAnnotation(LogFilter.class);        String value = logFilter.value() ;        LOGGER.info("模块描述:"+value);        Object[] args = point.getArgs();        LOGGER.info("请求参数:"+ JSONObject.toJSONString(args));    }    private void saveRequestLog (ProceedingJoinPoint point){        HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getRequest();        LOGGER.info("请求路径:"+request.getRequestURL());        MethodSignature signature = (MethodSignature) point.getSignature();        Method method = signature.getMethod();        LOGGER.info("请求方法:"+method.getName());        // 获取方法上LogFilter注解        LogFilter logFilter = method.getAnnotation(LogFilter.class);        String value = logFilter.value() ;        LOGGER.info("模块描述:"+value);        Object[] args = point.getArgs();        LOGGER.info("请求参数:"+ JSONObject.toJSONString(args));    }}

4、请求日志测试

@LogFilter("保存请求日志")@RequestMapping("/saveRequestLog")public String saveRequestLog (@RequestParam("name") String name){    return "success:"+name ;}

切面类信息打印

5、异常日志测试

@LogFilter("保存异常日志")@RequestMapping("/saveExceptionLog")public String saveExceptionLog (@RequestParam("name") String name){    int error = 100 / 0 ;    System.out.println(error);    return "success:"+name ;}

切面类信息打印

三、源代码地址

GitHub·地址https://github.com/cicadasmile/spring-boot-baseGitEE·地址https://gitee.com/cicadasmile/spring-boot-base

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 资料下载
  • 历年真题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯