在知識庫的建立的時候,用普通VARCHAR2存放文章是顯然不夠的,只有區區4000的位元組,放不了多少字,
而CLOB資料類型,則能最多存放8G的資料。但是這個欄位處理起來有比較多的特殊性,記錄一下。
插入:
直接寫在SQL裡面是不行的,一來SQL指令碼有字元數限制,而來文章內容包含許多特殊字元,如換行,引號,
之類的東西,很麻煩。網上流行通用做法是先插入一個空CLOB欄位,用empty_clob()方法來建立空欄位,如:
複製代碼 代碼如下:INSERT INTO T_TOPIC(TOPIC_ID,TOPIC_CONTENT) VALUES(‘0000001',empty_clob());
然後再用SELECT TOPIC_CONTENT FROM T_TOPIC WHERE TOPIC_ID='0000001'FOR UPDATE的查詢語句,
來構造一個更新的STATEMENT,在擷取到ResultSet之後,對CLOB欄位進行更新。 複製代碼 代碼如下:ResultSet rs = pstm.executeQuery();
if(rs.next()){
oracle.sql.CLOB lob =(CLOB)rs.getClob(1);
try {
Writer os = lob.getCharacterOutputStream();
os.write(dr.getField("FLD_CONTENT").asString());
os.flush();
} catch (IOException e) {
e.printStackTrace();
}
}
這個插入和更新操作要放在事務中,即擷取到Connection後要設定setAutoCommit(false);
更新:
更新的時候也是採用SEELCT … FOR UPDATE方式
也要設定事務
讀取: 複製代碼 代碼如下:CLOB clob = (CLOB)rs.getClob("FLD_CONTENT");
Reader reader = clob.getCharacterStream();
StringBuffer sb=new StringBuffer();
char[] cb = new char[1024];
try {
for(int len = reader.read(cb);len>0;len= reader.read(cb)){
sb.append(cb,0,len);
}
} catch (IOException e) {
throw new SQLException("讀取文章內容失敗.",e);
}
查詢的特殊性:
有CLOB欄位的資料表,在SQL語句中不能使用DISTINCT關鍵字進行篩選,即便關鍵字不用在CLOB欄位名前,
實際上DISTINCT關鍵字都是對於SQL中所有欄位有效。而CLOB欄位是不能進行如同LIKE類似的匹配的,所以,
不能進行去重複操作。
兩種解決辦法:
1、在SQL中調用方法轉成VARCHAR2欄位後,再DISTINCT,這種方式的局限顯而易見。
2、改變SQL指令碼的書寫方式,先查出沒有CLOB欄位的集合,然後在在外層用EXISTS關鍵字或IN關鍵字進行篩選。複製代碼 代碼如下://String sqlsel2 = "select jsonbody from db_ps_listcatalog where" +
// " listtype ='sh11' for update";
// String col="jsonbody";
public boolean updateClob(String sql,String col,String buf){
boolean flag=false;
Statement stem=null;
Connection conn=null;
ResultSet rs=null;
Writer wr = null;
try{
conn= dp.getConnection();
conn.setAutoCommit(false);
stem=conn.createStatement();
rs = stem.executeQuery(sql);
if (rs.next()) {
CLOB clob = (CLOB) rs.getClob(col);
java.lang.reflect.Method methodToInvoke = clob.getClass().getMethod(
"getCharacterOutputStream", (Class[]) null);
wr = (Writer) methodToInvoke.invoke(clob, (Object[]) null);
BufferedWriter bw = new BufferedWriter(wr);
bw.write(buf);
bw.flush();
bw.close();
conn.commit();
conn.close();
}
flag=true;
} catch (Exception ex){
try {
conn.rollback();
} catch (SQLException e) {
e.printStackTrace();
}
}
return flag;
}