当前位置: 首页 > 知识库问答 >
问题:

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

范宏大
2023-03-14

我试图通过连接eclipse IDE中的Oracle数据库,使用JDBC程序启用日志。

我已经完成了这个SO post JDBC日志记录到文件,然后我创建了下面的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中有以下内容。属性文件:

.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\u g.jar,那么我也会遇到同样的异常。

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

更新:现在我放置了OJDBC6DM。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

共有3个答案

程和畅
2023-03-14

如果您使用的是Spring框架,那么数据源-代理框架非常方便。您基本上可以环绕任何DataSource,只需添加日志记录行为。

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

p6spy在幕后提供了驱动程序级别的语句拦截器,这对于Java EE应用程序来说更加方便,因为数据源是由应用程序服务器提供的。

屠浩
2023-03-14

我知道,这是一个非常古老的话题,但尚未提及的是,对于Oracle,存在一种解决方案,它不需要对应用程序代码进行任何更改,只需使用所需的启用跟踪的Oracle JDBC驱动程序并在启动时通过JVM属性启用日志记录。

Oracle自己在这里描述了这一点,经过一些反复试验,我让它发挥了作用:

>

  • 将启用跟踪的ojdbc jar文件放在类路径中。引用链接的Oracle页面:"要获得日志输出,您必须使用调试JAR文件,文件名中以"_g"表示,如ojdbc5_g.jar或ojdbc6_g.jar."包含我的Oracle 11g安装

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

    oracle.jdbc.handlersjava.util.logging.FileHandlerjava.util.logging.FileHandler.leveljava.util.logging.FileHandler.patternjdbc.logjava.util.logging.FileHandler.countjava.util.logging.FileHandler.formatterjava.util.logging.

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

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

  • 郎恺
    2023-03-14

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

    有很多间谍框架可用于此目的,请检查log4jdbc,我觉得这就是您正在寻找的。

    特色

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

    用法

    >

    例如,如果您的正常jdbc url是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

    jdbc.sqltiming:记录执行后的SQL,包括SQL执行时间的统计信息

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

    jdbc。resultset:甚至更大,因为对resultset对象的所有调用都会被记录。1

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

     类似资料:
    • 如何获得Spring Security的调试输出?

    • 我下载了kafka-clients-0.9.0.0。jar与maven一起使用,我希望我会看到类似于Kafka日志链接中的日志记录 然而,我不知道为什么我没有得到任何日志记录,即使我设置了引导。purpuse上的服务器错误,但它只是在没有发出任何警告的情况下被卡住了。我添加了几行代码以使用log4j打印到一个文件中,这似乎是可行的,但不知道为什么Kafka不能将事件记录到log4j中。 请注意,s

    • 我想调试ffmpeg。我添加以下代码来打印日志: 或 但它不能工作。没有任何调试信息。 然后启用调试生成选项: 它不能工作。 我确信我添加跟踪的地方会被执行。 我只想打印一些简单的信息,怎么做?

    • 问题内容: 我在Spring 3.5容器中将JpaTransactionManager与hibernate3一起使用。我无法为JPA启用日志记录。我希望看到用于调试我的某些服务的事务管理日志。我正在使用log4j。 这是我的log4j.properties中的内容 问题答案: 在启用JPATransactionManager的日志记录方面,这对我有用:1.下载logback jar(logback

    • 我有一个使用Hibernate作为提供者的Spring数据JPA存储库。我想记录SQL语句,但我做不到。我尝试了各种解决方案: 在我的HibernateJPavendorAdapter中将showSql设置为true 将log4j.logger.org.hibernate.sql=debug添加到我的log4j.properties文件(值得一提的是,log4j.logger.org.hibern

    • 我刚刚在EC2 Ubuntu 14.04 LTS上安装了我的第一个uWSGI服务器,使用以下配置: