java日志篇(2)-JUL(java.util.logging)

慢慢来比较快,虚心学技术

原文链接:java日志篇(2)--JUL(java.util.logging)

前言:之所以未按照前一篇文章 java日志篇(1)-日志概述所描述的日志发展史进行文章编写,是因为JUL是jdk自带的日志实现,其他日志实现原理大同小异,理解起来也是相对简单,如有兴趣可先去看看上一篇文章

JUL是什么?

JUL是jdk自带的一个日志实现,使用简单,控制灵活,如果是小型系统或者测试程序,显然JUL会是比log4j等更好的选择

JUL基本使用

使用java.util.logging.Logger的静态方法getLogger(loggerName)来获取或创建一个日志logger

import java.io.IOException;
import java.util.logging.*;

public class JULTest {

     public static Logger log = Logger.getLogger("TestLog");  //获取日志对象

    public static void main(String[] args) throws IOException {
        log.info("info");    //信息日志
        log.warning("warning"); //警告日志
        log.log(Level.SEVERE,"server"); //严重日志
        log.fine("fine");
    }
}

使用日志我们可以使用默认的等级实现方法进行输出:如log.info("****")等,也可以使用特定的等级进行输出:如log.log(等级,输出信息)

如上代码输出日志结果如下:

一月 14, 2019 9:20:32 下午 JULTest main
信息: info
一月 14, 2019 9:20:32 下午 JULTest main
警告: warning
一月 14, 2019 9:20:32 下午 JULTest main
严重: server

JUL使用分析

一、Logger

为什么log.fine("fine");没有输出?

我们首先看一下第一个问题,需要先了解一下JUL的日志等级划分:

JUL日志等级划分(优先级递减)及内置代表的整数如下:

  • OFF(Integer.MAX_VALUE)
  • SEVERE(1000)
  • WARNING(900)
  • INFO(800)
  • CONFIG(700)
  • FINE(500)
  • FINER(400)
  • FINEST(300)
  • ALL(Integer.MIN_VALUE)

为了更好更灵活的控制日志的输出,jul使用如上的9个等级作为日志等级划分,其中常用的日志等级主要是server,warning和info,当为 Logger 指定了一个 Level, 该 Logger 会包含当前指定级别以及更高级别的日志,logger默认的级别是INFO,比INFO更低的日志将不显示。由上述输出结果不难看出,只是输出了info及以上等级的日志,而自动忽略info等级以下的日志内容,为什么会有这样的实现呢?此处涉及到JUL的默认配置文件loging.properties,该配置文件位于jdk安装目录的lib包下

handlers= java.util.logging.ConsoleHandler

.level= INFO

java.util.logging.FileHandler.pattern = %h/java%u.log
java.util.logging.FileHandler.limit = 50000
java.util.logging.FileHandler.count = 1
java.util.logging.FileHandler.formatter = java.util.logging.XMLFormatter

java.util.logging.ConsoleHandler.level = INFO
java.util.logging.ConsoleHandler.formatter = java.util.logging.SimpleFormatter

com.xyz.foo.level = SEVERE

logger.log(Level.FINE,"file");调用方法如下

public void log(Levellevel, String msg) {
          if (level.intValue() < levelValue ||levelValue == offValue) {
              return;
          }
          LogRecord lr = new LogRecord(level, msg);
          doLog(lr);
}

其中,我们可以看到.level属性指定了logger默认的日志级别为info,而logger类会先读取默认的日志级别,当选定的日志level整数值低于默认值,就不会进行日志输出

所以我们如果要更改logger的输出等级,可以通过修改对应配置项的level等级,也可以通过代码的方式去动态设置logger的等级

如下:结果输出为所有日志信息

import java.io.IOException;
import java.util.logging.*;

public class JULTest {

     public static Logger log = Logger.getLogger("TestLog");  //获取日志对象

    public static void main(String[] args) throws IOException {

        log.setLevel(Level.ALL);//设置logger的日志级别为全部,默认输出所有级别日志信息

        log.info("info");    //信息日志
        log.warning("warning"); //警告日志
        log.log(Level.SEVERE,"server"); //严重日志
        log.fine("fine");
    }
}

输出结果:
一月 15, 2019 12:12:23 上午 JULTest main
信息: info
一月 15, 2019 12:12:23 上午 JULTest main
警告: warning
一月 15, 2019 12:12:23 上午 JULTest main
严重: server
一月 15, 2019 12:12:23 上午 JULTest main
较详细: fine

二、Handler

既已引出logging.properties文件,我们就接着以此作为分析入口

由文件内容可以看到上一篇文章中介绍的handler组件的配置

默认Handler:

JUL中用的比较多的是两个Handler类:ConsoleHandler和FileHandler

其中,ConsoleHandler是对控制台输出的默认处理类,FileHandler是对文件输出的默认处理类

虽然我们可以通过修改配置文件更改系统日志的处理方式,但是更灵活的方式是从代码进行更改

举例:1、更改控制台Handler处理类的默认级别

import java.io.IOException;
import java.util.logging.*;

public class JULTest {

     public static Logger log = Logger.getLogger("TestLog");  //获取日志对象

    public static void main(String[] args) throws IOException {

        log.setLevel(Level.ALL);//设置logger的日志级别为全部,默认输出所有级别日志信息

        log.setUseParentHandlers(false); //禁用日志原本处理类

        ConsoleHandler consoleHandler = new ConsoleHandler(); //创建控制台输出控制Handler
        consoleHandler.setLevel(Level.INFO); //设置控制台输出级别
        log.addHandler(consoleHandler); //将Handler加入logger中

        log.info("info");    //信息日志
        log.warning("warning"); //警告日志
        log.log(Level.SEVERE,"server"); //严重日志
        log.fine("fine");
    }
}

输出结果:
一月 15, 2019 12:03:56 上午 JULTest main
信息: info
一月 15, 2019 12:03:56 上午 JULTest main
警告: warning
一月 15, 2019 12:03:56 上午 JULTest main
严重: server

上面的代码新建了一个控制台Handler,级别设置为INFO,所以结果又变成了只输出INFO及以上级别的日志

注:如果没有 log.setUseParentHandlers(false); 父Handler与子Handler都会生效,此时会输出两遍日志内容

2、更改文件FileHandler处理类的处理方式

import java.io.IOException;
import java.util.logging.*;

public class JULTest {

     public static Logger log = Logger.getLogger("TestLog");  //获取日志对象

    public static void main(String[] args) throws IOException {

        log.setLevel(Level.ALL);//设置logger的日志级别为全部,默认输出所有级别日志信息

        log.setUseParentHandlers(false); //禁用日志原本处理类

        FileHandler fileHandler = new FileHandler("日志路径/testJUL.log");
        fileHandler.setLevel(Level.ALL); //记录级别
        log.addHandler(fileHandler); //添加Handler

        log.info("info");    //信息日志
        log.warning("warning"); //警告日志
        log.log(Level.SEVERE,"server"); //严重日志
        log.fine("fine");
    }
}

testJUL.log的内容如下

<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<!DOCTYPE log SYSTEM "logger.dtd">
<log>
<record>
  <date>2019-01-15T00:22:58</date>
  <millis>1547482978598</millis>
  <sequence>0</sequence>
  <logger>TestLog</logger>
  <level>INFO</level>
  <class>JULTest</class>
  <method>main</method>
  <thread>1</thread>
  <message>info</message>
</record>
<record>
  <date>2019-01-15T00:22:58</date>
  <millis>1547482978622</millis>
  <sequence>1</sequence>
  <logger>TestLog</logger>
  <level>WARNING</level>
  <class>JULTest</class>
  <method>main</method>
  <thread>1</thread>
  <message>warning</message>
</record>
<record>
  <date>2019-01-15T00:22:58</date>
  <millis>1547482978622</millis>
  <sequence>2</sequence>
  <logger>TestLog</logger>
  <level>SEVERE</level>
  <class>JULTest</class>
  <method>main</method>
  <thread>1</thread>
  <message>server</message>
</record>
<record>
  <date>2019-01-15T00:22:58</date>
  <millis>1547482978622</millis>
  <sequence>3</sequence>
  <logger>TestLog</logger>
  <level>FINER</level>
  <class>JULTest</class>
  <method>main</method>
  <thread>1</thread>
  <message>fine</message>
</record>
</log>

3、自定义Handler

有时候我们在日志中做些自定义的操作,此时我们需要编写自定义Handler并加入logger的Handlers中

//继承Handler类编写自定义Handler
public class MyHandler extends Handler {

    private LogRecord record;

    //
    @Override
    public void publish(LogRecord record) {
        this.record = record;
        ........(自定义操作)
    }

    @Override
    public void flush() {
        System.out.println("logger:"+this.record.getLoggerName()+"flush");
    }

    @Override
    public void close() throws SecurityException {
        System.out.println("logger:"+this.record.getLoggerName()+"close");
    }
}

public class JULTest {

     public static Logger log = Logger.getLogger("TestLog");  //获取日志对象

    public static void main(String[] args) throws IOException {

        log.setLevel(Level.ALL);//设置logger的日志级别为全部,默认输出所有级别日志信息

        log.setUseParentHandlers(false); //禁用日志原本处理类

        MyHandler myHandler = new MyHandler();  //创建自定义日志处理类实体
        log.addHandler(myHandler); //添加日志处理实体类

        log.info("info");    //信息日志
        log.warning("warning"); //警告日志
        log.log(Level.SEVERE,"server"); //严重日志
        log.fine("fine");
    }
}

三、Formatter

可以看到上述我们的日志文件中默认输出是xml格式,很明显是不利于查看的。此时我们需要去更改日志输出样式,同样的可以使用代码实现:

//继承Formatter自定义日志输出形式
public class MyFormate extends Formatter {

    @Override
    public String format(LogRecord record) {
        return new Date()+"-["+record.getSourceClassName()+"."+record.getSourceMethodName()+"]"+record.getLevel()+":"+record.getMessage()+"\n";
    }
}

public class JULTest {

     public static Logger log = Logger.getLogger("TestLog");  //获取日志对象

    public static void main(String[] args) throws IOException {

        log.setLevel(Level.ALL);//设置logger的日志级别为全部,默认输出所有级别日志信息

        log.setUseParentHandlers(false); //禁用日志原本处理类

        FileHandler fileHandler = new FileHandler("日志路径/testJUL.log");
        fileHandler.setLevel(Level.ALL); //记录级别

       fileHandler.setFormatter(new MyFormate()); //设置自定义样式

        log.addHandler(fileHandler); //添加Handler

        log.info("info");    //信息日志
        log.warning("warning"); //警告日志
        log.log(Level.SEVERE,"server"); //严重日志
        log.fine("fine");
    }
}

此时testJUL.log的输出样式更直观简洁了:

Tue Jan 15 00:36:09 CST 2019-[JULTest.main]INFO:info
Tue Jan 15 00:36:09 CST 2019-[JULTest.main]WARNING:warning
Tue Jan 15 00:36:09 CST 2019-[JULTest.main]SEVERE:server
Tue Jan 15 00:36:09 CST 2019-[JULTest.main]FINER:fine

[图片上传中...(image-1622e2-1547516929311-0)]

总结

1、JUL通过对级别的整数值比对决定是否输出日志,对外则是使用通俗易懂的Level名,这样的设计使得系统设计清晰简便,值得学习。

   2、JUL的logging.propertise配置了基本的日志控制属性,可以通过更改该配置文件控制系统日志的输出

   3、JUL的Handler使得JUL的日志控制十分的灵活,但是注意logger可以具备多个Handler,需要考虑Handler之间的干扰和冗余可能

  4、JUL的Formatter使JUL的日志输出更加优雅美观,在学习的过程中,我们可以更加多的考虑如何精准输出日志所需要的信息,比如类名,方法名等

备注

多参考前辈,源码未深入理解,还望多多探讨,文章结构不大严谨,后续继续完善,谢谢

部分内容参考文章:qingkangxu-JDK Logging深入分析

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351

推荐阅读更多精彩内容