基于log4j的消息流的实现之一消息获取
需求:
目前的程序中都是基于log4j来实现日志的管理,想要获取日志中的一部分消息,展示给用户。
约束:
由于程序中除了自己开发的代码,还会有层层依赖的第三方jar中的日志输出。需要展示给用户的消息,也有部分包括在第三方的包中。不可能去修改第三方jar来获得消息,所以只能从Log4j本身的消息入手,获取log4j的消息来进行处理。
方案:
第一步,增加新的logger
- log4j.rootLogger=INFO, console
- 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:ss,SSS} %-5p %-20c %X{key} %x - %m%n
- log4j.logger.hermes=INFO, hermes
- log4j.appender.hermes=org.apache.log4j.ConsoleAppender
- log4j.appender.hermes.layout=org.apache.log4j.PatternLayout
- log4j.appender.hermes.layout.ConversionPattern=%X{key} %x %m%n
- log4j.logger.org.apache.flink.yarn=INFO, hermes
如上,第1行定义了一个全局的logger,这个没有疑义,所有的日志都会输出到这个logger。
关键是第7行,又定义了一个叫hermes的logger,它有一个同名的appender叫hermes,简化了他的layout,只是拿出必须的输出日志即可。注意这里有一个%X{key}的配置项,这里是为了拿到MDC的值,好区分这条消息是哪个session打印出来的。
第12行也很关键,将包org.apache.flink.yarn的日志都按照hermes是标准输出,这里是包名,当然也可以指定到具体的类名。
如此这样,对于包org.apache.flink.yarn中的日志,会打印两份,一份是以console的形式输出,一份是以hermes的形式输出。
当然,我们不希望在日志中见到两行实质上一样的内容,这就是第二步要做到事情。
第二步:截取消息
这里有两个方案:一个方案是自定义appender,获取消息;一个方案是获取指定的logger,获取消息。
方案1:自定义appender
- import org.apache.log4j.AppenderSkeleton;
- import org.apache.log4j.LogManager;
- import org.apache.log4j.spi.LoggingEvent;
- import org.slf4j.Logger;
- import org.slf4j.LoggerFactory;
- public class AdapterAppender extends AppenderSkeleton {
- @Override
- protected void append(LoggingEvent loggingEvent) {
- String message = this.layout.format(loggingEvent);
- System.out.println("<<<<<<<<<<***********"+message+"***********>>>>>>>>>>");
- }
- @Override
- public void close() {
- }
- @Override
- public boolean requiresLayout() {
- return true;
- }
- }
自定义appender还是很简单的,只要继承AppenderSkeleton即可。其中的append方法就可以拿到日志消息,标准化之后就可以拿来用了。
当然,上面log4j的配置也就需要改下了:
- log4j.appender.hermes=cn.123.flink.log.AdapterAppender
方案2:获取logger来获取消息
- import java.io.IOException;
- import java.io.PipedReader;
- import java.io.PipedWriter;
- import java.io.Writer;
- import java.util.Scanner;
- import org.apache.log4j.*;
- public class LogAppender extends Thread{
- private PipedReader reader;
- public LogAppender(String appenderName) {
- try {
- Logger root = Logger.getLogger(appenderName);
- // 获取子记录器的输出源
- Appender appender = root.getAppender(appenderName);
- // 定义一个未连接的输入流管道
- reader = new PipedReader();
- // 定义一个已连接的输出流管理,并连接到reader
- Writer writer = new PipedWriter(reader);
- // 设置 appender 输出流
- ((WriterAppender) appender).setWriter(writer);
- }catch (IOException ioe){
- ioe.printStackTrace();
- }
- }
- @Override
- public void run() {
- // 不间断地扫描输入流
- Scanner scanner = new Scanner(reader);
- // 将扫描到的字符流显示在指定的JLabel上
- while (scanner.hasNextLine()) {
- try {
- String line = scanner.nextLine();
- System.out.println("*****************************"+line);
- //睡眠
- Thread.sleep(100);
- } catch (Exception ex) {
- System.out.println("Exception from LogAppender:"+ex.getMessage());
- }
- }
- }
- }
对比上面的日志的配置,13行的构造函数就可以传入hermes来获取hermes这个logger。
参考了下面的文档:
http://logging.apache.org/log4j/1.2/apidocs/org/apache/log4j/PatternLayout.html
http://logging.apache.org/log4j/1.2/apidocs/org/apache/log4j/MDC.html
http://logging.apache.org/log4j/1.2/apidocs/org/apache/log4j/NDC.html
https://logging.apache.org/log4j/2.x/manual/thread-context.html
http://yshjava.iteye.com/blog/1325036
https://stackoverflow.com/questions/2763740/log4j-log-output-of-a-specific-class-to-a-specific-appender
https://stackoverflow.com/questions/5549838/get-live-log4j-messages
https://sysgears.com/articles/how-to-redirect-stdout-and-stderr-writing-to-a-log4j-appender/
基于log4j的消息流的实现之一消息获取的更多相关文章
- Storm概念学习系列之Stream消息流 和 Stream Grouping 消息流组
不多说,直接上干货! Stream消息流是Storm中最关键的抽象,是一个没有边界的Tuple序列. Stream Grouping 消息流组是用来定义一个流如何分配到Tuple到Bolt. Stre ...
- Stream消息流 和 Stream Grouping 消息流组
- 基于log4j的消息流的实现之二消息传递
在“基于log4j的消息流的实现之一消息获取”中获取日志消息的部分,修改如下: import org.apache.commons.collections.map.HashedMap; import ...
- [蓝牙] 6、基于nRF51822的蓝牙心率计工程消息流Log分析(详细)
开机初始化Log Log编号 函数名 所在文件名 000001: main ..\main.c 000002: timers_init ..\main.c 000003: gpiote_init ...
- 开源一个自己造的轮子:基于图的任务流引擎GraphScheduleEngine
GraphScheduleEngine是什么: GraphScheduleEngine是一个基于DAG图的任务流引擎,不同语言编写.运行于不同机器上的模块.程序,均可以通过订阅GraphSchedul ...
- BTARN 接收消息流以3A7为例
 1.RNIFReceive.aspx 页接收来自发起方的传入消息. (如果发起方是BizTalk则类似于:http://localhost/BTARNApp/RNIFSend.aspx?TPUrl ...
- 大数据平台消息流系统Kafka
Kafka前世今生 随着大数据时代的到来,数据中蕴含的价值日益得到展现,仿佛一座待人挖掘的金矿,引来无数的掘金者.但随着数据量越来越大,如何实时准确地收集并分析如此大的数据成为摆在所有从业人员面前的难 ...
- IM系统-消息流化一些常见问题
原创不易,求分享.求一键三连 之前说过IM系统的一些优化,但是在网络上传输数据对于数据的流化和反流化也是处理异常情况的重点环节,不处理好可能会出现一些消息发送成功,但是解析失败的情况,本文就带大家来一 ...
- 基于SQL Server 2008 Service Broker构建企业级消息系统
注:这篇文章是为InfoQ 中文站而写,文章的地址是:http://www.infoq.com/cn/articles/enterprisemessage-sqlserver-servicebroke ...
随机推荐
- Super Reduced String
https://www.hackerrank.com/challenges/reduced-string/problem He wants to reduce the string to its sh ...
- 让微信内置浏览器兼容clipboard.js 复制粘贴 ios 安卓
<!--js copy事件--><script type="text/javascript" src="/static/js/clipboard.min ...
- Angular : IOC的方式:依赖注入
依赖注入 @Component, @Injectable 可以允许别的声明在providers里面的Service等注入到被这两个装饰器装饰的类中 Service等可以被声明在app-module.t ...
- BGP(边界网关协议)简述
BGP的起源 不同自治系统(路由域)间路由交换与管理的需求推动了EGP的发展,但是EGP的算法简单,无法选路,从而被BGP取代. 自治系统:(AS) IGP:自治系统内部协议,ospf,rip,is- ...
- STM32进阶之串口环形缓冲区实现(转载)
转载自微信公众号“玩转单片机”,感谢原作者“杰杰”. 队列的概念 在此之前,我们来回顾一下队列的基本概念:队列 (Queue):是一种先进先出(First In First Out ,简称 FIFO) ...
- 洛谷P1364 医院设置
LITTLESUN的第一道图论,撒花~~ 题目链接 这道题是Floyd的板子题 注意对于矩阵图的初始值赋值要全部赋值成最大值 十六进制的最大值表示方式是0x3f3f3f3f memset(G,0x3f ...
- Spring常用注解用法总结
转自http://www.cnblogs.com/leskang/p/5445698.html 1.@Controller 在SpringMVC 中,控制器Controller 负责处理由Dispat ...
- 做小Leader的心得体会
只是自己的工作心得体会,代码属于也不够专业,大家不喜勿喷. 8月份来到这家新公司,没过一个月给派了个活:带着两个小弟给某银行开发一个小工具.功能很简单,就是用Java做一个windows上的C端工具, ...
- 13.0 Excel表格写入
Excel表格写入 安装 xlutils 和 xlwt Excel写入输入 分两种方式: 第一种是向一张新表之中写入..这种不多说,我几乎没怎么用,直接贴代码 import xlwt Excel_na ...
- Struts2(三.用户登录状态显示及Struts2标签)
1.编写main.jsp /WebContent/main.jsp 之前用户登录时已把用户存入session <%@ page language="java" content ...