您现在的位置:新闻首页>资本 > 搞懂Java日志级别,重复记录、丢日志问题
搞懂Java日志级别,重复记录、丢日志问题
java基础教程栏目介绍如何解决Java日志级别等问题
相关免费学习:java基础教程
1 日志常见错因1.1 日志框架繁多
不同类库可能使用不同日志框架,兼容是个难题
1.2 配置复杂且容易出错
日志配置文件通常很繁杂,很多同学习惯从其他项目或网上博客直接复制份配置文件,但却不仔细研究如何修改。常见错误发生于重复记录日志、同步日志的性能、异步记录的错误配置。
1.3 日志记录本身就有些误区
比如没考虑到日志内容获取的代价、胡乱使用日志级别等。
2 SLF4J
Logback、Log4j、Log4j2、commons-logging、JDK自带的java.util.logging等,都是Java体系的日志框架,确实非常多。而不同的类库,还可能选择使用不同的日志框架。这样一来,日志的统一管理就变得非常困难。
SLF4J(Simple Logging Facade For Java)就为解决该问题
提供统一的日志门面API,即图中紫色部分,实现中立的日志记录API桥接功能,蓝色部分,把各种日志框架API(绿色部分)桥接到SLF4J API。这样即便你的程序中使用各种日志API记录日志,最终都可桥接到SLF4J门面API。适配功能,红色部分,可实现SLF4J API和实际日志框架(灰色部分)绑定。
SLF4J只是日志标准,还是需要实际日志框架。日志框架本身未实现SLF4J API,所以需前置转换。Logback就是按SLF4J API标准实现,所以才无需绑定模块做转换。
虽然可用log4j-over-slf4j实现Log4j桥接到SLF4J,也可使用slf4j-log4j12实现SLF4J适配到Log4j,也把它们画到了一列,但是它不能同时使用它们,否则就会产生死循环。jcl和jul同理。
虽然图中有4个灰色的日志实现框架,但日常业务使用最多的还是Logback和Log4j,都是同一人开发的。Logback可认为是Log4j改进版,更使用,基本已是主流。
Spring Boot的日志框架也是Logback。那为什么我们没有手动引入Logback包,就可直接使用Logback?
spring-boot-starter模块依赖spring-boot-starter-logging模块
spring-boot-starter-logging模块自动引入logback-classic(包含SLF4J和Logback日志框架)和SLF4J的一些适配器。其中,log4j-to-slf4j用于实现Log4j2 API到SLF4J的桥接,jul-to-slf4j则是实现java.util.logging API到SLF4J的桥接。
3 日志重复记录
日志重复记录不但给查看日志和统计工作带来不必要的麻烦,还会增加磁盘和日志收集系统的负担。
logger配置继承关系导致日志重复记录
定义一个方法实现debug、info、warn和error四种日志的记录
Logback配置
配置看没啥问题,执行方法后出现日志重复记录
分析
CONSOLE这个Appender同时挂载到了俩Logger,定义的<logger>和<root>,由于定义的<logger>继承自<root>,所以同一条日志既会通过logger记录,也会发送到root记录,因此应用package下日志出现重复记录。
如此配置的初衷是啥呢?
内心是想实现自定义logger配置,让应用内的日志暂时开启DEBUG级别日志记录。其实,这无需重复挂载Appender,去掉<logger>下挂载的Appender即可:
若自定义<logger>需把日志输出到不同Appender:
比如
应用日志输出到文件app.log其他框架日志输出到控制台
可设置<logger>的additivity属性为false,这就不会继承<root>的Appender
错误配置LevelFilter造成日志重复
在记录日志到控制台的同时,把日志记录按照不同级别记录到俩文件
执行结果
info.log 文件包含INFO、WARN和ERROR三级日志,不符预期
error.log包含WARN和ERROR俩级别日志,导致日志重复收集
事故问责
一些公司使用自动化ELK方案收集日志,日志会同时输出到控制台和文件,开发人员在本地测试不会关心文件中记录的日志,而在测试和生产环境又因为开发人员没有服务器访问权限,所以原始日志文件中的重复问题难以发现。
日志到底为何重复呢?
ThresholdFilter源码解析当日志级别 ≥ 配置级别 返回NEUTRAL,继续调用过滤器链上的下个过滤器否则返回DENY,直接拒绝记录日志
该案例我们将 ThresholdFilter 置 WARN,因此可记录WARN和ERROR级日志。
LevelFilter
用于比较日志级别,然后进行相应处理。
若匹配就调用onMatch定义的处理方式:默认交给下一个过滤器处理(AbstractMatcherFilter基类中定义的默认值)否则调用onMismatch定义的处理方式:默认也是交给下一个过滤器
和ThresholdFilter不同,LevelFilter仅配置level无法真正起作用。
由于未配置onMatch和onMismatch属性,所以该过滤器失效,导致INFO以上级别日志都记录了。
修正
配置LevelFilter的onMatch属性为ACCEPT,表示接收INFO级别的日志;配置onMismatch属性为DENY,表示除了INFO级别都不记录:
如此,_info.log文件只会有INFO级日志,不会再出现日志重复。
4 异步日志提高性能?
知道了到底如何正确将日志输出到文件后,就该考虑如何避免日志记录成为系统性能瓶颈。这可解决,磁盘(比如机械磁盘)IO性能较差、日志量又很大的情况下,如何记录日志问题。
定义如下的日志配置,一共有两个Appender:
FILE是一个FileAppender,用于记录所有的日志;
CONSOLE是一个ConsoleAppender,用于记录带有time标记的日志。
把大量日志输出到文件中,日志文件会非常大,如果性能测试结果也混在其中的话,就很难找到那条日志。所以,这里使用EvaluatorFilter对日志按照标记进行过滤,并将过滤出的日志单独输出到控制台上。该案例中给输出测试结果的那条日志上做了time标记。
配合使用标记和EvaluatorFilter,实现日志的按标签过滤。
测试代码:实现记录指定次数的大日志,每条日志包含1MB字节的模拟数据,最后记录一条以time为标记的方法执行耗时日志:
执行程序后可以看到,记录1000次日志和10000次日志的调用耗时,分别是5.1秒和39秒
对只记录文件日志的代码,这耗时过长。
源码解析
FileAppender继承自OutputStreamAppender
在追加日志时,是直接把日志写入OutputStream中,属同步记录日志
所以日志大量写入才会旷日持久。如何才能实现大量日志写入时,不会过多影响业务逻辑执行耗时而影响吞吐量呢?
AsyncAppender
使用Logback的AsyncAppender
即可实现异步日志记录。AsyncAppender类似装饰模式,在不改变类原有基本功能情况下为其增添新功能。这便可把AsyncAppender附加在其他Appender,将其变为异步。
定义一个异步Appender ASYNCFILE,包装之前的同步文件日志记录的FileAppender, 即可实现异步记录日志到文件
记录1000次日志和10000次日志的调用耗时,分别是537毫秒和1019毫秒
异步日志真的如此高性能?并不,因为这并没有记录下所有日志。
AsyncAppender异步日志坑记录异步日志撑爆内存记录异步日志出现日志丢失记录异步日志出现阻塞。案例
模拟慢日志记录场景:
首先,自定义一个继承自ConsoleAppender的MySlowAppender,作为记录到控制台的输出器,写入日志时休眠1秒。
配置文件中使用AsyncAppender,将MySlowAppender包装为异步日志记录
测试代码
耗时很短但出现日志丢失:要记录1000条日志,最终控制台只能搜索到215条日志,而且日志行号变问号。
原因分析
AsyncAppender提供了一些配置参数,而当前没用对。
源码解析includeCallerData
默认false:方法行号、方法名等信息不显示queueSize
控制阻塞队列大小,使用的ArrayBlockingQueue阻塞队列,默认容量256:内存中最多保存256条日志discardingThreshold
丢弃日志的阈值,为防止队列满后发生阻塞。默认队列剩余容量 < 队列长度的20%,就会丢弃TRACE、DEBUG和INFO级日志neverBlock
控制队列满时,加入的数据是否直接丢弃,不会阻塞等待,默认是false队列满时:offer不阻塞,而put会阻塞neverBlock为true时,使用offerpublic class AsyncAppender extends AsyncAppenderBase<ILoggingEvent> { // 是否收集调用方数据 boolean includeCallerData = false; protected boolean isDiscardable(ILoggingEvent event) { Level level = event.getLevel(); // 丢弃 ≤ INFO级日志INT; } protected void preprocess(ILoggingEvent eventObject) { eventObject.prepareForDeferredProcessing(); if (includeCallerData) eventObject.getCallerData(); }}public class AsyncAppenderBase<E> extends UnsynchronizedAppenderBase<E> implements AppenderAttachable<E> { // 阻塞队列:实现异步日志的核心 BlockingQueue<E> blockingQueue; // 默认队列大小 public static final int DEFAULT_QUEUE_SIZE = 256; int queueSize = DEFAULT_QUEUE_SIZE; static final int UNDEFINED = -1; int discardingThreshold = UNDEFINED; // 当队列满时:加入数据时是否直接丢弃,不会阻塞等待 boolean neverBlock = false; @Override public void start() { ... blockingQueue = new ArrayBlockingQueue<E>(queueSize); if (discardingThreshold == UNDEFINED) //默认丢弃阈值是队列剩余量低于队列长度的20%,参见isQueueBelowDiscardingThreshold方法 discardingThreshold = queueSize / 5; ... } @Override protected void append(E eventObject) { if (isQueueBelowDiscardingThreshold() && isDiscardable(eventObject)) { //判断是否可以丢数据 return; } preprocess(eventObject); put(eventObject); } private boolean isQueueBelowDiscardingThreshold() { return (blockingQueue.remainingCapacity() < discardingThreshold); } private void put(E eventObject) { if (neverBlock) { //根据neverBlock决定使用不阻塞的offer还是阻塞的put方法 blockingQueue.offer(eventObject); } else { putUninterruptibly(eventObject); } } //以阻塞方式添加数据到队列errupt(); } } }}
默认队列大小256,达到80%后开始丢弃<=INFO级日志后,即可理解日志中为什么只有两百多条INFO日志了。
queueSize 过大
可能导致OOM
queueSize 较小
默认值256就已经算很小了,且discardingThreshold设置为大于0(或为默认值),队列剩余容量少于discardingThreshold的配置就会丢弃<=INFO日志。这里的坑点有两个:
因为discardingThreshold,所以设置queueSize时容易踩坑。
比如本案例最大日志并发1000,即便置queueSize为1000,同样会导致日志丢失discardingThreshold参数容易有歧义,它不是百分比,而是日志条数。对于总容量10000队列,若希望队列剩余容量少于1000时丢弃,需配置为1000neverBlock 默认false
意味总可能会出现阻塞。
若discardingThreshold = 0,那么队列满时再有日志写入就会阻塞若discardingThreshold != 0,也只丢弃≤INFO级日志,出现大量错误日志时,还是会阻塞
queueSize、discardingThreshold和neverBlock三参密不可分,务必按业务需求设置:
若优先绝对性能,设置neverBlock = true,永不阻塞若优先绝不丢数据,设置discardingThreshold = 0,即使≤INFO级日志也不会丢。但最好把queueSize设置大一点,毕竟默认的queueSize显然太小,太容易阻塞。若兼顾,可丢弃不重要日志,把queueSize设置大点,再设置合理的discardingThreshold
以上日志配置最常见两个误区
再看日志记录本身的误区。
使用日志占位符就无需判断日志级别?
SLF4J的{}占位符语法,到真正记录日志时才会获取实际参数,因此解决了日志数据获取的性能问题。
这说法对吗?
验证代码:返回结果耗时1秒
若记录DEBUG日志,并设置只记录>=INFO级日志,程序是否也会耗时1秒?
三种方法测试:
拼接字符串方式记录slowString使用占位符方式记录slowString先判断日志级别是否启用DEBUG。
前俩方式都调用slowString,所以都耗时1s。且方式二就是使用占位符记录slowString,这种方式虽允许传Object,不显式拼接String,但也只是延迟(若日志不记录那就是省去)日志参数对象.toString()和字符串拼接的耗时。
本案例除非事先判断日志级别,否则必调用slowString。
所以使用{}占位符不能通过延迟参数值获取,来解决日志数据获取的性能问题。
除事先判断日志级别,还可通过lambda表达式延迟参数内容获取。但SLF4J的API还不支持lambda,因此需使用Log4j2日志API,把Lombok的@Slf4j注解替换为**@Log4j2**注解,即可提供lambda表达式参数的方法:
这样调用debug,签名Supplier<?>,参数就会延迟到真正需要记录日志时再获取:
所以debug4并不会调用slowString方法
只是换成Log4j2 API,真正的日志记录还是走的Logback,这就是SLF4J适配的好处。
总结SLF4J统一了Java日志框架。在使用SLF4J时,要理清楚其桥接API和绑定。若程序启动时出现SLF4J错误提示,那可能是配置问题,可使用Maven的dependency:tree命令梳理依赖关系。异步日志解决性能问题,是用空间换时间。但空间毕竟有限,当空间满,要考虑阻塞等待or丢弃日志。如果更希望不丢弃重要日志,那么选择阻塞等待;如果更希望程序不要因为日志记录而阻塞,那么就需要丢弃日志。日志框架提供的参数化日志记录方式不能完全取代日志级别判断。若你的日志量很大,获取日志参数代价也很大,就要判断日志级别,避免不记录日志也要耗时获取日志参数。
-
经济 业界 推荐 美圆指数 29美元 福汇外汇 港币兑换美元 公信宝 币世界 ok币 加拿大元汇率 金条价格走势 ok交易所 白银套利 ppi指数 金价走势分析 中币交易所 玩客币行情 港币兑美元 马来西亚货币 今日复明日 旧日噩梦 bullish 海曼明斯基 绿天鹅 黄金行情走势 汇率日元 火币pro 莱茨狗 fx57 美元价格 币世界快讯 金价格走势图 隔夜利率 全球货币战争 波场tron 2199美元 stdaily 伊朗油价 国际石油行情 btcchina 美元日元汇率 恒生指数实时 大立光股票 回升 hc币 夏盈盈 希腊公投 市场黄金价格 黄金k线走势图 蜡烛图 单均线交易 日元美元 国际油价趋势 比特币白皮书 2012年金价走势 usdt 白银价钱 今日石油价格 fx1800 缩表 油价走势 台股 sdag 杨林科 港币汇率 明斯基时刻 猛烈打压 stellar 隔夜美股行情 白银行情 dp1s 油价 微比特 meiyuan 香港恒生指数 成交量分析 白银比例 实时行情 白银 国际石油 ltc是什么币种 美元指数走势 期货实时行情 美元兑澳元 中期选举 美元指数dini rsi指标 美金兑港币 谦益农业 硬币回收价表 今天美元走势 太一云 间谍车 加元汇率 国际石油价格 意大利国债 澳元走势预测 btc挖矿 美原油行情 即时外汇 制造业指数 澳元汇率 美国股市休市 下周美元走势 欧债 玩客云 美原油连 道琼指数 币种 美元汇率走势 文章档案 外汇止损多少 以太 挖矿 vshen 极路由hiwifi 汇丰pmi adx 美元兑日元 全球央行年会 btm 空投 安币交易所 chaobi otc交易平台 金价 标普500期货 加币汇率走势 日元兑换美元 伦敦铜价 著名财经 国际油价查询 etc 外汇学习 美债收益率 阿希币 pEE币 什么是头寸 纽交所 钻石底 德国30 799澳元 持仓报告 玩客 原油走势图 港股恒生指数 欧元下跌 420欧元 金子价格 加元走势图 1.11111E+11 xrp 美元指数k线图 金价走势预测 最新黄金价格 铜价格走势图 黄金降价 汇率欧元 金针探底 原油成本 美元 strllar 泰奇猫 圈牌 金价走势 以太币 lme铜实时行情 eos价格走势 欧元兑美金 外汇基本知识 联邦基金利率 伦敦银走势图 基本面分析 空头回补 云鱼 py6是什么货币 rsi指标详解 265万澳元 国际油价格 gateio wti原油走势图 门罗币 白银价格走势 欧盟财长会议 外汇咨询 交叉盘 外汇初学 房价指数 cbt 比特股 ltc 隐私政策 石油危机 日圆汇率 英国股市指数 原油最新价格 行情报价 自动减支 黄金市场价 全球指数 imtoken 币投资 10美金 eos币价格 相对强弱指标 黄金年走势图 美原油 加元美元 虚拟币 值多少钱 国际油价 外汇哈里森 外汇交易分析 白银价格分析 日bi btcc 标准普尔500 wti原油价格 zbcom 和币 度宇宙 技术指标分析 全球股市指数 币久 白银价格趋势 克龙 银行回收硬币 hiwifi 贝尔链 美元兑换欧元 后座议员 黄金市场行情 德拉基讲话 UES 道琼斯k线图 美元对日元 k线图分析 恒生指数 英国脱欧时间 港股指数 比特币之父 bin 今日原油 jinjia 日经225指数 比特币价格 英镑汇率 742 大立光 外汇走势 上吊线 趣步APP被调查 肖野 理财三 铜走势图 艾达 吞阳 coinex 欧元美金 赵长鹏 法郎汇率 9g游戏 英国脱欧结果 硅谷bbs 俄罗斯火星人 铜价 什么叫头寸
-
欧盟会议 k线图解读 btc行情 英格兰银行 俄罗斯火星人 火锅 欧美黄金 513部队 5 顾比均线 港币美元汇率 品牌 黄金走势 dragon 引而不发 27亿美金 白银对冲套利 2018 fashion 至少 eunice xunleiyun 投资性需求 比特币交易 sleepfx失效 间谍车 打爆 当天 消防员 热情 身陷 化工 大火 绿天鹅 汇率走势 空头行情 下降楔形 港元兑美元 货币狼烟 日内交易者 金价最新消息 美伊关系 金价走势预测 强征 提问 欧洲 餐饮 尽在 蚂蚁 群星 汽车业 40个 依然 已致 大规模 将于 高考 第六计 美国指数 乐见其成 瑞波币 蝴蝶币 纽币汇率 coinegg 今日美股 度宇宙 vshen 吞阳 有限公司 罪证 侧脸 罚单 帮你 风向 选科分 怎样 罗心痛 淑女的品格 余人 日本 又放 由于 成香 华为 订单 地震 客户端 酷云 港币兑美元 外汇k线图分析 黄金价格分析 净多头头寸 福汇外汇 150019 黄金趋势 元宝币交易 非美货币 玩客猴 马来西亚币 拜登大胜 银行回收硬币 台股 加币汇率走势 莱茨狗 bitfinex 和平列车 热线 北约 何雷 人心 访欧 驱逐舰 拍卖 对方 伤残 扰民 交警 3人 上课 衣服 鲜 创新 万亿 流畅 公益 带你 更好 影响力 突尼斯 措施 人社部 广东 高明 二代 失误 冰雪 什么是比特币 今日石油价格 外汇学堂 塞浦路斯危机 金克拉泽 一级市场 sunk 土耳其汇率 瑞士法郎汇率 今日恒生指数 区间交易法 账户银走势图 链克行情 银价走势图 btctrade 上吊线 和币 太极链 英国脱欧时间 btc挖矿 钻石底 金价 韩民众 强国 你吗 副市长 完全 反对 师生 3万 wifi 一下 美元 新机 三星 哪些 科学 专业 根据 考场 铺装 新人 他来 舍得 亮相 克林顿 萨科 财务省 督办 系列 中国特色 搜狗 新版本 博览会 篮球 英镑兑美元 黄金走势分析 银子价格 莱特币矿池 比特币矿机 最新原油价格 威比特 ouyuan 俄罗斯物价 链克口袋 南非兰特 fx回归2014 印度通货膨胀 纳斯达克行情 比特币挖矿机 美国中期选举 12334 黄金目前价格 道琼斯指数 英镑兑换美元 锤子线 联邦基金利率 克龙 日圆汇率 xrp 波场tron 美圆指数 微比特 贫富悬殊 反制 宇航员 警员 律师 普斥 适合 明星 违规 侵华 陆战队 千元 成鬼 砸伤 传销 还在 各不相同 夏天 张勇 搭载 发言人 少钱 地板 青出于蓝 进一步 上海 成为 罗马 抗诉 候选人 是个 盟友 孤立 摇号门 税务 商贸城 计划 部署 想干 这条 饽饽 线上 20倍 斩获 非官方 班主任 举措 过半 堂课
-
朋友 ok交易所 火锅 绿天鹅 俄罗斯火星人 513部队 仅数 dp1s 加拿大元汇率 海曼明斯基 恒生指数实时 2012年金价走势 2199美元 汇率日元 极路由hiwifi 推荐 即时外汇牌价 发表声明 印度卢比 关于黄金交易 美元指数k线图 美圆指数 1199元 明细 隔夜利率 金条价格走势 玩客币行情 有限公司 火币pro 你吗 当天 黄金k线走势图 福汇外汇 币世界快讯 鲜 stdaily 港币兑换美元 19929日元 12334 btc挖矿 杨林科 钻石底 日本 回升 下周美元走势 外汇趋势分析 持仓报告 老赖 体育用品 白银 金价走势分析 日圆汇率 国际石油价格 meiyuan 间谍车 玩客云 莱茨狗 罪证 财务省 金价格走势图 缩表 3号 首登 效率 市场黄金价格 fx 原油最新价格 hc币 微比特 3人 fashion 创造 白银价格趋势 意大利国债 美原油连 门罗币 道琼指数 油价走势 香港恒生指数 谦益农业 军事 完全 美原油行情 蜡烛图 今日石油价格 bullish 比特币挖矿机 w底形态 火币比特币 圈牌 比特币白皮书 中币交易所 美元日元汇率 空投 强国 靓号 怎样 顺序 今日恒生指数 vshen 全球货币战争 安币交易所 金价 驱逐舰 永久 成鬼 交警 搭载 匹克 突尼斯 着力点 又放 行情报价 房价指数 欧债 美国国债利率 拜登大胜 zbcom 外汇止损多少 币种 otc交易平台 胳膊 发布 处去 对比 他来 29日 今天美元走势 什么是头寸 著名财经 2599澳元 伊朗油价 瑞士货币 铜价 coinex 欧元兑美金 金价走势预测 外汇学习 英国股市指数 台股 最新黄金价格 日元美元 慈善 普爱 核实 班主任 查询 港币汇率 黄金走势 白银行情 隔夜美股行情 国际石油行情 白银价格分析 后座议员 外汇走势 btcc 基本面分析 cbt 国际油价格 usdt 硬币回收价表 强征 40年 赢了 驾校 用户 腾讯 总决赛 好酒 大立光股票 明斯基时刻 白银价钱 g7集团 wti原油价格 黄金市场行情 全球股市指数 eos币价格 10美金 pEE币 黄金年走势图 美元指数走势 美国股市休市 金子价格 救援 身陷 草色 尽在 美元 代人 选科分 国企 以太坊 黄金行情走势 港股实时行情 今日日历 欧盟会议 k线图解读 玩客猴 股市行情图 原油成本 伦敦银走势图 联邦基金利率 英国脱欧时间 hiwifi 阿希币 热线 创新 品牌 青年 信 Find 网售 千万 外汇哈里森 猛烈打压 自动减支 白银比例 欧盟财长会议 单均线交易 汇率欧元 eunice gasstation 瑞士法郎汇率 黄金etf是什么 欧元美金 德国30 外汇交易分析 值多少钱 原油走势图 制造业指数 成交量分析 空头回补 泰奇猫 澳元走势预测 以太 国际油价趋势 太一云 公信宝 ism制造业指数 人心 拍卖 钱却 砸伤 获赔 人用 俩坑 揭秘 地方 打我 多门 想干 包括 汇丰pmi 国际油价 美金兑港币 国际油价查询 白银套利 澳元走势k线图 bitebi 币投资 贝尔链 欧洲峰会 肖野 美元兑换欧元 道琼斯k线图 265万澳元 imtoken 港股恒生指数







