摘要:背景大量项目在使用记日志,有部分项目使用日志混乱,格式不统一,多数人搞不懂配置文件,导致配置错误,现在需要开发一套统一的少配置的日志组件,使用方便。
背景
大量项目在使用logback记日志,有部分项目使用日志混乱,格式不统一,多数人搞不懂配置文件,导致配置错误,现在需要开发一套统一的、少配置的日志组件,使用方便。
设计思路尽量采用0配置,无logback.xml
日志格式统一,方便后续日志分析系统
只有两个日志级别,一个是正常日志,一个是异常日志
提供log4j、jcl、logback、commons-log等桥接方案及版本兼容方案
提子线程、json格式化输出、map格式化、数组格式化、请求响应参数(供耗时)等便捷日志输出方法
支持redis、db、http自动开关配置**
新增日志类型(logger)api采用流式结构,类似StringBuffer
概要设计 零配置调研代码
java static LoggerContext lc; static { lc = (LoggerContext) LoggerFactory.getILoggerFactory(); // 对应配置中的appender ConsoleAppender ca = new ConsoleAppender(); ca.setContext(lc); ca.setName("console"); // 格式 PatternLayoutEncoder pl = new PatternLayoutEncoder(); pl.setContext(lc); pl.setPattern("%d{MMddHHmmss.SSS} [%thread] %-5level %logger{36} - %msg%n"); pl.start(); ca.setEncoder(pl); ca.start(); // 对应配置中的logger ch.qos.logback.classic.Logger rootLogger = lc.getLogger("com.test"); rootLogger.addAppender(ca);}
上面代码等价于下面的xml
%d{MMddHHmmss.SSS} [%thread] %-5level %logger{36} - %msg%n
由此可以随意把配置文件中的内容以代码形式编写,理论已经可以实现0配置。
输出路径约定固定将日志输出到,相对路径log/xxx.yyyy-MM-dd-HH.log,其中xxx为logger的name
日志格式格式固定:
MMddHHmmss.SSS||id||【交易名★子步骤】||context ||level
例:
150000.311||N-XrUTQzIc1531897200311||【CiTeeFilter★ci拦截器】||ci拦截器 请求的完整参数为:{"merchantId":["0012444"],"userId":["13112341232"]} ||INFO
固定格式的核心代码,拦截到日志请求,按照格式拼装,主要方法为继承ThrowableProxyConverter和MessageConverter来实现对日志的拦截,并修改为想要的格式,其中使用的例如id等放到本地变量内,核心是对MDC的使用
所有日志都默认输出到这里 logger name:service 系统初始化时,定义这个Logger和appender,即这个Logger为root log
自定义的logger提供addLogger方法,参数 packageName 包名,例如:com.test 必输参数 如果name未设置时,name默认为包名最后一个.后面的字符 name 名字,决定日志文件的名字 非必输 path 日志路径 非必输 additivity 是否输出到root log内
特殊的log
提供特殊组件的log配置,例如: redis 默认ERROR http 默认ERROR db连接池 默认ERROR kafka 默认ERROR schedul 默认ERROR spring 默认ERROR
提供exception异常栈格式打印 提供带换行的格式化打印 代码思路:继承ThrowableProxyConverter,获取异常栈,在每行的前面插入固定格式文本
普通日志api(VirgoLog)方法 | 方法描述 |
---|---|
setUniqKey(id) | 设置当前线程id,线程开始时设置即可,后面无需设置 |
updateStep(trade, step) | 更新当前id的步骤信息 |
log(msg, param) | 记录普通日志,msg替换规则,普通替换为{},如果想替换为业务日志api中的格式,使用``替换 |
logErr(msg, e) | 记录异常日志 |
log( trade, step, msg, param) | 记录普通日志,此方法会自动更新id、trade、step,不建议使用 |
logErr(trade, step, msg, e) | 记录异常日志 |
log(cid, trade, step, msg, param) | 记录普通日志,此方法会自动更新id、trade、step,不建议使用 |
logErr(cid, trade, step, msg, e) | 记录异常日志 |
debug(msg, param) | 记录debug级别日志,不建议使用 |
平时记日志时,如果某个类没有时间toString方法,会无法正确打印出数据,此时提供替换方法,直接将object替换为json打印,核心代码思路为
MessageFormatter是处理{}替换的类,重新写个类,稍加改动即支持{}也支持`` ,并判断替换为json还是toString api如下
方法 | 方法描述 |
---|---|
begin(msg) | 记录开始 |
end(msg) | 记录完成,会打印本线程内上一个begin到现在的耗时 |
logJson(json, format) | 记录json格式化日志,format表示是否换行 |
logMap(map, format) | 记录map格式化日志 |
logCollection(list, format) | 记录集合格式化日志 |
logArray(array, format) | 记录数组格式化日志 |
logObjct(obj, format) | 记录Object格式化日志 |
方法 | 方法描述 |
---|---|
getLogger() | 获取logger,用于记日志 |
getLogger(name) | 通过name获取logger |
addLogger() | 参考自定义Logger,如果logger已经创建,则不再创建,一般不使用,除非想自定义日志名等 |
consoleOpen() | 打开控制台日志,系统启动时默认配置控制台日志 |
commonOpen(name, level) | 默认的组件都是error级别,这个方法可以变更日志级别,例如redis http等 |
map:即转化为json,然后再格式化
collection:同上
array:也同上
object:同上
1、密码脱敏、加解密有必要多带带提取方法吗
2、提供父线程打印开关
maven依赖democom.cdc.ecliptic virgo 1.5_1.6-SNAPSHOT
public static void main(String[] args) throws InterruptedException { // 启动 VirgoLancher.start("hahaha", "com.cdc.virgo", "D:/test/hahah.log"); LoggerHelper.commonOpen("hahaha", LogLevel.DEBUG); Logger logger1 = LoggerFactory.getLogger("druid"); // VirgoLancher.commonStart("abc", "com.cdc.virgo"); // 打开控制台 LoggerHelper.consoleOpen(); // 设置cid VirgoLog.setUniqKey(null); // 设置步骤名和交易名 VirgoLog.updateStep("adfa", "saf"); // 获取Logger VirgoLog logger = VirgoLog.getLogger(); // 打开debug级别(只有在开发阶段可以打开) // logger.changeLevel(LogLevel.DEBUG); // 记录换行 logger.log("a"); logger1.info("ffffdffffdffffdd"); logger1.error("ffffdffffdffffdd"); // logger1.info("sfdasfaf" + // " afafdasfd" + // " asfdasf"); logger.log("sfdasfaf" + " afafdasfd" + " asfdasf"); // logger1.info("b"); // 正常日志 // logger.log("我只有一行"); Mapmap = new HashMap(); map.put("asdf", "1"); map.put("asdf2", "2"); map.put("asdf3", "13"); map.put("asdf4", "14"); map.put("asdf5", "15"); map.put("asdf6", "16"); // // 异常日志也支持格式化 // logger.logErr("我错了:{},你没错:~~", new Exception("asdfsaflk"), "啊", map); // logger.log("----------------------------------------------"); // // {}替换普通对象,调用toString() ~~把对象转换为json并且格式化输出 ``把对象转换为json不格式化输出 logger.log("你好{},你是谁~~``,sd~xx {}", map, map, map, "tttt"); VirgoLog.updateStep("saf2"); // // 把对象转换为json输出 // logger.logJson(map, false); // // 更新步骤名和交易名 // VirgoLog.updateStep("bbbbb", "ccccc"); // // 耗时日志打印 logger.begin("处理内容"); logger.begin("处理第二个"); logger.begin("处理第三个"); Thread.sleep(3000L); logger.end(); Thread.sleep(1000L); logger.end(); VirgoLog.updateStep("saf3"); logger.end(); // // 记录debug日志,一般调试用 // logger.logDebug("jajajajaja"); // List l = new ArrayList(); // B b = new B(); // try { // b.b(); // } catch (Exception e) { // logger.logErr("woqu", e); // } }
宜信技术学院
文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。
转载请注明本文地址:https://www.ucloud.cn/yun/72861.html
摘要:容器化应用日志收集挑战应用日志的收集分析和监控是日常运维工作重要的部分,妥善地处理应用日志收集往往是应用容器化重要的一个课题。日志来源识别采用统一应用日志收集方案,日志分散在很多不同容器的相互隔离的环境中,需要解决日志的来源识别问题。 容器化应用日志收集挑战 应用日志的收集、分析和监控是日常运维工作重要的部分,妥善地处理应用日志收集往往是应用容器化重要的一个课题。 Docker处理日志...
摘要:应用的研发上线运维运营形成闭环,顺利完成从对内服务到公共平台的升级。从功能角度,只能支持静态方式设置反向代理,然后,而平台有服务对应的后端服务和端口是有动态调整需求。架构上是基础组件需要进行升级,数据访问层日志监控系统等。 介绍 MaxLeap早期是一家研发、运营移动应用和手机游戏公司,发展过程中积累了很多通用组件。这些组件很大程度帮公司在移动研发过程中节省了时间和成本,...
摘要:将开发环境和生产环境的差异降至最低,并使用持续交付实施敏捷开发。可以在工具架构和开发流程不发生明显变化的前提下实现扩展。我们的初衷是分享在现代软件开发过程中发现的一些系统性问题,并加深对这些问题的认识。 简介 如今,软件通常会作为一种服务来交付,它们被称为网络应用程序,或软件即服务(SaaS)。12-Factor 为构建如下的 SaaS 应用提供了方法论: 使用标准化流程自动配置,从...
摘要:而要实现物物相连,一共有个阶段性任务,而这个阶段性任务,也伴随着巨大的挑战本文分享自华为云社区云驻共创以小窥大,从一盏路灯看亿万物联网之路云驻共创以小窥大,从一盏路灯看亿万物联网之路,作者启明。 摘要:IoT, Internet of Things,物联网,顾名思义,是物物相连。而要实现物...
阅读 1560·2021-09-02 15:41
阅读 974·2021-09-02 15:11
阅读 1231·2021-07-28 00:15
阅读 2272·2019-08-30 15:55
阅读 1120·2019-08-30 15:54
阅读 1654·2019-08-30 15:54
阅读 2923·2019-08-30 14:02
阅读 2490·2019-08-29 16:57