怎么在SpringBoot中使用AOP技术操作日志

怎么在SpringBoot中使用AOP技术操作日志?相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。

一、基本概念

项目描述
Aspect(切面)跨越多个类的关注点的模块化,切面是通知和切点的结合。通知和切点共同定义了切面的全部内容——它是什么,在何时和何处完成其功能。事务处理和日志处理可以理解为切面
Join point(连接点)程序执行过程中的一个点,如方法的执行或异常的处理
Advice(通知)切面在特定连接点上采取的动作
Pointcut(切点)匹配连接点的断言。通知与切入点表达式相关联,并在切入点匹配的任何连接点上运行(例如,具有特定名称的方法的执行)。切入点表达式匹配的连接点概念是AOP的核心,Spring默认使用AspectJ切入点表达式语言
Introduction(引用)为类型声明其他方法或字段。Spring AOP允许您向任何建议的对象引入新的接口(和相应的实现)。例如,您可以使用介绍使bean实现IsModified接口,以简化缓存
Target object(目标)由一个或多个切面通知的对象。也称为“通知对象”。由于Spring AOP是通过使用运行时代理实现的,所以这个对象始终是代理对象
AOP proxy(代理)AOP框架为实现切面契约(通知方法执行等)而创建的对象。在Spring框架中,AOP代理是JDK动态代理或CGLIB代理
Weaving(织入)织入是将通知添加对目标类具体连接点上的过程,可以在编译时(例如使用AspectJ编译器)、加载时或运行时完成

Spring切面可以应用5种类型的通知:

  • 前置通知(Before):在目标方法被调用之前调用通知

  • 后置通知(After):在目标方法完成之后调用通知(无论是正常还是异常退出)

  • 返回通知(After-returning):在目标方法成功执行之后调用通知

  • 异常通知(After-throwing):在目标方法抛出异常后调用通知

  • 环绕通知(Around):通知包裹了被通知的方法,在被通知的方法调用之前和调用之后执行自定义的行为

其执行的顺序为:

怎么在SpringBoot中使用AOP技术操作日志

怎么在SpringBoot中使用AOP技术操作日志

后续的基本应用,会将 环绕通知前置通知后置通知返回通知异常通知进行实现,并演示其执行顺序。

二、基本应用

声明通知大家可以将下面的代码复制出来,验证上面的执行顺序。

@Aspect
publicclassTest{
privatestaticintstep=0;

@Pointcut("@annotation(com.chenyanwu.erp.erpframework.annotation.Log)")//thepointcutexpression
privatevoidoperation(){}

@Before("operation()")
publicvoiddoBeforeTask(){
System.out.println(++step+"前置通知");
}

@After("operation()")
publicvoiddoAfterTask(){
System.out.println(++step+"后置通知");
}

@AfterReturning(pointcut="operation()",returning="retVal")
publicvoiddoAfterReturnningTask(ObjectretVal){
System.out.println(++step+"返回通知,返回值为:"+retVal.toString());
}

@AfterThrowing(pointcut="operation()",throwing="ex")
publicvoiddoAfterThrowingTask(Exceptionex){
System.out.println(++step+"异常通知,异常信息为:"+ex.getMessage());
}

/**
*环绕通知需要携带ProceedingJoinPoint类型的参数
*环绕通知类似于动态代理的全过程ProceedingJoinPoint类型的参数可以决定是否执行目标方法
*且环绕通知必须有返回值,返回值即目标方法的返回值
*/
//@Around("operation()")
publicObjectdoAroundTask(ProceedingJoinPointpjp){
Stringmethodname=pjp.getSignature().getName();
Objectresult=null;
try{
//前置通知
System.out.println("目标方法"+methodname+"开始,参数为"+Arrays.asList(pjp.getArgs()));
//执行目标方法
result=pjp.proceed();
//返回通知
System.out.println("目标方法"+methodname+"执行成功,返回"+result);
}catch(Throwablee){
//异常通知
System.out.println("目标方法"+methodname+"抛出异常:"+e.getMessage());
}
//后置通知
System.out.println("目标方法"+methodname+"结束");
returnresult;
}
}

其中需要注意的是切入点:@Pointcut的表达式格式:

execution(modifiers-pattern? ret-type-pattern declaring-type-pattern? name-pattern(param-pattern)throws-pattern?)括号中各个pattern分别表示:

  • 修饰符匹配(modifier-pattern?)

  • 返回值匹配(ret-type-pattern)可以为*表示任何返回值,全路径的类名等

  • 类路径匹配(declaring-type-pattern?)

  • 方法名匹配(name-pattern)可以指定方法名 或者 代表所有, set 代表以set开头的所有方法

  • 参数匹配((param-pattern))可以指定具体的参数类型,多个参数间用“,”隔开,各个参数也可以用“”来表示- 匹配任意类型的参数,如(String)表示匹配一个String参数的方法;(,String) 表示匹配有两个参数的方法,第一个参数可以是任意类型,而第二个参数是String类型;可以用(…)表示零个或多个任意参数

  • 异常类型匹配(throws-pattern?)

  • 其中后面跟着“?”的是可选项

示例:

1)execution(* (…))//表示匹配所有方法2)execution(public * com. savage.service.UserService.(…))//表示匹配com.savage.server.UserService中所有的公有方法3)execution(* com.savage.server….(…))//表示匹配com.savage.server包及其子包下的所有方法

三、日志管理实战

有了上面基本应用的理解,现在我们直接就贴代码:

1、依赖的jar包

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

2、自定义注解

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public@interfaceLog{
Stringvalue()default"";
}

3、实现切面

@Aspect
@Order(5)
@Component
publicclassLogAspect{

privateLoggerlogger=LoggerFactory.getLogger(LogAspect.class);

@Autowired
privateErpLogServicelogService;

@Autowired
ObjectMapperobjectMapper;

privateThreadLocal<Date>startTime=newThreadLocal<Date>();

@Pointcut("@annotation(com.chenyanwu.erp.erpframework.annotation.Log)")
publicvoidpointcut(){

}

/**
*前置通知,在Controller层操作前拦截
*
*@paramjoinPoint切入点
*/
@Before("pointcut()")
publicvoiddoBefore(JoinPointjoinPoint){
//获取当前调用时间
startTime.set(newDate());
}

/**
*正常情况返回
*
*@paramjoinPoint切入点
*@paramrvt正常结果
*/
@AfterReturning(pointcut="pointcut()",returning="rvt")
publicvoiddoAfter(JoinPointjoinPoint,Objectrvt)throwsException{
handleLog(joinPoint,null,rvt);
}

/**
*异常信息拦截
*
*@paramjoinPoint
*@parame
*/
@AfterThrowing(pointcut="pointcut()",throwing="e")
publicvoiddoAfter(JoinPointjoinPoint,Exceptione)throwsException{
handleLog(joinPoint,e,null);
}

@Async
privatevoidhandleLog(finalJoinPointjoinPoint,finalExceptione,Objectrvt)throwsException{
//获得注解
Methodmethod=getMethod(joinPoint);
Loglog=getAnnotationLog(method);
if(log==null){
return;
}
Datenow=newDate();
//操作数据库日志表
ErpLogerpLog=newErpLog();
erpLog.setErrorCode(0);
erpLog.setIsDeleted(0);
//请求信息
HttpServletRequestrequest=ToolUtil.getRequest();
erpLog.setType(ToolUtil.isAjaxRequest(request)?"Ajax请求":"普通请求");
erpLog.setTitle(log.value());
erpLog.setHost(request.getRemoteHost());
erpLog.setUri(request.getRequestURI().toString());
//erpLog.setHeader(request.getHeader(HttpHeaders.USER_AGENT));
erpLog.setHttpMethod(request.getMethod());
erpLog.setClassMethod(joinPoint.getSignature().getDeclaringTypeName()+"."+joinPoint.getSignature().getName());
//请求的方法参数值
Object[]args=joinPoint.getArgs();
//请求的方法参数名称
LocalVariableTableParameterNameDiscovereru
=newLocalVariableTableParameterNameDiscoverer();
String[]paramNames=u.getParameterNames(method);
if(args!=null&&paramNames!=null){
StringBuilderparams=newStringBuilder();
params=handleParams(params,args,Arrays.asList(paramNames));
erpLog.setParams(params.toString());
}
StringretString=JsonUtil.bean2Json(rvt);
erpLog.setResponseValue(retString.length()>5000?JsonUtil.bean2Json("请求参数数据过长不与显示"):retString);
if(e!=null){
erpLog.setErrorCode(1);
erpLog.setErrorMessage(e.getMessage());
}
Datestime=startTime.get();
erpLog.setStartTime(stime);
erpLog.setEndTime(now);
erpLog.setExecuteTime(now.getTime()-stime.getTime());
erpLog.setUsername(MySysUser.loginName());
HashMap<String,String>browserMap=ToolUtil.getOsAndBrowserInfo(request);
erpLog.setOperatingSystem(browserMap.get("os"));
erpLog.setBrower(browserMap.get("browser"));
erpLog.setId(IdUtil.simpleUUID());
logService.insertSelective(erpLog);
}

/**
*是否存在注解,如果存在就获取
*/
privateLoggetAnnotationLog(Methodmethod){
if(method!=null){
returnmethod.getAnnotation(Log.class);
}
returnnull;
}

privateMethodgetMethod(JoinPointjoinPoint){
Signaturesignature=joinPoint.getSignature();
MethodSignaturemethodSignature=(MethodSignature)signature;
Methodmethod=methodSignature.getMethod();
if(method!=null){
returnmethod;
}
returnnull;
}

privateStringBuilderhandleParams(StringBuilderparams,Object[]args,ListparamNames)throwsJsonProcessingException{
for(inti=0;i<args.length;i++){
if(args[i]instanceofMap){
Setset=((Map)args[i]).keySet();
Listlist=newArrayList();
ListparamList=newArrayList<>();
for(Objectkey:set){
list.add(((Map)args[i]).get(key));
paramList.add(key);
}
returnhandleParams(params,list.toArray(),paramList);
}else{
if(args[i]instanceofSerializable){
Class<?>aClass=args[i].getClass();
try{
aClass.getDeclaredMethod("toString",newClass[]{null});
//如果不抛出NoSuchMethodException异常则存在toString方法,安全的writeValueAsString,否则走Object的toString方法
params.append("").append(paramNames.get(i)).append(":").append(objectMapper.writeValueAsString(args[i]));
}catch(NoSuchMethodExceptione){
params.append("").append(paramNames.get(i)).append(":").append(objectMapper.writeValueAsString(args[i].toString()));
}
}elseif(args[i]instanceofMultipartFile){
MultipartFilefile=(MultipartFile)args[i];
params.append("").append(paramNames.get(i)).append(":").append(file.getName());
}else{
params.append("").append(paramNames.get(i)).append(":").append(args[i]);
}
}
}
returnparams;
}
}

4、对应代码添加注解

@Log("新增学生")
@RequestMapping(value="/create",method=RequestMethod.POST)
@ResponseBody
publicResultBean<String>create(@RequestBody@ValidatedErpStudentitem){
if(service.insertSelective(item)==1){
//插入
insertErpSFamilyMember(item);
returnnewResultBean<String>("");
}

returnnewResultBean<String>(ExceptionEnum.BUSINESS_ERROR,"新增学生异常!","新增失败!","");
}

看完上述内容,你们掌握怎么在SpringBoot中使用AOP技术操作日志的方法了吗?如果还想学到更多技能或想了解更多相关内容,欢迎关注恰卡编程网行业资讯频道,感谢各位的阅读!

发布于 2021-03-24 01:22:02
收藏
分享
海报
0 条评论
162
上一篇:怎么在vue项目中使用v-anchor锚点指令 下一篇:使用Spring怎么创建一个web应用
目录

    0 条评论

    本站已关闭游客评论,请登录或者注册后再评论吧~

    忘记密码?

    图形验证码