how tomcat works 七 日志记录器

大家可以松一口气了,这个组件比较简单,这一节和前面几节想比,也简单的多。

Logger接口

Tomcat中的日志记录器都必须实现org.apache.catalina.Logger接口。

package org.apache.catalina;

import java.beans.PropertyChangeListener;

public interface Logger {

    public static final int FATAL = Integer.MIN_VALUE;
    public static final int ERROR = 1;
    public static final int WARNING = 2;
    public static final int INFORMATION = 3;
    public static final int DEBUG = 4;

    public Container getContainer();
    public void setContainer(Container container);
    public String getInfo();
    public int getVerbosity();
    public void setVerbosity(int verbosity);
    public void addPropertyChangeListener(PropertyChangeListener listener);

    public void log(String message);
    public void log(Exception exception, String msg);
    public void log(String message, Throwable throwable);
    public void log(String message, int verbosity);
    public void log(String message, Throwable throwable, int verbosity);
    public void removePropertyChangeListener(PropertyChangeListener listener);
}

Logger定义了五种日志记录基本,当我们在调用log(String message, int verbosity)方法记录日志的时候,只有传进来的verbosity小于等于系统的默认值才会记录。话说大家知道Integer.MIN_VALUE等于多少吗?大概负的21亿左右吧。setVerbosity与getVerbosity是干什么的还需要我说么?

Tomcat的日志记录器

LoggerBase类(抽象类)

它实现了Logger接口中除log(String message)方法外的其余方法;

在类中默认了日志记录级别为

protected int verbosity = ERROR;

但是我们可以通过setVerbosity(int verbosity)来重新设置记录级别;

看看定义的接受日志级别的两个log方法

public void log(String message, int verbosity) {
    if (this.verbosity >= verbosity)
        log(message);
}
public void log(String message, Throwable throwable, int verbosity) {
    if (this.verbosity >= verbosity)
        log(message, throwable);
}

SystemOutLogger类

看名字就知道,信息最后输出到控制台。

SysteErrLogger类

看名字就知道,它是错误输出,信息仍在控制台,不过是红色的。

FileLogger类

看名字就知道,它是吧信息输出的文件里。

不过具体的来说,里面还是有点说头的,慢慢来。

首先FileLogger还实现了Lifecycle接口,因此它可以像其他组件一样被父组件所启动。

在本节中FileLogger的start与stop方法其实只是改变了布尔值started而已,没有做其他的,另一方面,FileLogger的start与stop方法也并没有被调用!

现在我们就看看怎么给文件里写信息。

public void log(String msg) {

        // Construct the timestamp we will use, if requested
        Timestamp ts = new Timestamp(System.currentTimeMillis());
        String tsString = ts.toString().substring(0, 19);
        String tsDate = tsString.substring(0, 10);
        System.out.println("tsString   "+tsString); //tsString   2014-10-20 15:25:27
        System.out.println("ts   "+ts);            //ts   2014-10-20 15:25:27.406
        System.out.println("tsDate   "+tsDate);        //tsDate   2014-10-20

        // If the date has changed, switch log files
        if (!date.equals(tsDate)) {
            synchronized (this) {
                if (!date.equals(tsDate)) {
                    close();
                    date = tsDate;
                    open();
                }
            }
        }

        // Log this message, timestamped if necessary
        if (writer != null) {
            if (timestamp) {
                writer.println(tsString + " " + msg);
            } else {
                writer.println(msg);
            }
        }
    }

File默认是每一天一个新的文件,文件内格式就是日期+具体信息(如果timestamp为true的话);

那close(),open()方法和writer属性又是什么呢?

   private void open() {

        // Create the directory if necessary
        File dir = new File(directory);
        if (!dir.isAbsolute())
            dir = new File(System.getProperty("catalina.base"), directory);  //至于catalina.base是什么 一会再说
        dir.mkdirs();

        // Open the current log file
        try {
            String pathname = dir.getAbsolutePath() + File.separator +
                prefix + date + suffix;
            writer = new PrintWriter(new FileWriter(pathname, true), true);
        } catch (IOException e) {
            writer = null;
        }

    }

先检查给定的目录是否存在,如果不存在新建一个;接着根据前缀,时间,后缀设定文件名;根据文件名填充writer对象。

    private void close() {

        if (writer == null)
            return;
        writer.flush();
        writer.close();
        writer = null;
        date = "";
    }

FileLogger中的date代表的就是当前的时间,当然关闭后为空。

应用程序

Bootstrap类

...
    System.setProperty("catalina.base", System.getProperty("user.dir"));
    FileLogger logger = new FileLogger();
    logger.setPrefix("FileLog_");
    logger.setSuffix(".txt");
    logger.setTimestamp(true);
    logger.setDirectory("webroot");
    context.setLogger(logger);
....

SimpleContext类

public synchronized void start() throws LifecycleException {
    log("starting Context");
    ........
    lifecycle.fireLifecycleEvent(AFTER_START_EVENT, null);
    log("Context started");
  }
private void log(String message) {
    Logger logger = this.getLogger();
    if (logger!=null)
      logger.log(message);
  }

运行后在项目的webroot目录下就可以看到FileLog_2014-10-20.txt文件;

每次运行,在文件中都会多出如下的内容

2014-10-20 14:34:21 HttpConnector Opening server socket on all host IP addresses

2014-10-20 14:34:22 HttpConnector[8080] Starting background thread

2014-10-20 14:34:22 HttpProcessor[8080][0] Starting background thread

2014-10-20 14:34:22 HttpProcessor[8080][1] Starting background thread

2014-10-20 14:34:22 HttpProcessor[8080][2] Starting background thread

2014-10-20 14:34:22 HttpProcessor[8080][3] Starting background thread

2014-10-20 14:34:22 HttpProcessor[8080][4] Starting background thread

2014-10-20 14:34:22 starting Context

2014-10-20 14:34:22 Context started

时间: 2024-08-04 03:24:51

how tomcat works 七 日志记录器的相关文章

how tomcat works读书笔记 七 日志记录器

大家能够松一口气了,这个组件比較简单,这一节和前面几节想比,也简单的多. Logger接口 Tomcat中的日志记录器都必须实现org.apache.catalina.Logger接口. package org.apache.catalina; import java.beans.PropertyChangeListener; public interface Logger { public static final int FATAL = Integer.MIN_VALUE; public s

How Tomcat works — 七、tomcat发布webapp

目录 什么叫发布 webapp发布方式 reload 总结 什么叫发布 发布就是让tomcat知道我们的程序在哪里,并根据我们的配置创建Context,进行初始化.启动,如下: 程序所在的位置 创建Context,添加到Host 初始化(创建解析webxml的digester) 启动(初始化filter.listener.servlet) webapp发布方式 在tomcat 中发布webapp的方式不同会导致app启动的先后顺序不一样(这里按照启动顺序或者时机不同进行划分): 在xml中配置,

tomcat源码阅读之日志记录器(Logger)

UML图: 1.Logger接口中定义了日志的级别:FATAL.ERROR.WARNING.INFORMATION.DEBUG,通过接口函数getVerbosity获取日志级别,setVerbosity设置日志级别: 2.接口中提供一组重载的log方法来记录日志,在LoggerBase实现类中除了log(String msg)方法没有实现外,其他重载的log方法均实现为调用这个方法来实现日志的记录: 3.在LoggerBase类中,默认的日志级别是ERROR:带日志级别参数的log方法在记录日志

how tomcat works 总结 三

第七章 日志记录器 第 7 章包括日志,该组件是用来记录错误信息和其他信息的. 这一章比较简单,类图如下: 根据名字我想大家都能猜出来三个实现类都是做什么的,一个按常规输出到控制台,一个按错误模式输出到控制点,一个输出到文件. Logger类中有一个参数,verbosity,用来表示日志等级,默认为ERROR. public static final int FATAL = Integer.MIN_VALUE; 在输出日志的时候,只有给定的message的verbosity小于默认的才会输出.

How Tomcat Works 2

上一节(How Tomcat Works 1 编写一个简单静态web服务器)编写了一个简单的web服务器,只能处理静态的资源,本节将继续向前迈出一个小步,创建两个不同的servlet容器,能够利用servlet简单的处理动态内容.注意每节的代码都是基于上一节的继续丰富,因此有必要从第一节开始看起. 在编写代码之前,需要先大体了解一下Servlet是什么,方便后面的理解,下面就是一个最简单的Servlet什么也没做: package prymont; import java.io.IOExcept

通用的日志记录器(PHP)

一个通用的日志记录器 1 <?php 2 class Log4p{ 3 4 const LOG_PATH = "/data/logs/mobact2rd/drawcard"; 5 6 static function writeLog($str, $filename='debug', $fileCutByDate = true, $dirCutByDate = true) { 7 if (empty($str)) return; 8 $dir = ''; 9 10 if ( $di

通用的日志记录器(java)

线程安全的java日志记录器 1 import java.io.BufferedWriter; 2 import java.io.File; 3 import java.io.FileWriter; 4 import java.io.IOException; 5 import java.text.SimpleDateFormat; 6 import java.util.Date; 7 import java.util.concurrent.atomic.AtomicReference; 8 9

how tomcat works 读书笔记四 tomcat的默认连接器

其实在第三章,就已经有了连接器的样子了,不过那只是一个学习工具,在这一章我们会开始分析tomcat4里面的默认连接器. 连接器 Tomcat连接器必须满足以下几个要求 1 实现org.apache.cataline.Connector接口 2 负责创建实现了org.apache.cataline.Request接口的request对象 3 负责创建实现了org.apache.cataline.Response接口的response对象 这里默认的连接器的原理很简单,就是等待http请求,创建re

阿里云X-Forwarded-For 发现tomcat记录的日志全部来自于SLB转发的IP地址,不能获取到请求的真实IP。

1.背景:阿里云的SLB的负载均衡,在tomcat中获取不到真实IP,而是阿里的内网IP,SLB中俩台或者3台本身是局域网,这里是SLB原理,可以看看,没怎么看懂,呵呵,要细细读下. 2.需要开启tomcat的X-Forwarded-For,在tomcat/conf/server.xml中有一个如下的AccessLogValve 日志纪录功能,当配置中的pattern=common时,对应的日志是如下,无论正常请求和非法请求都会记录. <Valve className="org.apach