【Spring Boot学习之五】切面日志管理
环境
eclipse 4.7
jdk 1.8
Spring Boot 1.5.2
一、log4j
常见方式:log4j.properties + org.apache.log4j.Logger
比如:
log4j.properties:
log4j.rootLogger=info,error,CONSOLE,DEBUG log4j.appender.CONSOLE=org.apache.log4j.ConsoleAppender
log4j.appender.CONSOLE.layout=org.apache.log4j.PatternLayout
log4j.appender.CONSOLE.layout.ConversionPattern=%d{yyyy-MM-dd-HH-mm} [%t] [%c] [%p] - %m%n log4j.logger.info=info
log4j.appender.info=org.apache.log4j.DailyRollingFileAppender
log4j.appender.info.layout=org.apache.log4j.PatternLayout
log4j.appender.info.layout.ConversionPattern=%d{yyyy-MM-dd-HH-mm} [%t] [%c] [%p] - %m%n
log4j.appender.info.datePattern='.'yyyy-MM-dd
log4j.appender.info.Threshold = info
log4j.appender.info.append=true
#log4j.appender.info.File=/home/admin/pms-api-services/logs/info/api_services_info
log4j.appender.info.File=../logs/info/api_services_info log4j.logger.error=error
log4j.appender.error=org.apache.log4j.DailyRollingFileAppender
log4j.appender.error.layout=org.apache.log4j.PatternLayout
log4j.appender.error.layout.ConversionPattern=%d{yyyy-MM-dd-HH-mm} [%t] [%c] [%p] - %m%n
log4j.appender.error.datePattern='.'yyyy-MM-dd
log4j.appender.error.Threshold = error
log4j.appender.error.append=true
#log4j.appender.error.File=/home/admin/pms-api-services/logs/error/api_services_error
log4j.appender.error.File=../logs/error/api_services_error log4j.logger.DEBUG=DEBUG
log4j.appender.DEBUG=org.apache.log4j.DailyRollingFileAppender
log4j.appender.DEBUG.layout=org.apache.log4j.PatternLayout
log4j.appender.DEBUG.layout.ConversionPattern=%d{yyyy-MM-dd-HH-mm} [%t] [%c] [%p] - %m%n
log4j.appender.DEBUG.datePattern='.'yyyy-MM-dd
log4j.appender.DEBUG.Threshold = DEBUG
log4j.appender.DEBUG.append=true
#log4j.appender.DEBUG.File=/home/admin/pms-api-services/logs/debug/api_services_debug
log4j.appender.DEBUG.File=../logs/debug/api_services_debug
controller:
package com.wjy.controller; import org.apache.log4j.Logger;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController; import com.wjy.test1.service.UserServiceTest1; @RestController
public class UserController { private static Logger log = Logger.getLogger(UserController.class); @Autowired
public UserServiceTest1 userServiceTest1; @RequestMapping("/insertTest1ByService")
public String insertTest1ByService(String name,Integer age) {
log.info("insertTest1ByService:"+name+","+age); userServiceTest1.insertuser1(name, age);
return "success";
} }
二、AOP 切面记录日志
1、引入依赖
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>
2、切面
package com.wjy.aop; import java.util.Arrays;
import java.util.Enumeration; import javax.servlet.http.HttpServletRequest; import org.apache.log4j.Logger;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.AfterReturning;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Around;
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; @Aspect
@Component
public class WebLogAspect {
private static Logger logger = Logger.getLogger(WebLogAspect.class); //执行顺序
/**@Before -->目标方法 --> @AfterReturning --> @After*/ /**
* @Pointcut 切点定义
* 整个表达式可以分为五个部分:
1、execution(): 表达式主体。
2、第一个*号:表示返回类型,*号表示所有的类型。
3、包名:表示需要拦截的包名,后面的两个句点表示当前包和当前包的所有子包,com.sample.service.impl包、子孙包下所有类的方法。
4、第二个*号:表示类名,*号表示所有的类。
5、*(..):最后这个星号表示方法名,*号表示所有的方法,后面括弧里面表示方法的参数,两个句点表示任何参数。
*/
@Pointcut("execution(public * com.wjy.controller..*.*(..))")
public void weblog(String a,String b) {
//具体业务处理 比如报文入表等等
logger.info("执行目标方法 weblog(.....)");
} /**
* @Before: 前置通知, 在方法执行之前执行
* 被切点类名:joinPoint.getTarget().getClass().getName()
* 被切点方法:joinPoint.getSignature().getName()
* 被切点方法入参:JSON.toJSONString(joinPoint.getArgs())
*/
@Before("weblog()")
public void doBefore(JoinPoint joinPoint) {
// 接收到请求,记录请求内容
ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
HttpServletRequest request = attributes.getRequest();
logger.info("#####################请求开始####################");
// 记录下请求内容
logger.info("URL : " + request.getRequestURL().toString());
logger.info("HTTP_METHOD : " + request.getMethod());
logger.info("IP : " + request.getRemoteAddr());
Enumeration<String> enu = request.getParameterNames();
while (enu.hasMoreElements()) {
String name = (String) enu.nextElement();
logger.info("name:{" + name + "},value:{" + request.getParameter(name) + "}");
}
} /**
* @After: 后置通知, 在方法执行之后执行 。
*/
@After("weblog()")
public void doafter(JoinPoint joinPoint) {
logger.info("{}.{} params:{}:"+
joinPoint.getTarget().getClass().getName()+ "."+
joinPoint.getSignature().getName()+ "."+
Arrays.asList(joinPoint.getArgs()));
} /**
* @AfterRunning: 返回通知, 在方法返回结果之后执行
* 如果连接点方法出现异常,则返回通知方法不执行
*/
@AfterReturning(returning="ret",pointcut="weblog()")
public void doAfterReturning(JoinPoint joinPoint,Object ret) {
// 处理完请求,返回内容
logger.info("RESPONSE : " + ret);
logger.info("#####################请求结束####################");
} /**
* @AfterThrowing: 异常通知, 在方法抛出异常之后
*/
@AfterThrowing(pointcut = "weblog()", throwing = "e")
public void afterThrowExec(Throwable e) {
logger.error(e.getMessage(), e);
} /**
* @Around: 环绕通知, 围绕着方法执行
* 环绕通知方法可以包含上面四种通知方法,环绕通知的功能最全面。环绕通知需要携带 ProceedingJoinPoint 类型的参数,且环绕通知必须有返回值, 返回值即为目标方法的返回值。
* 环绕通知实现了上面几种通知的结合
*/
@Around("weblog()")
public Object aroundMethod(ProceedingJoinPoint pdj){
/*result为连接点的返回结果 不可以在执行目标方法时在定义result变量*/
Object result = null;
String methodName = pdj.getSignature().getName(); /*前置通知方法*/
System.out.println("前置通知方法>目标方法名:" + methodName + ",参数为:" + Arrays.asList(pdj.getArgs())); /*执行目标方法*/
try {
result = pdj.proceed();
/*返回通知方法*/
System.out.println("返回通知方法>目标方法名" + methodName + ",返回结果为:" + result);
} catch (Throwable e) {
/*异常通知方法*/
System.out.println("异常通知方法>目标方法名" + methodName + ",异常为:" + e);
} /*后置通知*/
System.out.println("后置通知方法>目标方法名" + methodName); return result;
} }
测试验证:http://localhost:8080/insertTest1ByService?name=wangsan&age=10
日志:
2019-07-24 09:00:04.039 INFO 2856 --- [nio-8080-exec-1] com.wjy.aop.WebLogAspect : #####################请求开始####################
2019-07-24 09:00:04.040 INFO 2856 --- [nio-8080-exec-1] com.wjy.aop.WebLogAspect : URL : http://localhost:8080/insertTest1ByService
2019-07-24 09:00:04.040 INFO 2856 --- [nio-8080-exec-1] com.wjy.aop.WebLogAspect : HTTP_METHOD : GET
2019-07-24 09:00:04.040 INFO 2856 --- [nio-8080-exec-1] com.wjy.aop.WebLogAspect : IP : 0:0:0:0:0:0:0:1
2019-07-24 09:00:04.040 INFO 2856 --- [nio-8080-exec-1] com.wjy.aop.WebLogAspect : name:{name},value:{wangsan}
2019-07-24 09:00:04.040 INFO 2856 --- [nio-8080-exec-1] com.wjy.aop.WebLogAspect : name:{age},value:{10}
2019-07-24 09:00:04.047 INFO 2856 --- [nio-8080-exec-1] com.wjy.controller.UserController : insertTest1ByService:wangsan,10
2019-07-24 09:00:04.280 INFO 2856 --- [nio-8080-exec-1] com.wjy.aop.WebLogAspect : {}.{} params:{}:com.wjy.controller.UserController.insertTest1ByService.[wangsan, 10]
2019-07-24 09:00:04.280 INFO 2856 --- [nio-8080-exec-1] com.wjy.aop.WebLogAspect : RESPONSE : success
2019-07-24 09:00:04.280 INFO 2856 --- [nio-8080-exec-1] com.wjy.aop.WebLogAspect : #####################请求结束####################
参考:
spring AOP切面表达式详解
AspectJ 切面注解中五种通知注解
【Spring Boot学习之五】切面日志管理的更多相关文章
- Spring Boot 学习摘要--关于日志框架
date: 2020-01-05 16:20:00 updated: 2020-01-08 15:50:00 Spring Boot 学习摘要--关于日志框架 学习教程来自:B站 尚硅谷 1. 关于日 ...
- Spring Boot AOP 简易操作日志管理
AOP (Aspect Oriented Programming) 面向切面编程. 业务有核心业务和边缘业务. 比如用户管理,菜单管理,权限管理,这些都属于核心业务. 比如日志管理,操作记录管理,这些 ...
- 四、spring boot 1.5.4 日志管理
spring boot日志默认采用logback进行输出,你可以对logback进行定制化,方法如下: 在resources文件夹下建立logback.xml配置文件 <?xml version ...
- 【spring boot 学习笔记】日志相关
1. 如何启用日志? maven依赖中添加:spring-boot-starter-logging <dependency> <groupId>org.springframew ...
- Spring Boot学习路线
Spring Boot 学习路线,本文计划根据作者近几年的工作.学习经验,来分析和制定一个学习使用 Spring Boot技术的步骤路线图. SpringBoot是伴随着Spring4.0诞生的: S ...
- Spring Boot系列一:默认日志logback配置解析
前言 今天来介绍下Spring Boot如何配置日志logback,我刚学习的时候,是带着下面几个问题来查资料的,你呢 如何引入日志? 日志输出格式以及输出方式如何配置? 代码中如何使用? 正文 Sp ...
- Spring Boot学习笔记2——基本使用之最佳实践[z]
前言 在上一篇文章Spring Boot 学习笔记1——初体验之3分钟启动你的Web应用已经对Spring Boot的基本体系与基本使用进行了学习,本文主要目的是更加进一步的来说明对于Spring B ...
- 15 个优秀开源的 Spring Boot 学习项目,一网打尽!
Spring Boot 算是目前 Java 领域最火的技术栈了,松哥年初出版的 <Spring Boot + Vue 全栈开发实战>迄今为止已经加印了 8 次,Spring Boot 的受 ...
- 15 个优秀开源的 Spring Boot 学习项目
Spring Boot 算是目前 Java 领域最火的技术栈了,松哥年初出版的 <Spring Boot + Vue 全栈开发实战>迄今为止已经加印了 8 次,Spring Boot 的受 ...
随机推荐
- NSSting NSData 与字符集合
NSString 是为了人类阅读而存在的,必须进行编码,以使得bytes对人类有意义 String Objects An NSString object encodes a Unicode-compl ...
- 使用apache 的FileUtils处理文件的复制等操作
今日思语:春风很柔,夏风很烈,秋风清爽,东风凛冽,愿你就是春夏秋冬的风~ 平时对一些文件进行操作,比如说写文件,读文件,复制一个文件等,使用原生File操作需要读取源文件,生成流对象,再写入一个新的文 ...
- 「GXOI / GZOI2019」宝牌一大堆 (DP)
题意 LOJ传送门 题解 可以发现「七对子」 和 「国士无双」直接暴力就行了. 唯一的就是剩下的"3*4+2". 考试的时候写了个爆搜剪枝,开了O2有50pts.写的时候发现可以D ...
- 大文件上传控件webupload插件
之前仿造uploadify写了一个HTML5版的文件上传插件,没看过的朋友可以点此先看一下~得到了不少朋友的好评,我自己也用在了项目中,不论是用户头像上传,还是各种媒体文件的上传,以及各种个性的业务需 ...
- luogu_1155: 双栈排序
洛谷1155:双栈排序 题意描述: 给定一个长度为\(n\)的序列\((n\leq 1000)\),两个初始为空的栈,问是否能借助以下四种操作将序列排为升序. \(1:\)如果序列不为空,将第一个元素 ...
- timeout/timelimit
timelimit
- .NET总结--ASP.NET工作原理
前言 前前后后写了不少关于某些技术啥的博客,一直在追新求深,而真正使用上的时候才发现了解的太少太少了,从事.net开发三年有余了不是它不行了而是我坚持不住了,如今不得不向生活低头,这个系列作为三年技术 ...
- 如何把Eclipse项目迁移到AndroidStudio(如何把项目导入安卓)--这我很困惑
学习android对我来说,就是兴趣,所以我以自己的兴趣写出的文章,希望各位多多支持!多多点赞,评论讨论加关注. 大佬必备功能. 把Eclipse项目迁移到AndroidStudio 现在就叫你如何把 ...
- 性能测试-GC问题整理
年轻代与老年代的调优(笔记整理) 几个常见名词 年轻代(young 区)从年轻代空间(包括Eden和Survivor 区域)回收内存被称为 Minor GC空间太小可能导致对象直接进入 old区 .如 ...
- js截取get参数乱码问题之解决
举个例子说一下: http://wwww.yctech.com/blog/post?id=1 像这样的话,通常通过如下代码直接截取不用做任何处理: function getQueryString(na ...