当前位置: 首页 > news >正文

【Java】自定义注解和AOP切面的使用

前言

我们在开发的过程中,一般都需要对方法的入参进行打印,或者Debug调试的时候我们要查看方法入参的参数是否数量和数据正确性。

一般我们需要知道请求的参数接口路径请求ip

但是考虑以后项目上线BUG排查的问题,最好的方式就是使用切面的方式来记录每个方法执行时要保存的日志处理,那么下面我们来实现一个使用自定义注解的方式来对每个请求的方法上进行日志存储

AOP切面:对某个方法进行增强处理,例如在某个方法执行前或者执行后进行操作。

案例

首先我们看一个controller接口

import lombok.extern.slf4j.Slf4j;@Slf4j
@RestController
@RequestMapping("/user/")
public class TestController {@PostMapping("getUserById")public String getUserById(@RequestBody User user) {log.info("/user/getUserById params:{}", user.toString());//执行代码逻辑...return "请求成功";}
}

这种一般没什么问题,一般我们测试的时候都可以这样来写,但如果有很多的方法,那么每个方法都这样写,显然很是繁琐,那么我们通过下面的方式来实现当进入方法前打印请求的一些信息

实现

我们自定义一个注解,名为Itboy

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)//ElementType.METHOD表示为该注解在方法上添加
public @interface Itboy {
}

然后需要用到一些依赖,这些是使用AOP的依赖,我们提前引入一下

<dependency><groupId>org.aspectj</groupId><artifactId>aspectjweaver</artifactId><version>1.8.14</version>
</dependency>
<dependency><groupId>org.aspectj</groupId><artifactId>aspectjrt</artifactId><version>1.9.19</version>
</dependency>

然后我们定义一个切面类,名为ItboyAspect
注:这篇文章我使用的System打印方式,如果需要保存日志,换为对应的logger.info()即可。

import com.alibaba.fastjson.JSON;
import lombok.extern.slf4j.Slf4j;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
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.text.SimpleDateFormat;
import java.util.Date;@Component
@Aspect
@Slf4j
public class ItboyAspect {//@Pointcut:为切入点,切入到Itboy这个注解上面@Pointcut("@annotation(com.mayikt.demo.Itboy)")public void itboyAspect() {}//@Before为在进入切点之前自动执行Before中的逻辑(进入之前的前提是方法上需要有我们设置的自定注解)@Before("itboyAspect()")public void beforeItboy(JoinPoint joinPoint) {//获取本次请求ServletRequestAttributes requestAttributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();HttpServletRequest request = requestAttributes.getRequest();//获取到方法名String methodName = joinPoint.getSignature().getName();System.err.println("======================================方法:" + methodName + "() 开始======================================");//执行时间SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");Date date = new Date();String time = sdf.format(date);System.err.println("时间                 : " + time);//请求URLSystem.err.println("URL                 : " + request.getRequestURL());//请求方法System.err.println("HTTP Method         : " + methodName);//打印controller全路径和执行方法System.err.println("Class Method        : " + joinPoint.getSignature().getDeclaringTypeName() + "." + methodName);//请求IPSystem.err.println("IP                  : " + request.getRemoteHost());//请求入参System.err.println("Requert params      : " + JSON.toJSONString(joinPoint.getArgs()));}//后置通知@After("itboyAspect()")public void afterItboy(JoinPoint joinPoint) {//获取到方法名String methodName = joinPoint.getSignature().getName();System.err.println("======================================方法:" + methodName + "() 结束======================================");}
}

然后我们只需要在需要保存的方法上添加@Itboy注解即可

	@Itboy@PostMapping("selectUserList")public String getUserById(@RequestBody User user) {log.info("/user/selectUserList params:{}", user.toString());//执行代码逻辑...return "请求成功";}

最后我们来看一下效果

2023-04-02 14:21:29.033  INFO 5912 --- [nio-8080-exec-1] o.s.web.servlet.DispatcherServlet        : Completed initialization in 1 ms
======================================方法:getUserById() 开始======================================
时间                 : 2023-04-02 14:21:29
URL                 : http://127.0.0.1:8080/user/selectUserList
HTTP Method         : getUserById
Class Method        : com.mayikt.demo.TestController.getUserById
IP                  : 127.0.0.1
Requert params      : [{"address":"华东","age":18,"id":10010,"name":"贾强"}]
======================================方法:getUserById() 结束======================================
2023-04-02 14:21:29.171  INFO 5912 --- [nio-8080-exec-1] com.mayikt.demo.TestController           : /user/selectUserList params:User(id=10010, name=贾强, age=18, address=华东)

总结

使用上面这种方式,减轻了我们自己手动日志打印的繁琐,而且配置也相对于比较简单,如果有其他需求是比较频繁的操作的话,我们就可以使用AOP切面的方式来完成。

肥肠好用

http://www.lryc.cn/news/56367.html

相关文章:

  • 前后台协议联调拦截器
  • 【还在传统绑骨骼动画?】让AI助力你实现2D游戏角色动画流程
  • 动态规划+例题
  • 快商通荣获多个政府科技、人才奖项
  • Linux的基本命令的使用
  • RecycleView小结
  • 【Python】如何实现Redis构造简易客户端(教程在这)
  • 326. 3 的幂 ——【Leetcode每日一题】
  • UE4 Sequence学习
  • 总结MySQL、Redis的优化措施与使用 mysql_upgrade升级数据结构
  • C++11线程库
  • 智能化生产,提高效率!使用关键词采集工具助力企业数字化转型
  • 浅谈自动化测试用例创建和文档
  • [Java Web]AJAX Axios | 一种结合HTML来取代传统JSP的技术
  • 【C++】多态问答题
  • 【设计模式】适配器模式
  • 跨域之CorsFilter
  • STM32基于HAL工程读取DS1302时间数据
  • 《Effective Objective-C 2.0 》 阅读笔记 item10
  • gpt3官网中文版-人工智能软件chat gpt安装
  • 工作常用、面试必问:Hive 窗口函数汇总
  • spring5(五):AOP操作
  • functional.partial
  • C#缩放PDF文件
  • 【Java面试八股文宝典之MySQL篇】备战2023 查缺补漏 你越早准备 越早成功!!!——Day20
  • Nsight System的安装和使用
  • Spring销毁的几种实现
  • 【 Spring 核⼼与设计思想 】
  • Arrays.sort()——逆序
  • 测试2年遇到瓶颈,如何跨过这个坎,实现涨薪5k?