我如何获得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查询 - 无论是在Java方面还是在数据库方面。
因此,没有办法获得准备好的语句的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
(至少在这种情况下),那么您可以简单地调用ResultSet
的getStatement()
如下所示:
ResultSet rs = pstmt.executeQuery();
String executedQuery = rs.getStatement().toString();
变量executedQuery
将包含用于创建ResultSet
的语句。
现在,我意识到这个问题是相当古老的,但我希望这可以帮助一个人..
链接地址: http://www.djcxy.com/p/27827.html