业务场景:通过aop的环绕模式,做请求外部接口的请求和响应日志收集。其中不同接口入参不一样,但是我需要获取一个唯一标识作为业务id。
1.定义一个注解:注入到方法
package com.jd.car.laserbeak.infrastructure.jd.jsf.logrecordaop;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
* @author xyd
* @description:
* @date 2020/2/24
@Target(ElementType.METHOD)//我是控制到方法,这个根据自己实际业务选
@Retention(RetentionPolicy.RUNTIME)//什么时候切,这个都可以根据业务定
public @interface LogRecord {
String flag() default "";//用来获取入参的标识(每个业务的入参属性不一样)
2.切面:可以根据自己业务定义自己切面的范围,和用什么方式advice
package com.jd.car.laserbeak.web.config;
import com.alibaba.fastjson.JSON;
import com.jd.car.laserbeak.service.requsetlog.RequsetLogService;
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.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
* @author xyd
* @description: 日志记录切面处理类
* @date 2020/2/24
@Aspect
@Component
public class LogRecordAspect {
private static Logger LOG = LoggerFactory.getLogger(LogRecordAspect.class);
@Autowired
private RequsetLogService requestLogServie;
//切的地方。我是对我的注解切入。我们可以根据需要定义作用域 @Pointcut("@annotation(com.jd.car.laserbeak.infrastructure.jd.jsf.logrecordaop.LogRecord)")
public void pointcut() {
@Around("pointcut()")
public Object around(ProceedingJoinPoint point) {
Object result = null;
String message;
try {
// 执行方法
result = point.proceed();
//成功后,返回的业务信息获取。用作存日志使用
message = JSON.toJSONString(result);
} catch (Throwable e) {
//实际流程接口失败,获取失败的信息。存起来。我们也可以定义我们失败后业务如何处理
LOG.warn("核销接口请求外部失败:{}", e.getMessage());
message = e.getMessage();
// 保存日志。这个我需要异步执行我的保存日志。
requestLogServie.saveLog(point, message);
return result;
3.异步执行的保存日志:这个是在其他service层定义的。供切面2调用。
我用的异步直接一个注解,但是可以根据业务用线程池等方式。可以自己在做优化
* 异步执行添加日志记录
* @param joinPoint
* @param message
@Async
@Override
public void saveLog(ProceedingJoinPoint joinPoint, String message) {
MethodSignature signature = (MethodSignature) joinPoint.getSignature();
Method method = signature.getMethod();
RequestLogEntity req = new RequestLogEntity();
// 请求的方法名
String className = joinPoint.getTarget().getClass().getName();
String methodName = signature.getName();
req.setMethod(className + "." + methodName + "()");
String flag = null;
LogRecord logAnnotation = method.getAnnotation(LogRecord.class);
if (logAnnotation != null) {
// 注解上的描述,获取到实际执行方法上请求的入参某个值
flag = logAnnotation.flag();
// 请求的方法参数值
Object[] args = joinPoint.getArgs();
//获取到接口请求入参,通过key获取入参的某个值(rag是一个数组,第几个arg[n]是外层实际接口请求的第几个参数)
String string = JSON.parseObject(JSON.toJSONString(args[0])).getString(flag);
req.setKeyWord(string);
// 请求的方法参数名称
LocalVariableTableParameterNameDiscoverer u = new LocalVariableTableParameterNameDiscoverer();
String[] paramNames = u.getParameterNames(method);
if (args != null && paramNames != null) {
String params = "";
for (int i = 0; i < args.length; i++) {
params += " " + paramNames[i] + ": " + args[i];
req.setReqStr(params);
// 获取request
ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
HttpServletRequest request = attributes.getRequest();
// 设置IP地址
req.setIp(IPUtil.getIpAddr(request));
req.setReqTime(new Date());
req.setRespStr(message);
// 保存系统日志
requestLogDao.insert(req);
4.日志实体类
package com.jd.car.laserbeak.repository.entity.requestlog;
import com.jd.car.laserbeak.repository.entity.BaseEntity;
import lombok.AllArgsConstructor;
import lombok.Builder;
import lombok.Data;
import lombok.NoArgsConstructor;
import lombok.experimental.Accessors;
import javax.persistence.Entity;
import javax.persistence.Table;
import java.util.Date;
* @author xyd
* @description 请求日志表实体类
* @date 2020/2/13 21:44
@Entity
@Table(name = "request_log")
@Builder
@NoArgsConstructor
@AllArgsConstructor
@Data
@Accessors(chain = true)
public class RequestLogEntity extends BaseEntity {
* 请求ip
private String ip;
* 请求入参
private String reqStr;
* 请求出参
private String respStr;
* 请求方法
private String method;
* 请求时间
private Date reqTime;
* 关键字 比如核销码等
private String keyWord;
5.切入的方法:里面调用外部的接口
1.请求平台a的方式,我需要获取参数,注意我用的flag。后面的值是ReqLocCodeInstall 对象里面的值
@LogRecord(flag = "pwdNumber")
public Result consumeCodeNum(ReqLocCodeInstall req) throws Exception {
log.info("[{}] 调用 [LOC核销码安装Jsf接口], Request = {}", "LOC核销码安装处理器", JSON.toJSON(req));
Result result = locOrderSoaService.consumeCodeNum(req.getVenderId(), req.getCodeNum(), req.getPwdNumber(), req.getShopId(), req.getShopName(), req.getCodeType());
log.info("[{}] 调用 [LOC核销码安装Jsf接口], Response = {}", "LOC核销码安装处理器", JSON.toJSON(result));
return result;
2.平台b的方式。 我都需要获取一个值为业务流程id,然后字段不一样。就用到注解里面的flag来获取
然后这个值在切面种,通过json串的key来获取。可以看看上面的代码。这两个方法都是外部提供的入参对象和返回对象
@LogRecord(flag = "verCode")
public InstallResponse findWsfInstallCode(InstallRequest req){
log.info("[{}] 调用 [汪师傅校验核销码Jsf接口findWsfInstallCode], Request = {}","汪师傅核销码核销业务", JSON.toJSON(req));
InstallResponse res = carService.install(req);
log.info("[{}] 调用 [汪师傅校验核销码Jsf接口findWsfInstallCode], Response = {}","汪师傅核销码核销业务", JSON.toJSON(res));
return res;
6.service测试调用
方法1:调用实力
@Override
public void aspect(){
RequestLogEntity requestLogEntity = new RequestLogEntity();
requestLogEntity.setKeyWord("ssss").setReqStr("dsd").setMethod("sdsd");
requestLogDao.insert(requestLogEntity);
InstallRequest installRequest = new InstallRequest();
installRequest.setSecKey("222");
installRequest.setSource("car");
installRequest.setVerStoreName("222");
installRequest.setWokerIdNum("fgfdd发的");
installRequest.setVerCode("WSF核销码001");
InstallResponse wsfInstallCode = wsfJsfTransfer.findWsfInstallCode(installRequest);
@Override
public void qyersdfsa(){
ReqLocCodeInstall reqLoc = new ReqLocCodeInstall();
reqLoc.setCodeType(0).setPwdNumber("LOCHXM3333").setVenderId(111L)
.setShopId(3333333333L).setShopName("sddddddddddd");
com.jd.pop.order.loc.assembledflow.soa.service.result.Result response;
try {
response = locJsfTransfer.consumeCodeNum(reqLoc);
}catch (Exception e){
response =null;
7.controll层
两个不同的方法:调用测试就可以
@ResponseBody
@PostMapping("/api/consume/test")
public ResponseEntity verificationCode() {
consumeCodeService.aspect();
return ok();
@ResponseBody
@PostMapping("/api/consume/qyersdfsa")
public ResponseEntity qyersdfsa() {
consumeCodeService.qyersdfsa();
return ok();
8.结果:
9.表sql
CREATE TABLE `request_log` (
`id` int(10) unsigned NOT NULL AUTO_INCREMENT COMMENT '主键',
`is_deleted` tinyint(1) DEFAULT '0' COMMENT '是否删除',
`gmt_create` datetime DEFAULT NULL COMMENT '创建时间',
`gmt_modified` datetime DEFAULT NULL COMMENT '修改时间',
`creator` bigint(10) unsigned DEFAULT '0' COMMENT '创建人id',
`modifier` bigint(10) unsigned DEFAULT '0' COMMENT '修改人id',
`ip` varchar(40) DEFAULT '' COMMENT '商品货号',
`req_str` text COMMENT '请求入参',
`resp_str` text COMMENT '请求出参',
`method` varchar(200) DEFAULT '' COMMENT '请求方法',
`req_time` datetime DEFAULT NULL COMMENT '请求时间',
`key_word` varchar(100) DEFAULT NULL COMMENT '关键字 比如核销码等',
PRIMARY KEY (`id`),
KEY `key_key_word` (`key_word`) USING BTREE
) ENGINE=InnoDB AUTO_INCREMENT=11 DEFAULT CHARSET=utf8 COMMENT='请求日志表';
业务场景:通过aop的环绕模式,做请求外部接口的请求和响应日志收集。其中不同接口入参不一样,但是我需要获取一个唯一标识作为业务id。1.定义一个注解:注入到方法package com.jd.car.laserbeak.infrastructure.jd.jsf.logrecordaop;import java.lang.annotation.ElementType;import j...
需求:为系统中所有的提交,修改,删除等等操作做日志记录,记录的内容包括:请求参数,返回参数,如果报错就存储报错信息。日志要添加一个日志类型。
因为有用到工具类,先放工具类吧
import java.util.List;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.da...
Spring AOP 实现日志记录
一. 功能要点
自定义注解用于描述某个记录日志的操作说明,再使用AOP获取这个操作说明及入参参数,将其记录在日志里。
二、自定义注解@Log
@Log主要用于定义切入点标识及操作说明
package com.test.annotation;
import java.lang.annotation.Documented;
import java.lang.ann...
在使用around环绕方法通知目标切面时,若调用者有返回值,则需要将proceed返回回去,不然将出现返回值为NULL的情况。
` @Around("execution(* com.example.AOP.service.UserService.*(..))")
public Object around(ProceedingJoinPoint jp) throws Throwable {
System.out.println("环绕前");
//执行目标方法pro
package com.example.logexample.controller;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
public class logTextController {
* 设置是否记录日志
* http://xxxxxxxxxxxx:8081/xxxxx/appi/busi/log/isWriteLog/true?sign=2d0e3c349dbf29470c1b89dfd895e2ae&timestamp=1539320912
* @param isWriteLo...