利用JDBC的PrepareStatement打印真實SQL的方法詳解
前言
本文主要給大家介紹了關于利用JDBC的PrepareStatement打印真實SQL的相關內容,分享出來供大家參考學習,下面來一起看看詳細的介紹:
我們知道,JDBC 的 PrepareStatement 優(yōu)點多多,通常都是推薦使用 PrepareStatement 而不是其基類 Statment。PrepareStatement 支持 ? 占位符,可以將參數(shù)按照類型轉自動換為真實的值。既然這一過程是自動的,封裝在 JDBC 內部的,那么我們外部就不得而知目標的 SQL 最終生成怎么樣——于是在調試過程中便有一個打印 SQL 的問題。我們對 PrepareStatement 傳入 SQL 語句,如 SELECT * FROM table WHERE id = ?,然后我們傳入對應的 id 參數(shù),假設是 id = 10,那怎么把得到參數(shù)的 SELECT * FROM table WHERE id = 12 結果完整地得出來呢?——這便是本文所要探討的問題。下面話不多說了,來一起看看詳細的介紹:
方法如下:
首先,我們看看典型的一個 PrepareStatement 調用方法,如下一個函數(shù),
/** * 查詢單個結果,保存為 Map<String, Object> 結構。如果查詢不到任何數(shù)據(jù)返回 null。 * * @param conn * 數(shù)據(jù)庫連接對象 * @param sql * SQL 語句,可以帶有 ? 的占位符 * @param params * 插入到 SQL 中的參數(shù),可單個可多個可不填 * @return Map<String, Object> 結構的結果。如果查詢不到任何數(shù)據(jù)返回 null。 */ public static Map<String, Object> query(Connection conn, String sql, Object... params) { Map<String, Object> map = null; printRealSql(sql, params); // 打印真實 SQL 的函數(shù) try (PreparedStatement ps = conn.prepareStatement(sql);) { if(params != null) for (int i = 0; i < params.length; i++) ps.setObject(i + 1, params[i]); try (ResultSet rs = ps.executeQuery();) { if (rs.isBeforeFirst()) { map = getResultMap(rs); } else { LOGGER.info("查詢 SQL:{0} 沒有符合的記錄!", sql); } } } catch (SQLException e) { LOGGER.warning(e); } return map; }
值得注意該函數(shù)里面:
printRealSql(sql, params); // 打印真實 SQL 的函數(shù)
其參數(shù)一 sql 就是類似 SELECT * FROM table WHERE id = ?
的語句,參數(shù)二 params 為 Object... params 的參數(shù)列表,可以是任意類似的合法 SQL 值。最后,通過 printRealSql 函數(shù)最終得出形如 SELECT * FROM table WHERE id = 12
的結果。
printRealSql 函數(shù)源碼如下:
/** * 在開發(fā)過程,SQL語句有可能寫錯,如果能把運行時出錯的 SQL 語句直接打印出來,那對排錯非常方便,因為其可以直接拷貝到數(shù)據(jù)庫客戶端進行調試。 * * @param sql * SQL 語句,可以帶有 ? 的占位符 * @param params * 插入到 SQL 中的參數(shù),可單個可多個可不填 * @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 語句中的占位符與參數(shù)個數(shù)不匹配。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 < cols; i++) { Object value = values[i]; if (value instanceof Date) { values[i] = "'" + value + "'"; } else if (value instanceof String) { values[i] = "'" + value + "'"; } else if (value instanceof Boolean) { values[i] = (Boolean) value ? 1 : 0; } } String statement = String.format(sql.replaceAll("\\?", "%s"), values); LOGGER.info("The SQL is------------>\n" + statement); ConnectionMgr.addSql(statement); // 用來保存日志 return statement; } /** * ? 和參數(shù)的實際個數(shù)是否匹配 * * @param sql * SQL 語句,可以帶有 ? 的占位符 * @param params * 插入到 SQL 中的參數(shù),可單個可多個可不填 * @return true 表示為 ? 和參數(shù)的實際個數(shù)匹配 */ private static boolean match(String sql, Object[] params) { if(params == null || params.length == 0) return true; // 沒有參數(shù),完整輸出 Matcher m = Pattern.compile("(\\?)").matcher(sql); int count = 0; while (m.find()) { count++; } return count == params.length; }
可見,上述思路是非常簡單的,——有多少個 ? 占位符,就要求有多少個參數(shù),然后一一對照填入(數(shù)組)。match 函數(shù)會檢查第一個步驟,檢查個數(shù)是否匹配,否則會返回“SQL 語句中的占位符與參數(shù)個數(shù)不匹配”的提示;然后,參數(shù)的值會被轉換為符合 SQL 值所要求的類型;最后,就是將 SQL 一一填入,——此處使用了一個字符串的技巧,先把 ? 字符通通轉換為 %s,——那是 String.format 可識別的占位符,如此再傳入 Object[] 參數(shù)列表,即可得出我們期待的 SQL 結果。
我們不能保證那 SQL 可以直接放到數(shù)據(jù)庫中被解析。因為我們的初衷只是把 SQL 打印出來,務求更近一步讓程序員在開發(fā)階段看到 SQL 是怎么樣子的,而且不是一堆 ?、?……,這樣會顯得更符合真實情形一點。
PrepareStatement 內部源碼肯定有這一步驟或者某個變量是表示那個真實 SQL 的,——只是沒有暴露出來。如果有,那么對程序員會更友好一些。
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作能帶來一定的幫助,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。