微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

利用JDBC的PrepareStatement打印真实SQL的方法详解

PreparedStatement是预编译的,对于批量处理可以大大提高效率. 也叫JDBC存储过程,下面这篇文章主要给大家介绍了关于利用JDBC的PrepareStatement打印真实sql方法,需要的朋友可以参考借鉴,下面来一起看看吧。

前言

本文主要给大家介绍了关于利用JDBC的PrepareStatement打印真实sql的相关内容分享出来供大家参考学习,下面来一起看看详细的介绍:

我们知道,JDBC 的 PrepareStatement 优点多多,通常都是推荐使用 PrepareStatement 而不是其基类 Statment。PrepareStatement 支持 ? 占位符,可以将参数按照类型转自动换为真实的值。既然这一过程是自动的,封装在 JDBC 内部的,那么我们外部就不得而知目标的 sql 最终生成怎么样――于是在调试过程中便有一个打印 sql 的问题。我们对 PrepareStatement 传入 sql 语句,如 SELECT * FROM table WHERE id = ?,然后我们传入对应的 id 参数,假设是 id = 10,那怎么把得到参数的 SELECT * FROM table WHERE id =  12 结果完整地得出来呢?――这便是本文所要探讨的问题。下面话不多说了,来一起看看详细的介绍:

方法如下:

首先,我们看看典型的一个 PrepareStatement 调用方法,如下一个函数

/** * 查询单个结果,保存为 Map 结构。如果查询不到任何数据返回 null。 * * @param conn * 数据库连接对象 * @param sql * sql 语句,可以带有 ? 的占位符 * @param params * 插入到 sql 中的参数,可单个可多个可不填 * @return Map 结构的结果。如果查询不到任何数据返回 null。 */ public static Map query(Connection conn, String sql, Object... params) { Map map = null; printRealsql(sql, params); // 打印真实 sql函数 try (PreparedStatement ps = conn.prepareStatement(sql);) { if(params != null) for (int i = 0; i

值得注意该函数里面:

printRealsql(sql, params); // 打印真实 sql函数

其参数一 sql 就是类似 SELECT * FROM table WHERE id = ? 的语句,参数二 params 为 Object... params 的参数列表,可以是任意类似的合法 sql 值。最后,通过 printRealsql 函数最终得出形如 SELECT * FROM table WHERE id =  12 的结果。

printRealsql 函数源码如下:

/** * 在开发过程,sql语句有可能写错,如果能把运行时出错的 sql 语句直接打印出来,那对排错非常方便,因为其可以直接拷贝到数据库客户端进行调试。 * * @param sql * sql 语句,可以带有 ? 的占位符 * @param params * 插入到 sql 中的参数,可单个可多个可不填 * @return 实际 sql 语句 */ public static String printRealsql(String sql, Object[] params) { if(params == null || params.length == 0) { LOGGER.info("The sql is------------>n" + sql); return sql; } if (!match(sql, params)) { LOGGER.info("sql 语句中的占位符与参数个数不匹配。sql:" + sql); return null; } int cols = params.length; Object[] values = new Object[cols]; System.arraycopy(params, 0, values, 0, cols); for (int i = 0; i n" + statement); ConnectionMgr.addsql(statement); // 用来保存日志 return statement; } /** * ? 和参数的实际个数是否匹配 * * @param sql * sql 语句,可以带有 ? 的占位符 * @param params * 插入到 sql 中的参数,可单个可多个可不填 * @return true 表示为 ? 和参数的实际个数匹配 */ private static boolean match(String sql, Object[] params) { if(params == null || params.length == 0) return true; // 没有参数,完整输出 Matcher m = Pattern.compile("(\?)").matcher(sql); int count = 0; while (m.find()) { count++; } return count == params.length; }

可见,上述思路是非常简单的,――有多少个 ? 占位符,就要求有多少个参数,然后一一对照填入(数组)。match 函数会检查第一个步骤,检查个数是否匹配,否则会返回“sql 语句中的占位符与参数个数不匹配”的提示;然后,参数的值会被转换为符合 sql 值所要求的类型;最后,就是将 sql 一一填入,――此处使用了一个字符串的技巧,先把 ? 字符通通转换为 %s,――那是 String.format 可识别的占位符,如此再传入 Object[] 参数列表,即可得出我们期待的 sql 结果。

我们不能保证那 sql 可以直接放到数据库中被解析。因为我们的初衷只是把 sql 打印出来,务求更近一步让程序员在开发阶段看到 sql 是怎么样子的,而且不是一堆 ?、?……,这样会显得更符合真实情形一点。

PrepareStatement 内部源码肯定有这一步骤或者某个变量是表示那个真实 sql 的,――只是没有暴露出来。如果有,那么对程序员会更友好一些。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对编程之家的支持

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐