前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Log4j 1.x JDBCAppender记录日志失效问题详解

Log4j 1.x JDBCAppender记录日志失效问题详解

作者头像
编程随笔
发布2019-09-11 15:43:51
6660
发布2019-09-11 15:43:51
举报
文章被收录于专栏:后端开发随笔后端开发随笔

官网:http://logging.apache.org/log4j/1.2/manual.html

事件: 最近在项目中使用log4j 1.x JDBCAppender记录管理员操作日志到数据库,在测试时发现系统启动后运行一段时间无法继续记录相关操作日志到数据库。 配置如下: log4j.properties:

代码语言:javascript
复制
log4j.logger.oplog=INFO, oplog
log4j.appender.oplog=com.lenovo.moc.portal.dao.LogJDBCAppender
log4j.appender.oplog.driver=com.mysql.jdbc.Driver
log4j.appender.oplog.URL=jdbc:mysql://192.168.2.164:3306/oplog?characterEncoding=utf8
log4j.appender.oplog.user=xxx
log4j.appender.oplog.password=xxx
log4j.appender.oplog.sql=insert into operation_loginfo (staff_id, staff_name, user_role, op_type, op_alias, create_time, content, content_alias) values ('%x{login_staff_id}', '%x{login_staff_name}','%x{login_user_role}', '%x{op_type}', '%x{op_alias}', '%d{yyyy-mm-dd hh:mm:ss}','%m', '%x{content_alias}')
log4j.appender.oplog.layout=org.apache.log4j.PatternLayout

java代码:

代码语言:javascript
复制
public class OperationLogService {
  private static final Logger logger = Logger.getLogger(OperationLogService.class);
  private static ExecutorService threadPool = Executors.newFixedThreadPool(3);;

  private static ExecutorService getThreadPool() {
    return threadPool;
  }

  /**
  * 记录操作日志
  * @param login_staff_id 员工id
  * @param login_staff_name 员工姓名
  * @param login_user_role 员工角色
  * @param op_type 操作类型
  * @param op_alias 操作别名
  * @param content_alias 操作内容
  * @param msg 附加信息
  */
  public static void log(String login_staff_id, String login_staff_name, String login_user_role, String op_type,
    String op_alias, String content_alias, final String msg) {
    getThreadPool().execute(new Runnable() {
      @Override
      public void run() {
        MDC.put("login_staff_id", login_staff_id);
        MDC.put("login_staff_name", login_staff_name);
        MDC.put("login_user_role", login_user_role);
        MDC.put("op_type", op_type);
        MDC.put("op_alias", op_alias);
        MDC.put("content_alias", content_alias);
       logger.info(msg);
      }
    });
  }

  public static void main(String[] args) {
    log("1", "zhangsan", "admin", "add_user", "添加用户", "zhangsan添加用户", "test msg");
  }
}

解决办法: 通过查看log4j 1.x JDBCAppender源码发现,并没有对数据库连接的有效性进行判断。即:一旦数据库连接断开,就无法继续写入日志。 故而,通过扩展JDBCAppender的方式,进行数据库连接重连处理:

代码语言:javascript
复制
/**
* 自定义实现Log4j日志组件,将日志记录到数据库<br />.
* 解决问题: 原生组件在系统运行过程中可能会出现数据库连接断开,导致无法正常记录日志信息到数据库.
*
* @desc com.lenovo.moc.portal.dao.LogJDBCAppender
* @author chench9@lenovo.com
* @date 2017年3月15日
*/
public class LogJDBCAppender extends JDBCAppender {
  private static final Logger logger = Logger.getLogger(LogJDBCAppender.class);

  @Override
  protected Connection getConnection() throws SQLException {
    Connection connection = super.getConnection();
    if(connection == null || connection.isClosed()) {
      logger.warn(String.format("reconnect log jdbc appender connection"));
      connection = reconnect();
    }
    return connection;
  }

  /**
  * 重新创建数据库连接
  * @return
  * @throws SQLException
  */
  private Connection reconnect() throws SQLException {
    Connection connection = DriverManager.getConnection(databaseURL, databaseUser,databasePassword);
    return connection;
  }

  /**
  * 重载父类方法,打印错误信息到日志文件 <br />
  * 同时,处理数据库重连并在出错时重试记录日志信息.
  */
  @Override
  protected void execute(String sql) throws SQLException {
    try {
      super.execute(sql);
    } catch (Exception e) {
      logger.error(String.format("log jdbc appender execute sql eror: %s", getSql()), e);
      closeConnectionInterval();
      super.execute(sql);
    }
  }

  // 真正地关闭数据库连接
  private void closeConnectionInterval() {
    if(connection == null) {
      return;
    }

    try {
      connection.close();
    } catch (SQLException e) {
      e.printStackTrace();
    } finally {
      connection = null;
    }
  }
}

log4j 1.x org.apache.log4j.jdbc.JDBCAppender类图:

org.apache.log4j.jdbc.JDBCAppender数据库连接实现:

log4j 2.x org.apache.logging.log4j.core.appender.db.jdbc.JdbcAppender类图:

显然,在log4j 2.x中,使用了数据库连接池,所以建议使用log4j 2.x版本的JdbcAppender。

【参考】 http://stackoverflow.com/questions/3880521/reconnect-to-db-within-log4j Reconnect to DB within log4j

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2017-05-02 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
相关产品与服务
数据库
云数据库为企业提供了完善的关系型数据库、非关系型数据库、分析型数据库和数据库生态工具。您可以通过产品选择和组合搭建,轻松实现高可靠、高可用性、高性能等数据库需求。云数据库服务也可大幅减少您的运维工作量,更专注于业务发展,让企业一站式享受数据上云及分布式架构的技术红利!
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档