log4j的数据库Appender的实现代码

1.JDBCAppender.java

package com.benqguru.palau.log.jdbc.test;

import java.sql.*;
import java.util.*;

import org.apache.log4j.*;
import org.apache.log4j.spi.*;

/**
 这个JDBCAppender用来把消息写进数据库.
 <p><b>JDBCAppender在运行时可选项配置的,通过在下面两者之间选择:</b></p>
 <dir>
 <p><b>1. 用配置文件</b></p>
 <p>在文件中定义选项,并且在你的代码中调用<code>PropertyConfigurator.configure(filename)</code></p>
 <p><b>2. 用JDBCAppender的方法来完成</b></p>
 <p>调用 <code>JDBCAppender::setOption(JDBCAppender.xxx_OPTION, String value)</code> 来做相相似的事情,在没用配置文件的情况下</p>
 </dir>

 <p>在JDBCAppender中,所有有用变量被定义为静态字符串常量,并命名为xxx_OPTION.</p>

 <p><b>下面是所有的有用选项的描述:</b></p>
 <dir>
 <p><b>1. 连接到数据库的数据库可选项(Database-options)</b></p>
 <p>- <b>URL_OPTION</b>  : 一个形如jdbc:subprotocol:subname数据库url</p>
 <p>- <b>USERNAME_OPTION</b> : 数据库用户</p>
 <p>- <b>PASSWORD_OPTION</b> : 用户密码</p>

 <p><b>2. 指定你自己的JDBCConnectionHandler的连接器选项(Connector-option)</b></p>
 <p>- <b>CONNECTOR_OPTION</b> : 一个实现了JDBCConnectionHandler接口的类</p>
 <p>这个接口可以用来获得自定义连接.</p>
 <p>假如其余的数据库选项是给定的,这些选项将用作为参数传给JDBCConnectionHandler接口.</p>
 <p>另外假如没有给定数据库选项,JDBCConnectionHandler接口将不带这些参数被调用</p>
 <p>另外假如这个可选项没有定义,JDBCAppender需要数据库选项来打开一个连接</p>

 <p><b>3. SQL选项指定一个静态的sql语句,在每次发生消息事件时,将执行这个语句</b></p>
 <p>- <b>SQL_OPTION</b>   : 一个用来写入数据到数据库的sql语句</p>
 <p>这sql语句的某个地方用<b>@MSG@</b>这个变量,这个变量必须要用消息内容来动态的替换</p>
 <p>假如你给定整个选项,表选项和列选项将被忽略!</p>

 <p><b>4. 表选项用来指定数据库中包含的一个表</b></p>
 <p>- <b>TABLE_OPTION</b>  : 保存日志信息的表</p>

 <p><b>5. 列选项用来描述表中重要的列(非空列必须描述!)</b></p>
 <p>- <b>COLUMNS_OPTION</b>  : 列描述的一个格式化列表</p>
 <p>每个列描述包涵如下内容</p>
 <dir>
  <p>- 列的名字 <b><i>(name)</i></b> (必须)</p>
  <p>- LogType类中的一个静态常量日志类型<b><i>(logtype)</i></b> (必须)</p>
  <p>- 依赖与LogType类的一个值<b><i>(value)</i></b>(可选/必须, 依赖日志类型(logtype)</p>
 </dir>
 <p>这是<b>{@link LogType}</b>类中的有用日志类型(logtype)的描述,并且怎样处理处理值<b><i>(value)</i></b>:</p>
 <dir>
  <p>o <b>MSG</b>  = 一个可以忽略的值,列将获得消息. (有一个列需要这种类型!)</p>
  <p>o <b>STATIC</b>  = 这个值可以填充每一个日志消息的列. (保证这种值的类型可以被转换为列的sql类型!)</p>
  <p>o <b>ID</b>  = 值必须是一个实现JDBCIDHandler接口的类名.</p>
  <p>o <b>TIMESTAMP</b>         = 一个可以忽略的值,这个列将用每个记录日志消息的确切时间戳填充.</p>
  <p>o <b>EMPTY</b>  = 一个可以忽略的值,当写数据到数据库的时候,这个列将被忽略 (保证用数据库触发器填充非空类!)</p>
 </dir>
 <p>假如有根多的列需要描述, 列必须要Tab分界符隔开(unicode0008) !</p>
 <p>列描述的参数必须用'~'隔开 !</p>
 <p><i>(实例:  name1~logtype1~value1   name2~logtype2~value2...)</i></p>

 <p><b>6. 布局器选项用来定义消息的布局 (可选)</b></p>
 <p>- <b>_</b> : 布局器不是用形如xxx_OPTION来设置的</p>
 <p>参考下面的配置文件和代码示例...</p>
 <p>默认的布局器是类 {@link org.apache.log4j.PatternLayout}, 这个布局器用仅仅代表消息的%m模式.</p>

 <p><b>7. 缓冲器选项用来定义一个消息事件缓冲区(可选)</b></p>
 <p>- <b>BUFFER_OPTION</b>  : 定义多少条消息将被缓存,直到被更新到数据库.</p>
 <p>默认的缓冲区大小是1, 当消息事件发生时,将进行更新操作.</p>

 <p><b>8. 提交选项用来定义一个自动提交</b></p>
 <p>- <b>COMMIT_OPTION</b>  : 定义更新消息是(Y)否(N)自动提交到数据库.</p>
 <p>默认的时commit=N.</p>
 </dir>

 <p><b>下面选项的顺序时很重要的:</b></p>
 <dir>
 <p><b>1. 连接器选项(Connector-option) OR/AND 数据库选项(Database-options)</b></p>
 <p>数据库连接时必须的!</p>
 <p><b>2. (表选项(Table-option) AND 列选项(Columns-option)) OR SQL选项(SQL-option)</b></p>
 <p>上面的任意一项是必须的! 写在哪或者写什么...;-)</p>
 <p><b>3. 其他选项可以在任何时候设置...</b></p>
 <p>其他选项时可选的,并且都有一个可以自定义的默认值.</p>
 </dir>

 <p><b>这是一个可以作为PropertyConfigurator参数的配置文件示例</b> : <A HREF="log4j.properties"> log4j.properties</A></p>

 <p><b>这是一个用配置文件配置JDBCAppender类的代码示例</b> : <A HREF="Log4JTest.java"> Log4JTest.java</A>.</p>

 <p><b>这是另一个没用配置文件来配置JDBCAppender类的代码示例</b> : <A HREF="Log4JTest2.java"> Log4JTest2.java</A></p>

* <p>Title: log4j1.2.8源码解析</p>
* <p>Description: </p>
* <p>Copyright: Copyright (c) 2004</p>
* <p>Company: 明基逐鹿</p>
* @author 王建
* @version 1.0
*/
public class JDBCAppender
    extends AppenderSkeleton {
  /**
   数据库选项,设置设置数据库url的形式为jdbc:subprotocol:subname
   */
  public static final String URL_OPTION = "url";

  /**
   数据库选项,设置可以连接这个数据库的用户
   */
  public static final String USERNAME_OPTION = "username";

  /**
   数据库选项,设置用户密码
   */
  public static final String PASSWORD_OPTION = "password";

  /**
   表选项,指定数据库中一个表
   */
  public static final String TABLE_OPTION = "table";

  /**
   连接器选项,指定你自己的JDBCConnectionHandler
   */
  public static final String CONNECTOR_OPTION = "connector";

  /**
   列选项,描述表中重要的列
   */
  public static final String COLUMNS_OPTION = "columns";

  /**
   sql选项,指定一个在发生消息事件时执行的静态sql语句
   */
  public static final String SQL_OPTION = "sql";

  /**
   缓冲器选项,定义消息事件缓冲区的大小
   */
  public static final String BUFFER_OPTION = "buffer";

  /**
   提交选项,定义一个自动提交
   */
  public static final String COMMIT_OPTION = "commit";

  //保存可以被setOption()方法设置的可选值的变量
  private String url = null;
  private String username = null;
  private String password = null;
  private String table = null;
  private String connection_class = null;
  private String sql = null;

//   若数据库不支持事务,择要设定为false,如Mysql
  private boolean docommit = false;
  private int buffer_size = 1;
  private JDBCConnectionHandler connectionHandler = null;

//   这个缓冲器保存了消息事件.当缓冲器达到一定大小时,缓冲器将被刷新,并且信息将被更新到数据库.
  private ArrayList buffer = new ArrayList();

//   数据库连接
  private Connection con = null;

//   这个类封装了记录日志进表中所必须的逻辑
  private JDBCLogger jlogger = new JDBCLogger();

//   标识:这是一个表明是否已经建立了数据库连接的标识
  private boolean connected = false;

//   一个表明配置状态的标识
  private boolean configured = false;

//   一个表明所有东西都已准备好,可以使用append()的标识
  private boolean ready = false;
//   假如程序结束,关闭数据库并且刷新缓冲器
  public void finalize() {
    close();
    super.finalize();
  }

  /**
   * 内部方法.返回一个包涵可以被方法setOption()设置的有用选项(option)的字符串数组.
   * @return String[]
   */
  public String[] getOptionStrings() {
    // The sequence of options in this string is important, because setOption() is called this way ...
    return new String[] {
        CONNECTOR_OPTION, URL_OPTION, USERNAME_OPTION, PASSWORD_OPTION,
        SQL_OPTION, TABLE_OPTION, COLUMNS_OPTION, BUFFER_OPTION, COMMIT_OPTION};
  }

  /**
   * 设置所有必须的选项
   * @param _option String
   * @param _value String
   */
  public void setOption(String _option, String _value) {
    _option = _option.trim();
    _value = _value.trim();

    if (_option == null || _value == null)return;
    if (_option.length() == 0 || _value.length() == 0)return;

    _value = _value.trim();

    if (_option.equals(CONNECTOR_OPTION)) {//连接器
      if (!connected) connection_class = _value;
    }
    else if (_option.equals(URL_OPTION)) {//URL
      if (!connected) url = _value;
    }
    else if (_option.equals(USERNAME_OPTION)) {//用户名
      if (!connected) username = _value;
    }
    else if (_option.equals(PASSWORD_OPTION)) {//密码
      if (!connected) password = _value;
    }
    else if (_option.equals(SQL_OPTION)) {//SQL语句
      sql = _value;
    }
    else if (_option.equals(TABLE_OPTION)) {//表
      if (sql != null)return;//若sql语句不是null,则返回
      table = _value;
    }
    else if (_option.equals(COLUMNS_OPTION)) {//列
      if (sql != null)return;

      String name = null;
      int logtype = -1;
      String value = null;
      String column = null;
      String arg = null;
      int num_args = 0;
      int num_columns = 0;
      StringTokenizer st_col;
      StringTokenizer st_arg;

      //Columns are TAB-separated
      st_col = new StringTokenizer(_value, " ");

      num_columns = st_col.countTokens();

      if (num_columns < 1) {
        errorHandler.error(
            "JDBCAppender::setOption(), Invalid COLUMN_OPTION value : " +
            _value + " !");
        return;
      }

      for (int i = 1; i <= num_columns; i++) {
        column = st_col.nextToken();

        //Arguments are ~-separated
        st_arg = new StringTokenizer(column, "~");

        num_args = st_arg.countTokens();

        if (num_args < 2) {
          errorHandler.error(
              "JDBCAppender::setOption(), Invalid COLUMN_OPTION value : " +
              _value + " !");
          return;
        }

        for (int j = 1; j <= num_args; j++) {
          arg = st_arg.nextToken();

          if (j == 1) name = arg;
          else if (j == 2) {
            try {
              logtype = Integer.parseInt(arg);
            }
            catch (Exception e) {
              logtype = LogType.parseLogType(arg);
            }

            if (!LogType.isLogType(logtype)) {
              errorHandler.error(
                  "JDBCAppender::setOption(), Invalid COLUMN_OPTION LogType : " +
                  arg + " !");
              return;
            }
          }
          else if (j == 3) value = arg;
        }

        if (!setLogType(name, logtype, value))return;
      }
    }
    else if (_option.equals(BUFFER_OPTION)) {//缓冲器
      try {
        buffer_size = Integer.parseInt(_value);
      }
      catch (Exception e) {
        errorHandler.error(
            "JDBCAppender::setOption(), Invalid BUFFER_OPTION value : " +
            _value + " !");
        return;
      }
    }
    else if (_option.equals(COMMIT_OPTION)) {//提交
      docommit = _value.equals("Y");
    }

    if (_option.equals(SQL_OPTION) || _option.equals(TABLE_OPTION)) {
      if (!configured)
        configure();
    }
  }

  /**
   * 内部方法,返回true,你可以定义你自己的layout
   * @return boolean
   */
  public boolean requiresLayout() {
    return true;
  }

  /**
   * 内部方法,关闭数据库连接并且冲刷(flush)缓冲器
   */
  public void close() {
    flush_buffer();
    if (connection_class == null) {
      try {
        con.close();
      }
      catch (Exception e) {
        errorHandler.error("JDBCAppender::close(), " + e);
      }
    }
    this.closed = true;
  }

  /**
   * 对日志表的所有列都必须调用这个方法
   * @param _name String
   * @param _logtype int
   * @param _value Object
   * @return boolean
   */
  public boolean setLogType(String _name, int _logtype, Object _value) {
    if (sql != null)return true;

    if (!configured) {//为什么?
      if (!configure())return false;
    }

    try {
      jlogger.setLogType(_name, _logtype, _value);
    }
    catch (Exception e) {
      errorHandler.error("JDBCAppender::setLogType(), " + e);
      return false;
    }

    return true;
  }

  /**
   * 内部方法.添加消息到数据库表
   * @param event LoggingEvent
   */
  public void append(LoggingEvent event) {
    if (!ready) {
      if (!ready()) {
        errorHandler.error("JDBCAppender::append(), Not ready to append !");
        return;
      }
    }

    buffer.add(event);

    if (buffer.size() >= buffer_size) flush_buffer();
  }

  /**
   * 内部方法.刷新缓冲器
   */
  public void flush_buffer() {
    try {
      int size = buffer.size();

      if (size < 1)return;

      for (int i = 0; i < size; i++) {
        LoggingEvent event = (LoggingEvent) buffer.get(i);

        //Insert message into database
        jlogger.append(layout.format(event));
      }

      buffer.clear();

      if (docommit) con.commit();
    }
    catch (Exception e) {
      errorHandler.error("JDBCAppender::flush_buffer(), " + e + " : " +
                         jlogger.getErrorMsg());
      try {
        con.rollback();
      }
      catch (Exception ex) {}
      return;
    }
  }

  /**
   * 内部方法.当JDBCAppender准备好可以添加消息到数据库,返回true;否则返回false
   * @return boolean
   */
  public boolean ready() {
    if (ready)return true;

    if (!configured)return false;

    ready = jlogger.ready();

    if (!ready) {
      errorHandler.error(jlogger.getErrorMsg());
    }

    return ready;
  }

  /**
   * 内部方法,连接数据库
   * @throws Exception
   */
  protected void connect() throws Exception {
    if (connected)return;

    try {
      if (connection_class == null) {
        if (url == null)throw new Exception(
            "JDBCAppender::connect(), No URL defined.");

        if (username == null)throw new Exception(
            "JDBCAppender::connect(), No USERNAME defined.");

        if (password == null)throw new Exception(
            "JDBCAppender::connect(), No PASSWORD defined.");

        connectionHandler = new DefaultConnectionHandler();
      }
      else {
        connectionHandler = (JDBCConnectionHandler) (Class.forName(
            connection_class).newInstance());
      }

      if (url != null && username != null && password != null) {
        con = connectionHandler.getConnection(url, username, password);
      }
      else {
        con = connectionHandler.getConnection();
      }

      if (con.isClosed()) {
        throw new Exception("JDBCAppender::connect(), JDBCConnectionHandler returns no connected Connection !");
      }
    }
    catch (Exception e) {
      throw new Exception("JDBCAppender::connect(), " + e);
    }
    //设置连接标识为true
    connected = true;
  }

  /**
   * 内部方法.检查是否配置,以便进行添加操作...
   * @return boolean
   */
  protected boolean configure() {
    if (configured)return true;

    if (!connected) {
      if ( (connection_class == null) &&
          (url == null || username == null || password == null)) {
        errorHandler.error(
            "JDBCAppender::configure(), Missing database-options or connector-option !");
        return false;
      }

      try {
        connect();
      }
      catch (Exception e) {
        connection_class = null;
        url = null;
        errorHandler.error("JDBCAppender::configure(), " + e);
        return false;
      }
    }

    if (sql == null && table == null) {
      errorHandler.error(
          "JDBCAppender::configure(), No SQL_OPTION or TABLE_OPTION given !");
      return false;
    }

    if (!jlogger.isConfigured()) {
      try {
        jlogger.setConnection(con);

        if (sql == null) {
          jlogger.configureTable(table);
        }
        else jlogger.configureSQL(sql);
      }
      catch (Exception e) {
        errorHandler.error("JDBCAppender::configure(), " + e);
        return false;
      }
    }

    //Default Message-Layout
    if (layout == null) {
      layout = new PatternLayout("%m");
    }

    //标明已经配置好了!
    configured = true;

    return true;
  }
  public String getUrl() {
    return url;
  }
  public void setUrl(String url) {
    this.url = url;
  }
  public String getUsername() {
    return username;
  }
  public void setUsername(String username) {
    this.username = username;
  }
  public String getPassword() {
    return password;
  }
  public void setPassword(String password) {
    this.password = password;
  }
  public String getConnection_class() {
    return connection_class;
  }
  public void setConnection_class(String connection_class) {
    this.connection_class = connection_class;
  }
  public String getSql() {
    return sql;
  }
  public void setSql(String sql) {
    this.sql = sql;
  }
  public String getTable() {
    return table;
  }
  public void setTable(String table) {
    this.table = table;
  }
  public int getBuffer_size() {
    return buffer_size;
  }
  public void setBuffer_size(int buffer_size) {
    this.buffer_size = buffer_size;
  }
  public String getDocommit() {
    if (docommit == true) {
      return "Y";
    }
    else {
      return "N";
    }
  }

  public void setDocommit(String commit) {
    if (commit.equals("N")) {
      this.docommit = false;
    }
    else {
      this.docommit = true;
    }
  }
}

/**
 * 这是一个JDBCAppender使用的默认JDBCConnectionHandler
 * <p>Title: log4j1.2.8源码解析</p>
 * <p>Description: </p>
 * <p>Copyright: Copyright (c) 2004</p>
 * <p>Company: 明基逐鹿</p>
 * @author 王建
 * @version 1.0
 */
class DefaultConnectionHandler
    implements JDBCConnectionHandler {
  Connection con = null;

  public Connection getConnection() {
    return con;
  }

  public Connection getConnection(String _url, String _username,
                                  String _password) {
    try {
      if (con != null && !con.isClosed()) con.close();
      con = DriverManager.getConnection(_url, _username, _password);
      con.setAutoCommit(true);
    }
    catch (Exception e) {
      e.printStackTrace();
    }

    return con;
  }
}

2.JDBCConnectionHandler.java

package com.benqguru.palau.log.jdbc.test;

import java.sql.*;

/**
 * 必须实现这个接口来处理数据库连接,在JDBCLogger类中会用到这个接口
 * <p>Title: log4j1.2.8源码解析</p>
 * <p>Description: </p>
 * <p>Copyright: Copyright (c) 2004</p>
 * <p>Company: 明基逐鹿</p>
 * @author 王建
 * @version 1.0
 */
public interface JDBCConnectionHandler {

  /**
   * 获得一个连接
   * @return Connection
   */
  Connection getConnection();

  /**
   * 获得一个自定义的连接
   * @param _url String
   * @param _username String
   * @param _password String
   * @return Connection
   */
  Connection getConnection(String _url, String _username, String _password);
}

3.JDBCIDHandler.java

package com.benqguru.palau.log.jdbc.test;

/**
 * 必须实现这个接口来提供带有独一无二ID的ID-column,在JDBCLogger类中会用到这个接口
 * <p>Title: log4j1.2.8源码解析</p>
 * <p>Description: </p>
 * <p>Copyright: Copyright (c) 2004</p>
 * <p>Company: 明基逐鹿</p>
 * @author 王建
 * @version 1.0
 */
public interface JDBCIDHandler {

  /**
   * 获得独一无二的ID
   * @return Object
   */
  Object getID();
}

4.JDBCLogger.java

package com.benqguru.palau.log.jdbc.test;

import java.sql.*;
import java.util.*;

/**
 * 这个类封装了必须要记录进表中的逻辑,JDBCAppender使用这个类
 * <p>Title: log4j1.2.8源码解析</p>
 * <p>Description: </p>
 * <p>Copyright: Copyright (c) 2004</p>
 * <p>Company: 明基逐鹿</p>
 * @author 王建
 * @version 1.0
 */
public class JDBCLogger {

//  日志表的所有列
  private ArrayList logcols = null;

//  通过日志记录提供的列
  private String column_list = null;

//  所有列的数目
  private int num = 0;

//  表示configure()方法执行成功
  private boolean isconfigured = false;

//表示准备好用append()来记录日志
  private boolean ready = false;

//  当ready()方法失败的时候,将用错误字符串填充这个消息, 可以用getMsg()方法返回这个值
  private String errormsg = "";

  private Connection con = null;
  private Statement stmt = null;
  private ResultSet rs = null;
  private String table = null;

  //静态SQL语句记录日志的变量
  private String sql = null;
  private String new_sql = null;
  private String new_sql_part1 = null;
  private String new_sql_part2 = null;
  private static final String msg_wildcard = "@MSG@";
  private int msg_wildcard_pos = 0;

  /**
   * 把一条消息写进数据库表中,假如发生数据库错误,将抛出异常!
   * @param _msg String
   * @throws Exception
   */
  public void append(String _msg) throws Exception {
    if (!ready)
      if (!ready())
        throw new Exception(
            "JDBCLogger::append(), Not ready to append !");

    if (sql != null) {
      appendSQL(_msg);
      return;
    }

    LogColumn logcol;

    rs.moveToInsertRow();

    for (int i = 0; i < num; i++) {
      logcol = (LogColumn) logcols.get(i);

      if (logcol.logtype == LogType.MSG) {
        rs.updateObject(logcol.name, _msg);
      }
      else if (logcol.logtype == LogType.ID) {
        rs.updateObject(logcol.name, logcol.idhandler.getID());
      }
      else if (logcol.logtype == LogType.STATIC) {
        rs.updateObject(logcol.name, logcol.value);
      }
      else if (logcol.logtype == LogType.TIMESTAMP) {
        rs.updateObject(logcol.name,
                        new Timestamp( (new java.util.Date()).getTime()));
      }
    }

    rs.insertRow();
  }

  /**
   * 用给定的sql语句把一条消息写进数据库中.假如发生数据库错误,将抛出异常!
   * @param _msg String
   * @throws Exception
   */
  public void appendSQL(String _msg) throws Exception {
    if (!ready)if (!ready())throw new Exception(
        "JDBCLogger::appendSQL(), Not ready to append !");

    if (sql == null)throw new Exception(
        "JDBCLogger::appendSQL(), No SQL-Statement configured !");

    if (msg_wildcard_pos > 0) {
      new_sql = new_sql_part1 + _msg + new_sql_part2;
    }
    else new_sql = sql;

    try {
      stmt.executeUpdate(new_sql);
    }
    catch (Exception e) {
      errormsg = new_sql;
      throw e;
    }
  }

  /**
   * 通过读取日志表的结构来配置这个类.假如发生数据库错误,将抛出异常!
   * @param _table String
   * @throws Exception
   */
  public void configureTable(String _table) throws Exception {
    if (isconfigured)return;

    //用表列的META-informations填充日志列
    stmt = con.createStatement(ResultSet.TYPE_SCROLL_INSENSITIVE,
                               ResultSet.CONCUR_UPDATABLE);
    rs = stmt.executeQuery("SELECT * FROM " + _table + " WHERE 1 = 2");

    LogColumn logcol;

    ResultSetMetaData rsmd = rs.getMetaData();

    num = rsmd.getColumnCount();

    logcols = new ArrayList(num);

    for (int i = 1; i <= num; i++) {
      logcol = new LogColumn();
      logcol.name = rsmd.getColumnName(i).toUpperCase();
      logcol.type = rsmd.getColumnTypeName(i);
      logcol.nullable = (rsmd.isNullable(i) == rsmd.columnNullable);
      logcol.isWritable = rsmd.isWritable(i);
      if (!logcol.isWritable) logcol.ignore = true;
      logcols.add(logcol);
    }

    table = _table;

    isconfigured = true;
  }

  /**
   * 通过保存与解析给定的sql语句,来配置这个类.假如发生数据库错误,将抛出异常!
   * @param _sql String
   * @throws Exception
   */
  public void configureSQL(String _sql) throws Exception {
    if (isconfigured)return;

    if (!isConnected())throw new Exception(
        "JDBCLogger::configureSQL(), Not connected to database !");

    if (_sql == null || _sql.trim().equals(""))throw new Exception(
        "JDBCLogger::configureSQL(), Invalid SQL-Statement !");

    sql = _sql.trim();

    stmt = con.createStatement();

    msg_wildcard_pos = sql.indexOf(msg_wildcard);

    if (msg_wildcard_pos > 0) {
      new_sql_part1 = sql.substring(0, msg_wildcard_pos - 1) + "'";
      //between the message...
      new_sql_part2 = "'" +
          sql.substring(msg_wildcard_pos + msg_wildcard.length());
    }

    isconfigured = true;
  }

  /**
   * 设置一个连接.假如连接没打开,将抛出异常!
   * @param _con Connection
   * @throws Exception
   */
  public void setConnection(Connection _con) throws Exception {
    con = _con;

    if (!isConnected())throw new Exception(
        "JDBCLogger::setConnection(), Given connection isnt connected to database !");
  }

  /**
   * 设置一个列日志类型(LogTypes)并且依赖于logtype值.假如给定的参数不正确,将抛出异常!
   * @param _name String
   * @param _logtype int
   * @param _value Object
   * @throws Exception
   */
  public void setLogType(String _name, int _logtype, Object _value) throws
      Exception {
    if (!isconfigured)throw new Exception(
        "JDBCLogger::setLogType(), Not configured !");

    //setLogType() makes only sense for further configuration of configureTable()
    if (sql != null)return;

    _name = _name.toUpperCase();

    if (_name == null || ! (_name.trim().length() > 0))throw new Exception(
        "JDBCLogger::setLogType(), Missing argument name !");
    if (!LogType.isLogType(_logtype))throw new Exception(
        "JDBCLogger::setLogType(), Invalid logtype '" + _logtype + "' !");
    //除了消息类型和空类型,其他类型的值不能为空
    if ( (_logtype != LogType.MSG && _logtype != LogType.EMPTY) && _value == null)throw new
        Exception("JDBCLogger::setLogType(), Missing argument value !");

    LogColumn logcol;
    //设置列的值的来源
    for (int i = 0; i < num; i++) {
      logcol = (LogColumn) logcols.get(i);

      if (logcol.name.equals(_name)) {
        if (!logcol.isWritable)throw new Exception(
            "JDBCLogger::setLogType(), Column " + _name + " is not writeable !");

        //Column gets the message
        if (_logtype == LogType.MSG) {
          logcol.logtype = _logtype;
          return;
        }
        //Column will be provided by JDBCIDHandler::getID()
        else if (_logtype == LogType.ID) {
          logcol.logtype = _logtype;

          try {
            //Try to cast directly Object to JDBCIDHandler
            logcol.idhandler = (JDBCIDHandler) _value;
          }
          catch (Exception e) {
            try {
              //Assuming _value is of class string which contains the classname of a JDBCIDHandler
              logcol.idhandler = (JDBCIDHandler) (Class.forName( (String)
                  _value).newInstance());
            }
            catch (Exception e2) {
              throw new Exception(
                  "JDBCLogger::setLogType(), Cannot cast value of class " +
                  _value.getClass() + " to class JDBCIDHandler !");
            }
          }

          return;
        }

        //Column will be statically defined with Object _value
        else if (_logtype == LogType.STATIC) {
          logcol.logtype = _logtype;
          logcol.value = _value;
          return;
        }

        //Column will be provided with a actually timestamp
        else if (_logtype == LogType.TIMESTAMP) {
          logcol.logtype = _logtype;
          return;
        }

        //Column will be fully ignored during process.
        //If this column is not nullable, the column has to be filled by a database trigger,
        //else a database error occurs !
        //Columns which are not nullable, but should be not filled, must be explicit assigned with LogType.EMPTY,
        //else a value is required !
        else if (_logtype == LogType.EMPTY) {
          logcol.logtype = _logtype;
          logcol.ignore = true;
          return;
        }
      }
    }
  }

  /**
   * 假如这个类的append()方法准备好了,返回true,否则返回false.
   * 若没有准备好,一个原因字符串将被保存在实例变量msg中
   * @return boolean
   */
  public boolean ready() {
    if (ready)return true;

    if (!isconfigured) {
      errormsg = "Not ready to append ! Call configure() first !";
      return false;
    }

    //No need to doing the whole rest...
    if (sql != null) {
      ready = true;
      return true;
    }

    boolean msgcol_defined = false;

    LogColumn logcol;

    for (int i = 0; i < num; i++) {
      logcol = (LogColumn) logcols.get(i);

      if (logcol.ignore || !logcol.isWritable)continue;
      if (!logcol.nullable && logcol.logtype == LogType.EMPTY) {
        errormsg = "Not ready to append ! Column " + logcol.name +
            " is not nullable, and must be specified by setLogType() !";
        return false;
      }
      if (logcol.logtype == LogType.ID && logcol.idhandler == null) {
        errormsg = "Not ready to append ! Column " + logcol.name +
            " is specified as an ID-column, and a JDBCIDHandler has to be set !";
        return false;
      }
      else if (logcol.logtype == LogType.STATIC && logcol.value == null) {
        errormsg = "Not ready to append ! Column " + logcol.name +
            " is specified as a static field, and a value has to be set !";
        return false;
      }
      else if (logcol.logtype == LogType.MSG) msgcol_defined = true;
    }

    if (!msgcol_defined)return false;

    //create the column_list
    for (int i = 0; i < num; i++) {
      logcol = (LogColumn) logcols.get(i);

      if (logcol.ignore || !logcol.isWritable)continue;

      if (logcol.logtype != LogType.EMPTY) {
        if (column_list == null) {
          column_list = logcol.name;
        }
        else column_list += ", " + logcol.name;
      }
    }

    try {
      rs = stmt.executeQuery("SELECT " + column_list + " FROM " + table +
                             " WHERE 1 = 2");
    }
    catch (Exception e) {
      errormsg = "Not ready to append ! Cannot select columns '" + column_list +
          "' of table " + table + " !";
      return false;
    }

    ready = true;

    return true;
  }

  /**
   * 假如这个已经配置好了,返回true,否则返回false
   * @return boolean
   */
  public boolean isConfigured() {
    return isconfigured;
  }

  /**
   * 假如这个连接是打开的,返回true,否则返回false
   * @return boolean
   */
  public boolean isConnected() {
    try {
      return (con != null && !con.isClosed());
    }
    catch (Exception e) {
      return false;
    }
  }

  /**
   * 返回保存在实例变量msg中的内部错误信息
   * @return String
   */
  public String getErrorMsg() {
    String r = new String(errormsg);
    errormsg = null;
    return r;
  }
}

/**
 * 这个类封装了JDBCLogger类所需要的与列相关的数据
 * <p>Title: log4j1.2.8源码解析</p>
 * <p>Description: </p>
 * <p>Copyright: Copyright (c) 2004</p>
 * <p>Company: 明基逐鹿</p>
 * @author 王建
 * @version 1.0
 */
class LogColumn {

  /**
   列名
   */
  String name = null;

  /**
   列类型
   */
  String type = null;

  /**
   非空(not nullability)意味着这个列是必须的
   */
  boolean nullable = false;

  /**
   isWritable意味这个列可以更新,或者列仅仅可读.
   */
  boolean isWritable = false;

  /**
   假如ignore是true,这个列将在编译sql语句的时候忽略
   */
  boolean ignore = false;

  /**
   必须用非空列填充!其他情况是可选的.
   */
  int logtype = LogType.EMPTY;

  /**
   对应于包装器类Long,String等等的通用存贮器
   */
  Object value = null;
  /**
   JDBCIDHandler接口的实例
  */
  JDBCIDHandler idhandler = null;
}

/**
 * 这个类包含所有常量,这些常量对定义一个列日志类型是必须的.
 * <p>Title: log4j1.2.8源码解析</p>
 * <p>Description: </p>
 * <p>Copyright: Copyright (c) 2004</p>
 * <p>Company: 明基逐鹿</p>
 * @author 王建
 * @version 1.0
 */
class LogType {

  /**
   这个类型的列将接受消息
   */
  public static final int MSG = 1;

  /**
   这个类型的列将是记录行的唯一标识符
   */
  public static final int ID = 2;

  /**
   这个类型的列将包含一个静态的一次定义(one-time-defined)的值
   */
  public static final int STATIC = 3;

  /**
   这个类型的列将用一个的时间戳填充,这个时间戳依赖于日志记录的开始时间
   */
  public static final int TIMESTAMP = 4;

  /**
   这个类型的列将不包含值,并且不包含在日志记录的插入语句.这将是一个不需要创建来填充的列,而是在别的地方.
   */
  public static final int EMPTY = 5;

  /**
   * 检测_lt是否是日志类型
   * @param _lt int
   * @return boolean
   */
  public static boolean isLogType(int _lt) {
    if (_lt == MSG || _lt == STATIC || _lt == ID || _lt == TIMESTAMP ||
        _lt == EMPTY)return true;

    return false;
  }

  /**
   * 把字符串转换为日志类型
   * @param _lt String
   * @return int
   */
  public static int parseLogType(String _lt) {
    if (_lt.equals("MSG"))return MSG;
    if (_lt.equals("ID"))return ID;
    if (_lt.equals("STATIC"))return STATIC;
    if (_lt.equals("TIMESTAMP"))return TIMESTAMP;
    if (_lt.equals("EMPTY"))return EMPTY;

    return -1;
  }
}
5.log4j.properties

#这是一个配置文件实例,PropertyConfigurator将使用这个文件 :
#声明一个appender变量名为JDBC
log4j.rootLogger=DEBUG, JDBC

#JDBC是一个JDBCAppender类,这个类可以写消息到数据库
log4j.appender.JDBC=com.benqguru.palau.log.jdbc.test.JDBCAppender

#1.连接数据库的数据库选项
log4j.appender.JDBC.url=jdbc:mysql://localhost:3306/logtest
log4j.appender.JDBC.username=root
log4j.appender.JDBC.password=

#2.指定你自己的JDBCConnectionHandler的连接器选项
log4j.appender.JDBC.connection_class=com.benqguru.palau.log.jdbc.test.MyConnectionHandler

#3.指定一个静态的SQL语句的SQL选项,这个语句将在每次消息事件发生时被执行
log4j.appender.JDBC.sql=INSERT INTO LOGTEST (id, msg, created_on, created_by) VALUES (1, @MSG@, sysdate, 'me')

#4. 指定数据库中一个表的表选项。
log4j.appender.JDBC.table=logtest

#5.描述表的重要列的列选项(非空列是必须被描述的)
log4j.appender.JDBC.columns=id_seq~EMPTY id~ID~MyIDHandler msg~MSG created_on~TIMESTAMP created_by~STATIC~Thomas Fenner ([email protected])

#6.定义消息布局器的布局器选项(可选)
log4j.appender.JDBC.layout=org.apache.log4j.PatternLayout
log4j.appender.JDBC.layout.ConversionPattern=%m

#7.定义消息事件缓冲器的大小的缓冲器选项(可选)
log4j.appender.JDBC.buffer_size=1

#8.定义自动提交的提交选项(可选)
log4j.appender.JDBC.docommit=N

##########下面是英文说明#############
#Date - %d{DATE}[slf5s.DATE]
#Priority - %p[slf5s.PRIORITY]
#NDC - %x[slf5s.NDC]
#Thread - %t[slf5s.THREAD]
#Category - %c[slf5s.CATEGORY]
#Location - %l[slf5s.LOCATION]
#Message - %m[slf5s.MESSAGE]
#
#log4j.appender.R.layout.ConversionPattern=[slf5s.start]%d{DATE}[slf5s.DATE]%n/
#   %p[slf5s.PRIORITY]%n%x[slf5s.NDC]%n%t[slf5s.THREAD]%n/
#   %c[slf5s.CATEGORY]%n%l[slf5s.LOCATION]%n%m[slf5s.MESSAGE]%n%n
##########下面是中文说明#############
#%m 输出代码中指定的消息
#%p 输出优先级,即DEBUG,INFO,WARN,ERROR,FATAL
#%r 输出自应用启动到输出该log信息耗费的毫秒数
#%c 输出所属的类目,通常就是所在类的全名
#%t 输出产生该日志事件的线程名
#%n 输出一个回车换行符,Windows平台为“/r/n”,Unix平台为“/n”
#%d 输出日志时间点的日期或时间,默认格式为ISO8601,也可以在其后指定格式,
#比如:%d{yyy MMM dd HH:mm:ss,SSS},输出类似:2002年10月18日 22:10:28,921
#%l 输出日志事件的发生位置,包括类目名、发生的线程,以及在代码中的行数。举例:Testlog4.main(TestLog4.java:10)

6.Log4JTest.java


package com.benqguru.palau.log.jdbc.test;

import java.sql.*;

import org.apache.log4j.*;

/**
 * 这是一个用配置文件来配置JDBCAppender的实例代码
 * <p>Title: log4j1.2.8源码解析</p>
 * <p>Description: </p>
 * <p>Copyright: Copyright (c) 2004</p>
 * <p>Company: 明基逐鹿</p>
 * @author 王建
 * @version 1.0
 */
public class Log4JTest {

  /**
   * 根日志记录器
   */
  static Logger log = Logger.getLogger(Log4JTest.class.getName());

  public static void main(String[] args) {
    try {
      Driver d = (Driver) (Class.forName("org.gjt.mm.mysql.Driver").
                           newInstance());
      DriverManager.registerDriver(d);
    }
    catch (Exception e) {}

    // Configuration with configuration-file
    PropertyConfigurator.configure(
        "E:/Log4j_src/classes/com/benqguru/palau/log/jdbc/test/log4j.properties");

    // These messages with Priority >= setted priority will be logged to the database.
    log.debug("debug"); //this not, because Priority DEBUG is less than INFO
    log.info("info");
    log.error("error");
    log.fatal("fatal");
  }
}

7.Log4JTest2.java

package com.benqguru.palau.log.jdbc.test;

import java.sql.*;

import org.apache.log4j.*;

/**
 * 这是一个没有使用配置文件配置JDBCAppender的实例代码
 * <p>Title: log4j1.2.8源码解析</p>
 * <p>Description: </p>
 * <p>Copyright: Copyright (c) 2004</p>
 * <p>Company: 明基逐鹿</p>
 * @author 王建
 * @version 1.0
 */
public class Log4JTest2 {

  /**
   * 根日志记录器
   */
  static Logger log = Logger.getLogger(Log4JTest2.class.getName());

  public static void main(String[] args) {
    //返回事务的序列号
    MyIDHandler idhandler = new MyIDHandler();

    //确保已安装了所需的驱动器
    try {
      Driver d = (Driver) (Class.forName("org.gjt.mm.mysql.Driver").
                           newInstance());
      DriverManager.registerDriver(d);
    }
    catch (Exception e) {
      e.printStackTrace();
    }

    //设置消息可以被记录的优先权
    log.setLevel(Level.DEBUG);

    //创建一个JDBCAppender实例
    JDBCAppender appender = new JDBCAppender();

    //用setOption()方法设置可选项
    appender.setOption(JDBCAppender.CONNECTOR_OPTION, "com.benqguru.palau.log.jdbc.test.MyConnectionHandler");
    appender.setOption(JDBCAppender.URL_OPTION, "jdbc:mysql://localhost:3306/logtest");
    appender.setOption(JDBCAppender.USERNAME_OPTION, "root");
    appender.setOption(JDBCAppender.PASSWORD_OPTION, "");
    //进行这步操作的时候,JDBCAppender将调用JDBCLogger连接数据库,查找数据库中表的信息
    appender.setOption(JDBCAppender.TABLE_OPTION, "logtest");

    /*
       有两个方法建立列描述:
       1.用方法setOption(JDBCAppender.COLUMNS_OPTION, column-description)
       ja.setOption(JDBCAppender.COLUMNS_OPTION, "id_seq~EMPTY id~ID~MyIDHandler msg~MSG created_on~TIMESTAMP created_by~STATIC~:-) Thomas Fenner ([email protected])");
       2.用setLogType(String columnname, int LogType.xxx, Object xxx)是一个较好的编码方法
    */
    appender.setLogType("id_seq", LogType.EMPTY, "");
    appender.setLogType("id", LogType.ID, idhandler);
    appender.setLogType("msg", LogType.MSG, "");
    appender.setLogType("created_on", LogType.TIMESTAMP, "");
    appender.setLogType("created_by", LogType.STATIC, "FEN");

    //假如你恰恰想执行一个静态的sql语句,忘记表和列选项,用这个方法:
//    appender.setOption(JDBCAppender.SQL_OPTION, "INSERT INTO LOGTEST (id, msg, created_on, created_by) VALUES (1, @MSG@, sysdate, 'me')");
    // 设置数据库缓冲器选项的大小为1
//    appender.setOption(JDBCAppender.BUFFER_OPTION, "1");
    //若数据库用的是MySql,则一定要设置为"N",因为MySql不支持事务
//    appender.setOption(JDBCAppender.COMMIT_OPTION, "N");

    // 定义一个布局器,%m代表输出代码中指定的消息
//    appender.setLayout(new PatternLayout("%m"));

    //添加一个输出源到日志记录器
    log.addAppender(appender);

    //这些带优先权的消息若大于或等于已设定的优先权,将被记录进数据库
    log.debug("debug");
    log.info("info");
    log.error("error");
    log.fatal("fatal");
  }
}

/**
 * 实现JDBCConnectionHandler接口的实例
 * <p>Title: log4j1.2.8源码解析</p>
 * <p>Description: </p>
 * <p>Copyright: Copyright (c) 2004</p>
 * <p>Company: 明基逐鹿</p>
 * @author 王建
 * @version 1.0
 */
class MyConnectionHandler
    implements JDBCConnectionHandler {
  Connection con = null;
//  String url = "jdbc:oracle:thin:@(DESCRIPTION=(ADDRESS_LIST=(ADDRESS=(COMMUNITY=tcp.world)(PROTOCOL=TCP)(Host=LENZI)(Port=1521))(ADDRESS=(COMMUNITY=tcp.world)(PROTOCOL=TCP)(Host=LENZI)(Port=1526)))(CONNECT_DATA=(SID=LENZI)))";
  String url = "jdbc:mysql://localhost:3306/logtest";
  String username = "root";
  String password = "";

  /**
   * 获得默认数据库连接
   * @return Connection
   */
  public Connection getConnection() {
    System.out.println("url="+url);
    System.out.println("username="+username);
    System.out.println("password="+password);
    return getConnection(url, username, password);
  }

  /**
   * 根据给定条件获得数据库连接
   * @param _url String
   * @param _username String
   * @param _password String
   * @return Connection
   */
  public Connection getConnection(String _url, String _username,
                                  String _password) {
    try {
      if (con != null && !con.isClosed()) con.close();
      con = DriverManager.getConnection(_url, _username, _password);
      con.setAutoCommit(true);
    }
    catch (Exception e) {
      e.printStackTrace();
    }

    return con;
  }
}


/**
 * 实现JDBCIDHandler接口的实例
 * <p>Title: log4j1.2.8源码解析</p>
 * <p>Description: </p>
 * <p>Copyright: Copyright (c) 2004</p>
 * <p>Company: 明基逐鹿</p>
 * @author 王建
 * @version 1.0
 */
class MyIDHandler
    implements JDBCIDHandler {
  private static long id = 0;

  /**
   * 获得独一无二的ID
   * @return Object
   */
  public synchronized Object getID() {
    return new Long(++id);
  }
}

你可能感兴趣的:(log4j的数据库Appender的实现代码)