如何使用JDBC时启用SQL语句的日志记录

问题描述 投票:27回答:4

我试图通过在Eclipse IDE中连接到Oracle数据库来使用JDBC程序启用日志。

我已经看过这篇特别文章JDBC logging to file然后我在Java程序下面创建了一个程序,并从Eclipse IDE中运行了它,但是我看不到JDBC驱动程序类生成的任何日志。

import java.io.File;
import java.io.FileInputStream;
import java.io.PrintWriter;
import java.sql.Connection;
import java.sql.DriverManager;
import java.sql.SQLException;
import java.util.Properties;
import java.util.logging.LogManager;
import java.util.logging.Logger;

public class Logging {

    static Logger log = Logger.getLogger(Logging.class.toString());
    static Connection con = null;

    public static void main(String[] args) throws SQLException,
            ClassNotFoundException {
        System.setProperty("oracle.jdbc.Trace", Boolean.TRUE.toString());
        System.setProperty("java.util.logging.config.file",
                "OracleLog.properties");
        log.info("Test Message");
        enableLogging(false);
        getConnection();
        closeConnection();
    }

    static private void enableLogging(boolean logDriver) {
        try {
            oracle.jdbc.driver.OracleLog.setTrace(true);

            // compute the ObjectName
            String loader = Thread.currentThread().getContextClassLoader()
                    .toString().replaceAll("[,=:\"]+", "");
            javax.management.ObjectName name = new javax.management.ObjectName(
                    "com.oracle.jdbc:type=diagnosability,name=" + loader);

            // get the MBean server
            javax.management.MBeanServer mbs = java.lang.management.ManagementFactory
                    .getPlatformMBeanServer();

            // find out if logging is enabled or not
            System.out.println("LoggingEnabled = "
                    + mbs.getAttribute(name, "LoggingEnabled"));

            // enable logging
            mbs.setAttribute(name, new javax.management.Attribute(
                    "LoggingEnabled", true));

            File propFile = new File("path/to/properties");
            LogManager logManager = LogManager.getLogManager();
            logManager.readConfiguration(new FileInputStream(propFile));

            if (logDriver) {
                DriverManager.setLogWriter(new PrintWriter(System.err));
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    public static Connection getConnection() throws SQLException,
            ClassNotFoundException {
        Properties connectionProps = new Properties();
        connectionProps.put("user", "test_app");
        connectionProps.put("password", "test");

        Class.forName("oracle.jdbc.driver.OracleDriver");
        con = DriverManager.getConnection(
                "jdbc:oracle:thin:@"+HOST_IP+":1521:"+SID,
                connectionProps);
        System.out.println("Connected to database");
        return con;
    }

    public static void closeConnection() throws SQLException {
        if (con != null) {
            con.close();
        }
    }

}

并且我在OracleLog.properties文件中具有以下内容:

.level=SEVERE
oracle.jdbc.level=INFO
oracle.jdbc.handlers=java.util.logging.ConsoleHandler
java.util.logging.ConsoleHandler.level=INFO
java.util.logging.ConsoleHandler.formatter=java.util.logging.SimpleFormatter

但是当我通过将ojdbc6-11.2.0.3.jar放在类路径中运行程序时,我得到的异常是:

INFO: Test Message
javax.management.InstanceNotFoundException: com.oracle.jdbc:type=diagnosability,name=sun.misc.Launcher$AppClassLoader@73d16e93
    at com.sun.jmx.interceptor.DefaultMBeanServerInterceptor.getMBean(DefaultMBeanServerInterceptor.java:1095)
    at com.sun.jmx.interceptor.DefaultMBeanServerInterceptor.getAttribute(DefaultMBeanServerInterceptor.java:643)
    at com.sun.jmx.mbeanserver.JmxMBeanServer.getAttribute(JmxMBeanServer.java:678)
    at myjdbc.Logging.enableLogging(Logging.java:45)
    at myjdbc.Logging.main(Logging.java:24)
Connected to database

如果我在类路径中有ojdbc6_g.jar,那么我也会遇到同样的异常。

[请让我知道如何启用JDBC程序的日志记录?基本上,我希望看到内部JDBC代码生成的日志。

更新:现在,我将ojdbc6dms.jar文件放在类路径中,程序给出以下异常:

Nov 28, 2014 9:09:02 PM jdbc.chap2.Logging main
INFO: Test Message
javax.management.InstanceNotFoundException: com.oracle.jdbc:type=diagnosability,name=sun.misc.Launcher$AppClassLoader@73d16e93
    at com.sun.jmx.interceptor.DefaultMBeanServerInterceptor.getMBean(DefaultMBeanServerInterceptor.java:1095)
    at com.sun.jmx.interceptor.DefaultMBeanServerInterceptor.getAttribute(DefaultMBeanServerInterceptor.java:643)
    at com.sun.jmx.mbeanserver.JmxMBeanServer.getAttribute(JmxMBeanServer.java:678)
    at jdbc.chap2.Logging.enableLogging(Logging.java:45)
    at jdbc.chap2.Logging.main(Logging.java:24)
Exception in thread "main" java.lang.NoClassDefFoundError: oracle/dms/console/DMSConsole
    at oracle.jdbc.driver.DMSFactory.<clinit>(DMSFactory.java:48)
    at oracle.jdbc.driver.PhysicalConnection.createDMSSensors(PhysicalConnection.java:2121)
    at oracle.jdbc.driver.PhysicalConnection.<init>(PhysicalConnection.java:730)
    at oracle.jdbc.driver.T4CConnection.<init>(T4CConnection.java:433)
    at oracle.jdbc.driver.T4CDriverExtension.getConnection(T4CDriverExtension.java:32)
    at oracle.jdbc.driver.OracleDriver.connect(OracleDriver.java:608)
    at java.sql.DriverManager.getConnection(DriverManager.java:664)
    at java.sql.DriverManager.getConnection(DriverManager.java:208)
    at jdbc.chap2.Logging.getConnection(Logging.java:70)
    at jdbc.chap2.Logging.main(Logging.java:25)
Caused by: java.lang.ClassNotFoundException: oracle.dms.console.DMSConsole
    at java.net.URLClassLoader$1.run(URLClassLoader.java:372)
    at java.net.URLClassLoader$1.run(URLClassLoader.java:361)
    at java.security.AccessController.doPrivileged(Native Method)
    at java.net.URLClassLoader.findClass(URLClassLoader.java:360)
    at java.lang.ClassLoader.loadClass(ClassLoader.java:424)
    at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:308)
    at java.lang.ClassLoader.loadClass(ClassLoader.java:357)
    ... 10 more
java sql oracle logging jdbc
4个回答
8
投票

如果使用Spring框架,则datasource-proxy非常方便。您基本上可以包装任何DataSource,只需添加日志记录行为。

enter image description here

如果您使用的是Java EE,那么P6spy是一个不错的选择:

enter image description here

在幕后,p6spy提供了Driver级别的语句拦截器,这对于Java EE应用程序更加方便,因为DataSource由应用程序服务器提供。


13
投票

2019更新:自2015年以来未维护log4jdbc。p6spy似乎仍在积极维护。

原始答案

有很多用于此目的的Spy框架,请检查log4jdbc,我发现这是您想要的。

功能

  • 完全支持JDBC 3和JDBC 4!
  • 在大多数情况下,易于配置您需要做的就是将驱动程序类名称更改为net.sf.log4jdbc.DriverSpy并将“ jdbc:log4”添加到现有的前缀jdbc url,设置您的日志记录类别,就可以开始了!
  • 在记录的输出中,对于准备好的语句,绑定参数为自动插入到SQL输出中。这大大改善了在许多情况下的可读性和调试。
  • SQL时间信息可以生成,以帮助确定SQL语句运行多长时间,帮助识别运行速度太慢的语句,这可以使用附带的工具对数据进行后期处理以生成性能分析报告数据以快速识别应用程序中的慢速SQL。
  • SQL连接号信息生成以帮助识别连接池或线程问题。适用于任何基础具有JDK 1.4和更高版本以及SLF4J 1.x的JDBC驱动程序。
  • 开源软件,根据商业友好的Apache 2.0许可进行许可

用法

  • 将log4jdbc jar(基于JDK版本)放入应用程序的类路径中。
  • 支持选择使用的日志系统,log4j,logback,commons logging ..等
  • 在应用程序的配置中,将JDBC驱动程序类设置为net.sf.log4jdbc.DriverSpy。在许多情况下,被监视的基础驱动程序将自动加载,而无需任何其他配置。
  • 将jdbc:log4附加到您正在使用的普通jdbc URL。

    例如,如果您的常规jdbc网址是jdbc:derby:// localhost:1527 // db-derby-10.2.2.0-bin / databases / MyDatabase然后您将其更改为:jdbc:log4jdbc:derby:// localhost:1527 // db-derby-10.2.2.0-bin / databases / MyDatabase

  • 设置您的记录器。

    jdbc.sqlonly:仅记录SQL。在准备好的语句中执行的SQL会自动显示,其绑定参数替换为该位置上绑定的数据,从而大大提高了可读性。 1.0

    jdbc.sqltiming:在执行后记录SQL,包括有关SQL执行所需时间的计时统计信息。 1.0

    jdbc.audit:记录除结果集外的所有JDBC调用。这是非常庞大的输出,除非跟踪特定的JDBC问题,否则通常不需要。 1.0

    jdbc.resultset:更为庞大,因为记录了对ResultSet对象的所有调用。 1.0

    jdbc.connection:记录连接打开和关闭事件以及转储所有打开的连接号。这对于解决连接泄漏问题非常有用。


8
投票

我知道这是一个很老的话题,但是还没有提到的是,对于Oracle,存在一种解决方案,不需要使用应用程序代码进行任何更改,只需使用所需的启用了跟踪功能的Oracle JDBC驱动程序并通过启动时的JVM属性。

Oracle本身已经描述了此here,经过一番尝试和错误后,我认为它可以正常工作:

  1. 将启用了跟踪的ojdbc jar文件放在您的类路径中。从链接的Oracle页面中引用:“要获取日志输出,必须使用调试JAR文件,该文件的名称中带有“ _g”,例如ojdbc5_g.jar或ojdbc6_g.jar。我的Oracle 11g安装包含

  2. 按照链接的Oracle页面上的描述创建logging.properties文件,并根据需要调整日志记录级别。示例:

    .level=SEVERE oracle.jdbc.level=FINEST oracle.jdbc.handlers=java.util.logging.FileHandler java.util.logging.FileHandler.level=FINEST java.util.logging.FileHandler.pattern=jdbc.log java.util.logging.FileHandler.count=1 java.util.logging.FileHandler.formatter=java.util.logging.SimpleFormatter

  3. 将JVM属性“ -Doracle.jdbc.Trace = true -Djava.util.logging.config.file = logging.properties”添加到JDBC应用程序的java启动命令中。

JDBC应用程序现在应产生一个名为jdbc.log的文件,其中应包含所需的信息。在某些情况下,可能有必要指定logging.properties文件的完整路径。


2
投票

您是否尝试设置相应的系统属性?例如。对于TRACE:

System.setProperty( "oracle.jdbc.Trace", Boolean.TRUE.toString() );

(来自https://docs.oracle.com/cd/B28359_01/java.111/b31224/diagnose.htm

© www.soinside.com 2019 - 2024. All rights reserved.