From 79168b7708a0833ce5fc427fc2f6e15553dba116 Mon Sep 17 00:00:00 2001 From: obelisk0114 Date: Mon, 2 Jul 2018 21:50:41 -0700 Subject: [PATCH] Change some style and adjust space --- docs/CH20.md | 32 ++++++++++++++++---------------- docs/CH21.md | 46 ++++++++++++++++++++++++++++------------------ 2 files changed, 44 insertions(+), 34 deletions(-) diff --git a/docs/CH20.md b/docs/CH20.md index e415d08..ba162f9 100644 --- a/docs/CH20.md +++ b/docs/CH20.md @@ -10,9 +10,9 @@ JDBC(Java DataBase Connectivity)是用於執行 SQL 的 Java 解決方案, 在正式使用 JDBC 進行資料庫操作之前,先來認識一下 JDBC 的基本架構,了解資料庫驅動程式與資料庫之間的關係,在這個小節也將看到,如何設計一個簡單的工具類別,讓您在進行資料庫連接(Connection)時更為方便。 -20.1.1 簡介 JDBC +### 20.1.1 簡介 JDBC -如果要連接資料庫並進行操作,基本上必須了解所使用的資料庫所提供的 API 操作介面,然而各個廠商所提供的 API 操作介面並不一致,如果今天要使用A廠商的資料庫,就必須設計一個專用的程式來操作 A 廠商資料庫所提供的 API,將來如果要使用 B 廠商的資料庫,即使目的相同,也是要撰寫專用於 B 廠商資料庫之程式,十分的不方便。 +如果要連接資料庫並進行操作,基本上必須了解所使用的資料庫所提供的 API 操作介面,然而各個廠商所提供的 API 操作介面並不一致,如果今天要使用 A 廠商的資料庫,就必須設計一個專用的程式來操作 A 廠商資料庫所提供的 API,將來如果要使用 B 廠商的資料庫,即使目的相同,也是要撰寫專用於 B 廠商資料庫之程式,十分的不方便。 使用 JDBC,可由廠商實作操作資料庫介面的驅動程式,而 Java 程式設計人員呼叫 JDBC 的 API 並操作 SQL,實際對資料庫的操作由 JDBC 驅動程式負責,如果要更換資料庫,基本上只要更換驅動程式,Java 程式中只要載入新的驅動程式來源即可完成資料庫的變更,Java 程式的部份則無需改變。 @@ -62,7 +62,7 @@ JDBC 資料庫驅動程式依實作方式可以分為四個類型: ### 20.1.2 連接資料庫 -為了要連接資料庫系統,您必須要有JDBC驅動程式,由於接下來將使用 MySQL 資料庫進行操作,所以請將下載回來的tar.gz檔案使用解壓縮軟體解開,並將當中的 mysql-connector-java-*.jar 加入至 Classpath 的設定之中,假設是放在 c:\workspace\library\mysql-connector-java-3.1.13-bin.jar,則 Classpath 中必須有 c:\workspace\library\mysql-connector-java-3.1.13-bin.jar 這個路徑設定。 +為了要連接資料庫系統,您必須要有JDBC驅動程式,由於接下來將使用 MySQL 資料庫進行操作,所以請將下載回來的 tar.gz 檔案使用解壓縮軟體解開,並將當中的 mysql-connector-java-*.jar 加入至 Classpath 的設定之中,假設是放在 c:\workspace\library\mysql-connector-java-3.1.13-bin.jar,則 Classpath 中必須有 c:\workspace\library\mysql-connector-java-3.1.13-bin.jar 這個路徑設定。 在 Java SE 中與資料庫操作相關的 JDBC 類別都位於 java.sql 套件中,要連接資料庫,基本上必須有幾個動作: @@ -83,7 +83,7 @@ JDBC 資料庫驅動程式依實作方式可以分為四個類型: 協定:子協定:資料來源識別 - 「協定」在 JDBC 中總是 jdbc 開始;「子協定」是橋接的驅動程式或是資料庫管理系統名稱,使用 MySQL 的話是 "mysql";「資料來源識別」標出找出資料庫來源的位址與連接埠。舉個例子來說,MySQL 的 JDBC URL 撰寫方式如下: + 「協定」在 JDBC 中總是 jdbc 開始;「子協定」是橋接的驅動程式或是資料庫管理系統名稱,使用 MySQL 的話是 `mysql`;「資料來源識別」標出找出資料庫來源的位址與連接埠。舉個例子來說,MySQL 的 JDBC URL 撰寫方式如下: jdbc:mysql://主機名稱:連接埠/資料庫名稱?參數=值&參數=值 @@ -126,7 +126,7 @@ getConnection() 方法可以在參數上指定使用者名稱與密碼,例如 您可以將 JDBC URL、使用者名稱與密碼等設定資訊,撰寫在一個屬性檔案當中,由程式讀取這個屬性檔中的資訊,如果需要變更資訊,則只要修改屬性檔即可,無須修改程式、重新編譯,在 Java SE 當中,屬性檔的讀取可以交給 java.util.Properties 類別。 -舉個實際的例子,假設您使用了以下的指令在MySQL後建立了demo資料庫: +舉個實際的例子,假設您使用了以下的指令在 MySQL 後建立了 demo 資料庫: CREATE DATABASE demo; @@ -192,7 +192,7 @@ public class SimpleDBSource implements DBSource { } ``` -預設的建構方法設定中,是讀取 jdbc.properties 檔案中的設定,如果打算自行指定屬性檔案名稱,則可以使用另一個有參數的建構方法。Properties 的 getProperty() 方法會讀取屬性檔案中的"鍵(Key)"對應的"值(Value)",假設您的屬性檔案設定如下: +預設的建構方法設定中,是讀取 jdbc.properties 檔案中的設定,如果打算自行指定屬性檔案名稱,則可以使用另一個有參數的建構方法。Properties 的 getProperty() 方法會讀取屬性檔案中的 **鍵(Key)** 對應的 **值(Value)**,假設您的屬性檔案設定如下: #### **範例 20.3 jdbc.properties** ``` @@ -376,7 +376,7 @@ onlyfun.caterpillar.poolmax=10 Statement stmt = conn.createStatement(); -取得 Statement 物件之後,可以使用 executeUpdate()、executeQuery() 等方法來執行 SQL,executeUpdate() 主要是用來執行 CREATE TABLE、INSERT、DROP TABLE、ALTER TABLE 等會改變資料庫內容的 SQL,例如可以在 demo 資料庫中建立一個 t_message 表格: +取得 Statement 物件之後,可以使用 executeUpdate()、executeQuery() 等方法來執行 SQL,executeUpdate() 主要是用來執行 CREATE TABLE、INSERT、DROP TABLE、ALTER TABLE 等會改變資料庫內容的 SQL,例如可以在 demo 資料庫中建立一個 t_message 表格: Use demo; CREATE TABLE t_message ( @@ -413,7 +413,7 @@ Statement 的 executeQuery() 方法則是用於 SELECT 等查詢資料庫的 SQL System.out.print(result.getString(4) + "\t"); } -Statement 的 execute() 可以用來執行 SQL,並可以測試所執行的 SQL 是執行查詢或是更新,傳回 true 的話表示 SQL 執行將傳回 ResultSet 表示查詢結果,此時可以使用 getResultSet() 取得 ResultSet 物件,如果 execute() 傳回 false,表示 SQL 執行會傳回更新筆數或沒有結果,此時可以使用 getUpdateCount() 取得更新筆數。如果事先無法得知是進行查詢或是更新,就可以使用 execute()。 +Statement 的 execute() 可以用來執行 SQL,並可以測試所執行的 SQL 是執行查詢或是更新,傳回 true 的話表示 SQL 執行將傳回 ResultSet 表示查詢結果,此時可以使用 getResultSet() 取得 ResultSet 物件,如果 execute() 傳回 false,表示 SQL 執行會傳回更新筆數或沒有結果,此時可以使用 getUpdateCount() 取得更新筆數。如果事先無法得知是進行查詢或是更新,就可以使用 execute()。 範例 20.8 是個示範新增與查詢資料的範例,當中使用了前一節設計的 SimpleDBSource。注意在查詢結束後,要使用 Statement 的 close() 方法來釋放 Statement 的資源,而最後不使用連接時,也使用了 closeConnection() 來關閉連接。 @@ -580,7 +580,7 @@ public class PreparedStatementDemo { } ``` -setXXX() 方法的第一個參數指定"?"的位置,而第二個參數為要新增至資料表欄位的值,要讓 SQL 執行生效,要執行 executeQuery() 或 executeUpdate() 方法,使用 setXXX() 來設定的參數會一直有效,可以於下一次使用,如果想要清除設定好的參數,可以執行 clearParameters() 方法。以下是這個範例的執行結果參考: +setXXX() 方法的第一個參數指定 "?" 的位置,而第二個參數為要新增至資料表欄位的值,要讓 SQL 執行生效,要執行 executeQuery() 或 executeUpdate() 方法,使用 setXXX() 來設定的參數會一直有效,可以於下一次使用,如果想要清除設定好的參數,可以執行 clearParameters() 方法。以下是這個範例的執行結果參考: 1 justin justin@mail.com mesage... 2 momor momor@mail.com message2... @@ -612,7 +612,7 @@ fin.close(); 如果要從資料庫中取得 BLOB 或 CLOB 資料,您可以如下進行,其中 result 參考一個 ResultSet 的實例: Blob blob = result.getBlob(2); // 取得BLOB - Clob clob = result.getClob(2) // 取得CLOB + Clob clob = result.getClob(2); // 取得CLOB Blob 擁有 getBinaryStream()、getBytes() 等方法,可以取得二進位串流或 byte 等資料,同樣的,Clob 擁有 getCharacterStream()、getSubString() 等方法,可以取得字元串流或子字串等資料,您可以查看 API 文件來獲得更詳細的訊息。 @@ -736,9 +736,9 @@ public class LobDemo { 交易是一組原子(Atomic)操作(一組 SQL 執行)的工作單元,這個工作單元中的所有原子操作在進行期間,與其它交易隔離,免於數據來源的交相更新而發生混亂,交易中的所有原子操作,要嘛全部執行成功,要嘛全部失敗(即使只有一個失敗,所有的原子操作也要全部撤消)。 -舉個簡單的例子,一個客戶從 A 銀行轉帳至 B 銀行,要做的動作為從A銀行的帳戶扣款、在 B 銀行的帳戶加上轉帳的金額,兩個動作必須成功,如果有一個動作失敗,則此次轉帳失敗。 +舉個簡單的例子,一個客戶從 A 銀行轉帳至 B 銀行,要做的動作為從 A 銀行的帳戶扣款、在 B 銀行的帳戶加上轉帳的金額,兩個動作必須成功,如果有一個動作失敗,則此次轉帳失敗。 -在 JDB C中,可以操作 Connection 的 setAutoCommit() 方法,給它 false 引數,在下達一連串的 SQL 語句後,自行呼叫 Connection 的 commit() 來送出變更,如果中間發生錯誤,則呼叫 rollback() 來撤消所有的執行,一個示範的流程如下所示: +在 JDBC 中,可以操作 Connection 的 setAutoCommit() 方法,給它 false 引數,在下達一連串的 SQL 語句後,自行呼叫 Connection 的 commit() 來送出變更,如果中間發生錯誤,則呼叫 rollback() 來撤消所有的執行,一個示範的流程如下所示: ```java try { @@ -762,7 +762,7 @@ try { } ``` -如果您在交易管理時,僅想要撤回(rollback)某個SQL執行點,則您可以設定儲存點(save point),例如: +如果您在交易管理時,僅想要撤回(rollback)某個 SQL 執行點,則您可以設定儲存點(save point),例如: ```java conn.setAutoCommit(false); @@ -789,7 +789,7 @@ stmt.releaseSavepoint(savepoint); ### 20.2.6 批次處理 -Statement 的 execute 等方法一次只能執行一個 SQL 敘述,如果有多個 SQL 敘述要執行的話,可以使用 executeBatch() 方法,在一次方法呼叫中執行多個 SQL 敘述,以增加執行的效能,您可以使用 addBatch() 方法將要執行的 SQL 敘述加入,然後執行 executeBatch() 即可: +Statement 的 execute 等方法一次只能執行一個 SQL 敘述,如果有多個 SQL 敘述要執行的話,可以使用 executeBatch() 方法,在一次方法呼叫中執行多個 SQL 敘述,以增加執行的效能,您可以使用 addBatch() 方法將要執行的 SQL 敘述加入,然後執行 executeBatch() 即可: ```java conn.setAutoCommit(false); @@ -802,7 +802,7 @@ stmt.executeBatch(); conn.commit(); ``` -在執行 executeBatch() 時而 SQL 有錯誤的情況下,會丟出 BatchUpdateException 例外,您可以由這個例外物件的 getUpdateCounts() 方法取得發生錯誤的SQL句數,如果中間有個 SQL 執行錯誤,則應該撤回(rollback)整個批次處理過程的SQL操作。 +在執行 executeBatch() 時而 SQL 有錯誤的情況下,會丟出 BatchUpdateException 例外,您可以由這個例外物件的 getUpdateCounts() 方法取得發生錯誤的 SQL 句數,如果中間有個 SQL 執行錯誤,則應該撤回(rollback)整個批次處理過程的 SQL 操作。 使用 PreparedStatement 也可以進行批次處理,直接來看個例子: @@ -947,7 +947,7 @@ relative() 方法則從目前游標處指定相對位置,例如若目前在第 ### 20.2.9 ResultSetMetaData -Meta Data 即「資料的資料」(Data about data),ResultSet 用來表示查詢到的資料,而 ResultSe t資料的資料,即描述所查詢到的資料背後的資料描述,即用來表示表格名稱、欄位名稱、欄位型態等,這些訊息可以透過 ResultSetMetaData 來取得。 +Meta Data 即「資料的資料」(Data about data),ResultSet 用來表示查詢到的資料,而 ResultSet 資料的資料,即描述所查詢到的資料背後的資料描述,即用來表示表格名稱、欄位名稱、欄位型態等,這些訊息可以透過 ResultSetMetaData 來取得。 範例 20.12 直接示範如何取得查詢到的資料欄位數、表格名稱、欄位名稱與欄位資料型態: diff --git a/docs/CH21.md b/docs/CH21.md index 793ebbd..561424e 100644 --- a/docs/CH21.md +++ b/docs/CH21.md @@ -115,7 +115,7 @@ public class BinarySearchDemo { } ``` -在這個程式中,指定搜尋索引 6 到索引 9(不包括索引 9)的範圍中是否有 85 的值,如果找到的話,傳回索引值,如果沒有找到,傳回一個負值,該負值加 1 再乘以 -1,就是插入點位置,也就是第一個比指定搜尋值大的值之索引位置,就這個程式而言,就是顯示 "插入點8" 的結果。 +在這個程式中,指定搜尋索引 6 到索引 9(不包括索引 9)的範圍中是否有 85 的值,如果找到的話,傳回索引值,如果沒有找到,傳回一個負值,該負值加 1 再乘以 -1,就是插入點位置,也就是第一個比指定搜尋值大的值之索引位置,就這個程式而言,就是顯示 "插入點 8" 的結果。 在 18.1.2 使用 Calendar 中,在範例 18.6 中使用 switch 進行判斷以顯示中文的日期格式,在 Java SE 6 中,您可以直接使用 getDisplayNames() 或 getDisplayName() 方法取得區域化的日期格式顯示,例如可以改寫範例 18.6 為以下的程式: @@ -154,7 +154,7 @@ public class CalendarDemo { 23 日 星期四 -在 Java SE 6 中,對於集合物件還增加有 Deque 介面(繼承自Queue介面)、NavigableMap 介面(繼承自 SortedMap 介面)、NavigableSet 介面(繼承自 SortedSet 介面),建議您可以查閱一下 API 文件,以了解這些集合物件的使用方式。 +在 Java SE 6 中,對於集合物件還增加有 Deque 介面(繼承自 Queue 介面)、NavigableMap 介面(繼承自 SortedMap 介面)、NavigableSet 介面(繼承自 SortedSet 介面),建議您可以查閱一下 API 文件,以了解這些集合物件的使用方式。 ### 21.1.3 java.io 套件 @@ -266,7 +266,7 @@ public class FileDemo { 在 Java SE 6 中對於視窗程式設計也作了極大的改進,雖然本書並沒有深入講解 Java 的視窗程式設計,然而在這邊可以介紹幾個有特色且使用簡單的功能。 -有些視窗程式在啟動時,會有個啟動畫面,在 Java SE 6 之前,您要自己實作才可以擁有這個功能,現在您可以直接在使用 "java" 程式執行程式時下達 "-splash" 引數指定啟動畫面的圖片,就可以擁有這個功能,例如若執行 19.5 所製作出來的 Executable Jar 檔時,如下指定圖片: +有些視窗程式在啟動時,會有個啟動畫面,在 Java SE 6 之前,您要自己實作才可以擁有這個功能,現在您可以直接在使用 "java" 程式執行程式時下達 `-splash` 引數指定啟動畫面的圖片,就可以擁有這個功能,例如若執行 19.5 所製作出來的 Executable Jar 檔時,如下指定圖片: java -splash:caterpillar.jpg -jar JNotePad.jar @@ -276,13 +276,13 @@ public class FileDemo { 圖 21.1 指定啟動畫面的執行結果 -您也可以在製作 Executable JAR 檔案時,於 manifest 檔案中指定 "SplashScreen-Image" 為啟動畫面的圖片,並在使用 jar 程式進行包裝時一併包裝圖片,如此啟動 JAR 檔案時,就會自動展現啟動畫面,一個 manifest 檔案的寫法如下所示: +您也可以在製作 Executable JAR 檔案時,於 manifest 檔案中指定 `SplashScreen-Image` 為啟動畫面的圖片,並在使用 jar 程式進行包裝時一併包裝圖片,如此啟動 JAR 檔案時,就會自動展現啟動畫面,一個 manifest 檔案的寫法如下所示: Manifest-Version: 1.0 Main-Class: onlyfun.caterpillar.JNotePad SplashScreen-Image: caterpillar.jpg -如果您對於啟動畫面更進一步的控制感興趣,例如在不同的啟動階段顯示不同的圖片,或者是在啟動圖片上顯示進度列,則可以看看 java.awt. SplashScreen 的 API 文件說明。 +如果您對於啟動畫面更進一步的控制感興趣,例如在不同的啟動階段顯示不同的圖片,或者是在啟動圖片上顯示進度列,則可以看看 java.awt.SplashScreen 的 API 文件說明。 在 Java SE 6 中加入了系統工具列圖示的支援,您可以使用 SystemTray 類別的 isSupported() 方法,測試看看目前的系統是否支援系統工具列圖示,如果支援的話,可以使用 getSystemTray() 取得 SystemTray 實例,使用 add() 方法加入 TrayIcon 實例,如此就可以加入一個系統工具列圖示,例如: @@ -357,8 +357,9 @@ public class SystemTrayDemo2 { 圖 21.3 系統工具列蹦現視窗 -如果要移除系統工具列中的圖示,則可以使用SystemTray實例的remove()方法,指定要移除的圖示,例如: -tray.remove(trayIcon); +如果要移除系統工具列中的圖示,則可以使用 SystemTray 實例的 remove() 方法,指定要移除的圖示,例如: + + tray.remove(trayIcon); ### 21.1.5 Classpath 簡化設定 @@ -380,7 +381,7 @@ Java SE 6 中包含了 JDBC 4.0,對於 JDBC 的使用有了相當的簡化, 目前對於 JDBC 4.0 支援的資料庫驅動程式還不多,在 JDK 6 中綑綁了 [Apache Derby 資料庫](http://db.apache.org/derby/),這是一個純 Java 撰寫的資料庫,支援 JDBC 4.0,您可以在 JDK 6 包裝目錄的 db 目錄下找到 Apache Derby 的相關檔案,也可以至 http://db.apache.org/derby/derby_downloads.html 下載 Apache Derby 資料庫,所下載的檔案中將包括更多的文件等相關資源。 -使用 Apache Derby 資料庫最簡單的方式之一,是使用 NetBeans IDE 來啟動、連接、管理 Derby 資料庫,您可以在 [Sun 官方網站](http://java.sun.com),或是 [NetBeans官方網站](http://www.netbeans.org/) 下載最新的 NetBeans IDE 並進行安裝,在這邊將以 NetBeans IDE 5.5 來示範,如何使用 Derby 資料庫。 +使用 Apache Derby 資料庫最簡單的方式之一,是使用 NetBeans IDE 來啟動、連接、管理 Derby 資料庫,您可以在 [Oracle 官方網站](http://www.oracle.com/technetwork/java/index.html),或是 [NetBeans 官方網站](http://www.netbeans.org/) 下載最新的 NetBeans IDE 並進行安裝,在這邊將以 NetBeans IDE 5.5 來示範,如何使用 Derby 資料庫。 首先,請啟動 NetBeans IDE,執行選單上的「Tools/Options」,接著選擇「Advanced Options」,在「IDE Configuration」中選擇「Server and External Tool Settings」的「Java DB Database」,接著在右邊的「Java DB Location」中設定 Apache Derby 的位置,在這邊是指向「C:\Program Files\Java\jdk1.6.0\db」,而在「Database Location」中設定您的資料庫儲存位置,在這邊是設定為「C:\workspace\database」,如下圖所示: @@ -430,23 +431,32 @@ Java SE 6 中包含了 JDBC 4.0,對於 JDBC 的使用有了相當的簡化, 圖 21.11 查詢表格資料 -關於Apache Derby的介紹先到這邊告一段落,接下來將要介紹,如何撰寫程式連接Apache Derby,並使用JDBC 4.0的新功能。 +關於 Apache Derby 的介紹先到這邊告一段落,接下來將要介紹,如何撰寫程式連接 Apache Derby,並使用 JDBC 4.0 的新功能。 + +### 21.2.2 載入驅動程式 + +在之前的操作中,可以從圖中看到,要連接 Apache Derby,使用的 JDBC URL 是 "jdbc:derby://localhost:1527/demo",而 Apache Derby 的 JDBC 驅動程式是放在 derbyclient.jar 之中,因此請記得在您的 Classpath 之中加以設定。 -21.2.2 載入驅動程式 -在之前的操作中,可以從圖中看到,要連接Apache Derby,使用的JDBC URL是" jdbc:derby://localhost:1527/demo",而Apache Derby的JDBC驅動程式是放在derbyclient.jar之中,因此請記得在您的Classpath之中加以設定。 -在JDBC 4.0之前,如果您要連接資料庫的話,必須使用Class.forName()並指定驅動程式類別名稱,以載入JDBC驅動程式,例如: +在 JDBC 4.0 之前,如果您要連接資料庫的話,必須使用 Class.forName() 並指定驅動程式類別名稱,以載入 JDBC 驅動程式,例如: + +``` String url = …; String username = …; String password = …; String driver = …; Class.forName(driver); Connection conn = DriverManager.getConnection(url, username, password); -在JDBC 4.0之中,不需要再呼叫Class.forName()並指定驅動程式了,也就是說,只要一行就可以了: -Connection conn = DriverManager.getConnection(url, username, password); -那麼JVM如何得知要載入哪個驅動程式呢?JVM會自動在Classpath中尋找適當的驅動程式,在包裝有JDBC驅動程式的JAR檔案中,必須有一個"META-INF/services/java.sql.Driver"檔案,當中撰寫驅動程式類別名稱,以Apache Derby為例,在derbyclient.jar中的"META-INF/services/java.sql.Driver"檔案中,撰寫的是"org.apache.derby.jdbc.ClientDriver "。 -以第20章的範例20.2為例,若使用JDBC 4.0,則可以將Class.forName()該行移除,並在執行時指定Classpath中包括derbyclient.jar的位置,而範例20.4仍然正常運行。 +``` + +在 JDBC 4.0 之中,不需要再呼叫 Class.forName() 並指定驅動程式了,也就是說,只要一行就可以了: + + Connection conn = DriverManager.getConnection(url, username, password); + +那麼 JVM 如何得知要載入哪個驅動程式呢? JVM 會自動在 Classpath 中尋找適當的驅動程式,在包裝有 JDBC 驅動程式的 JAR 檔案中,必須有一個 `META-INF/services/java.sql.Driver` 檔案,當中撰寫驅動程式類別名稱,以 Apache Derby 為例,在 derbyclient.jar 中的 `META-INF/services/java.sql.Driver` 檔案中,撰寫的是 `org.apache.derby.jdbc.ClientDriver`。 + +以第 20 章的範例 20.2 為例,若使用 JDBC 4.0,則可以將 Class.forName() 該行移除,並在執行時指定 Classpath 中包括 derbyclient.jar 的位置,而範例 20.4 仍然正常運行。 -### 21.2.2 改進的例外處理 +### 21.2.3 改進的例外處理 在 JDBC 4.0 之中,SQLException 新增了幾個建構函式,可以接受 Throwable 實例進行 SQLException 的建構,您可以重新將某個例外包裝為 SQLException,例如由於 IOException 發生的 SQLException,這表示您對於 SQLException 所發生的原因更加可以細分與掌握,SQLException 並實作了 `Iterable` 介面,現在您可以在捕捉到 SQLException 時,使用加強的 for 迴圈來找出例外發生的原因,例如: @@ -468,7 +478,7 @@ catch(SQLException ex) { SQLException 在 JDBC 4.0 中多了幾個子類別,針對不同的錯誤將例外加以細分,您可以查看 API 文件中有關 SQLException 的說明。 -### 21.2.3 BLOB、CLOB 的改進 +### 21.2.4 BLOB、CLOB 的改進 在 20.2.3 中介紹過 JDBC 的 LOB 讀寫,當需要存入資料至 BLOB 或 CLOB 欄位時,要使用 PreparedStatement 的 setBinaryStream()、 setObject()、setAsciiStream()、setUnicodeStream() 等方法,在 API 名稱上並不是很明確。