我如何获得PreparedStatement的SQL?

我有一个具有以下方法签名的普通Java方法:

private static ResultSet runSQLResultSet(String sql, Object... queryParams)

它打开一个连接,使用sql语句和queryParams变量长度数组中的参数构建PreparedStatement ,运行它,缓存ResultSet (在CachedRowSetImpl ),关闭连接并返回缓存的结果集。

我在记录错误的方法中有异常处理。 我将sql语句记录为日志的一部分,因为它对调试非常有帮助。 我的问题是,日志记录的字符串变量sql记录模板语句的?的,而不是实际值。 我想记录已执行(或试图执行)的实际语句。

所以......有没有什么办法可以获得将由PreparedStatement运行的实际SQL语句? (如果我自己没有构建它,如果我找不到一种访问PreparedStatement's SQL的方法,那么我最终可能会在我的catch自行构建它。)


使用预准备语句,不存在“SQL查询”:

  • 你有一个包含占位符的声明
  • 它被发送到数据库服务器
  • 并在那里准备
  • 这意味着SQL语句被“分析”,解析,表示它的一些数据结构在内存中准备好了
  • 然后,你有绑定变量
  • 将其发送到服务器
  • 并准备好的陈述被执行 - 处理这些数据
  • 但是并没有重新构建实际的真正的SQL查询 - 无论是在Java方面还是在数据库方面。

    因此,没有办法获得准备好的语句的SQL - 因为没有这样的SQL。


    为了调试目的,解决方案要么是:

  • 输出语句的代码,占位符和数据列表
  • 或者“手动”建立一些SQL查询。

  • 它在JDBC API合同中没有定义,但如果幸运的话,所讨论的JDBC驱动程序可以通过调用PreparedStatement#toString()来返回完整的SQL。 即

    System.out.println(preparedStatement);
    

    至少MySQL 5.x和PostgreSQL 8.x JDBC驱动程序支持它。 但是,大多数其他JDBC驱动程序不支持它。 如果你有这样一个,那么你最好的选择是使用Log4jdbc或P6Spy。

    或者,您也可以编写一个通用函数,该函数接收Connection ,SQL字符串和语句值,并在记录SQL字符串和值之后返回PreparedStatement 。 开球示例:

    public static PreparedStatement prepareStatement(Connection connection, String sql, Object... values) throws SQLException {
        PreparedStatement preparedStatement = connection.prepareStatement(sql);
        for (int i = 0; i < values.length; i++) {
            preparedStatement.setObject(i + 1, values[i]);
        }
        logger.debug(sql + " " + Arrays.asList(values));
        return preparedStatement;
    }
    

    并将其用作

    try {
        connection = database.getConnection();
        preparedStatement = prepareStatement(connection, SQL, values);
        resultSet = preparedStatement.executeQuery();
        // ...
    

    另一种方法是实现一个自定义的PreparedStatement ,它在构造时包装(装饰)实际的PreparedStatement ,并覆盖所有方法,以便调用真实PreparedStatement的方法并收集所有setXXX()方法中的值,并且懒惰地构造“actual “一个executeXXX()方法被调用时的SQL字符串(相当executeXXX() ,但大多数IDE为decorator方法提供了自动生成器,Eclipse也是这样)。 最后只是用它来代替。 这也基本上是P6Spy和consorts在引擎盖下已经做的。


    如果您正在执行查询并期待ResultSet (至少在这种情况下),那么您可以简单地调用ResultSetgetStatement()如下所示:

    ResultSet rs = pstmt.executeQuery();
    String executedQuery = rs.getStatement().toString();
    

    变量executedQuery将包含用于创建ResultSet的语句。

    现在,我意识到这个问题是相当古老的,但我希望这可以帮助一个人..

    链接地址: http://www.djcxy.com/p/27827.html

    上一篇: How can I get the SQL of a PreparedStatement?

    下一篇: Split a listview into multiple columns