启用JDBC的日志记录

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

我已经完成了这个SO后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.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来运行我的程序时,我得到exception,因为:

 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 ,那么我也得到相同的exception。

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

更新:现在我将ojdbc6dms.jar文件放在classpath中,我的程序给出了以下exception:

 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.(DMSFactory.java:48) at oracle.jdbc.driver.PhysicalConnection.createDMSSensors(PhysicalConnection.java:2121) at oracle.jdbc.driver.PhysicalConnection.(PhysicalConnection.java:730) at oracle.jdbc.driver.T4CConnection.(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 

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

在此处输入图像描述

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

在此处输入图像描述

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

有很多Spy框架可用于此目的,请检查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许可证授权

用法

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

    例如,如果您的普通jdbc url是jdbc:derby:// localhost:1527 // db-derby-10.2.2.0-bin / databases / MyDatabase,那么您可以将其更改为:jdbc:log4jdbc:derby:// localhost: 1527 // DB-德比10.2.2.0斌/数据库/ MyDatabase的

  • 设置记录器。

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

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

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

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

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

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

Oracle自己在这里描述了这一点 ,经过一些反复试验后,我得到了它的工作:

  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 startup命令中。

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

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

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

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