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查询":
但是没有重构真正的实际SQL查询 - 既不在Java端,也不在数据库端.
因此,没有办法获得准备好的语句的SQL - 因为没有这样的SQL.
出于调试目的,解决方案要么:
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驱动程序不支持它.如果你有这样的,那么你最好的选择是使用Log4jdbc或P6Spy.
或者,您也可以编写一个泛型函数,它接受一个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(你在这种情况下,至少),那么你可以简单地调用ResultSet的getStatement(),像这样:
ResultSet rs = pstmt.executeQuery();
String executedQuery = rs.getStatement().toString();
Run Code Online (Sandbox Code Playgroud)
该变量executedQuery将包含用于创建的语句ResultSet.
现在,我意识到这个问题已经很老了,但我希望这有助于某些人......
toString使用PostgreSQL 9.6.x和官方 Java 驱动程序,42.2.4我们可以简单地调用PreparedStatement#toString.
myPreparedStatement.set\xe2\x80\xa6( \xe2\x80\xa6 , \xe2\x80\xa6 ) ;\nString sql = myPreparedStatement.toString() ;\nRun Code Online (Sandbox Code Playgroud)\n将显示已替换的 SQL ?。
我从来没有想到事情会如此简单。
\n