如何获取PreparedStatement的SQL?

fro*_*die 150 java sql jdbc prepared-statement

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

private static ResultSet runSQLResultSet(String sql, Object... queryParams)
Run Code Online (Sandbox Code Playgroud)

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

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

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

Pas*_*TIN 166

使用预准备语句,没有"SQL查询":

  • 您有一个包含占位符的声明
    • 它被发送到DB服务器
    • 并在那里准备
    • 这意味着SQL语句被"分析",解析,表示它的一些数据结构在内存中准备
  • 然后,你已经绑定了变量
    • 它们被发送到服务器
    • 并执行准备好的声明 - 处理这些数据

但是没有重构真正的实际SQL查询 - 既不在Java端,也不在数据库端.

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


出于调试目的,解决方案要么:

  • 使用占位符和数据列表输出语句的代码
  • 或者"手动""构建"一些SQL查询.

  • 虽然这在功能上是正确的,但没有什么能阻止实用程序代码重建等效的无准备语句.例如,在log4jdbc中:"在记录的输出中,对于预准备语句,绑定参数会自动插入到SQL输出中.这极大地提高了许多情况下的可读性和调试性." 对于调试非常有用,只要您知道这不是DB服务器实际执行该语句的方式. (21认同)
  • 这也取决于实施.在MySQL中 - 至少是我几年前使用的版本 - JDBC驱动程序实际上是从模板构建了一个传统的SQL查询并绑定变量.我想这个版本的MySQL本身不支持预处理语句,因此他们在JDBC驱动程序中实现了它们. (6认同)
  • 如果你正在使用java.sql.PreparedStatement,那么prepareStatement上的一个简单的.toString()将包含我在1.8.0_60中验证过的生成的SQL (5认同)
  • @Preston对于Oracle DB,PreparedStatement#toString()不显示SQL.因此我想这取决于DB JDBC驱动程序. (5认同)

Bal*_*usC 54

它在JDBC API合同中没有任何定义,但是如果你很幸运,有问题的JDBC驱动程序可以通过调用返回完整的SQL PreparedStatement#toString().即

System.out.println(preparedStatement);
Run Code Online (Sandbox Code Playgroud)

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

或者,您也可以编写一个泛型函数,它接受一个ConnectionSQL字符串和语句值,并PreparedStatement在记录SQL字符串和值后返回一个.开球示例:

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;
}
Run Code Online (Sandbox Code Playgroud)

并用它作为

try {
    connection = database.getConnection();
    preparedStatement = prepareStatement(connection, SQL, values);
    resultSet = preparedStatement.executeQuery();
    // ...
Run Code Online (Sandbox Code Playgroud)

另一个替代方法是实现一个自定义PreparedStatement,它包装(修饰)构造中的真实 PreparedStatement并覆盖所有方法,以便它调用真实 的方法PreparedStatement并收集所有setXXX()方法中的值并懒惰地构造"实际"SQL字符串,只要其中一个executeXXX()调用这些方法(非常有用,但是大多数IDE为装饰器方法提供了自动生成器,Eclipse确实如此).最后只需使用它.这也基本上是P6Spy和配件已经在引擎盖下做的事情.


use*_*ond 33

我正在使用带有MySQL连接器v.5.1.31的Java 8,JDBC驱动程序.

我可以使用此方法获得真正的SQL字符串:

// 1. make connection somehow, it's conn variable
// 2. make prepered statement template
PreparedStatement stmt = conn.prepareStatement(
    "INSERT INTO oc_manufacturer" +
    " SET" +
    " manufacturer_id = ?," +
    " name = ?," +
    " sort_order=0;"
);
// 3. fill template
stmt.setInt(1, 23);
stmt.setString(2, 'Google');
// 4. print sql string
System.out.println(((JDBC4PreparedStatement)stmt).asSql());
Run Code Online (Sandbox Code Playgroud)

所以它返回像这样的smth:

INSERT INTO oc_manufacturer SET manufacturer_id = 23, name = 'Google', sort_order=0;
Run Code Online (Sandbox Code Playgroud)


Ela*_*ern 18

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

ResultSet rs = pstmt.executeQuery();
String executedQuery = rs.getStatement().toString();
Run Code Online (Sandbox Code Playgroud)

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

现在,我意识到这个问题已经很老了,但我希望这有助于某些人......

  • @Elad Stern它打印,oracle.jdbc.driver.OraclePreparedStatementWrapper@1b9ce4b而不是打印执行的sql语句!请指导我们! (6认同)
  • rs.getStatement()只返回语句对象,所以这取决于你使用的驱动程序是否实现.toString()确定你是否会返回SQL (3认同)

Chr*_*ssy 5

toString

\n

使用PostgreSQL 9.6.x和官方 Java 驱动程序,42.2.4我们可以简单地调用PreparedStatement#toString.

\n
myPreparedStatement.set\xe2\x80\xa6( \xe2\x80\xa6 , \xe2\x80\xa6 ) ;\nString sql = myPreparedStatement.toString() ;\n
Run Code Online (Sandbox Code Playgroud)\n

将显示已替换的 SQL ?

\n

我从来没有想到事情会如此简单。

\n