1、CLOG/BLOG的插入
?找了很多OCI的例子以及一些PHP的例子,基本上都是同一種做法:
?a、OCIStmtPrepare預處理SQL語句:"insert into mytable (myimg) values (empty_blob()) returning myimg into :pImg"
?b、調用OCIDescriptorAlloc,用參數OCI_DTYPE_LOB,獲得一個LOB的句柄
?c、用OCIStmtBindByPos或者OCIStmtBindByName,把LOB句柄和參數:pImg綁定
?d、OCIStmtExecute執行
?e、OCILobWrite往LOB里面寫數據
?f、commit和釋放資源等等
?
?問題是這個SQL語句也太特別了,和其他數據庫的SQL不一樣。但是在java的實現中,并不需要這么奇怪的語法。
?insert into mytable (myimg) values(?)
?就足夠了。
?在我們的應用中,能夠統一同一種SQL的寫法當然是最好的,最好就是:
?insert into mytable (myimg) values (:pImg)?那就和其他的數據處理一致了。
?
?做了很多的嘗試,終于找到解決的方法:
?a、OCIStmtPrepare預處理SQL語句:"insert into mytable (myimg) values (:pImg)"
?b、調用OCIDescriptorAlloc,用參數OCI_DTYPE_LOB,獲得一個LOB的句柄
?c、通過OCILobCreateTemporary函數,為LOB句柄創建臨時LOB對象
?d、OCILobWriteAppend往LOB句柄寫數據
?e、用OCIStmtBindByPos或者OCIStmtBindByName,把LOB句柄和參數:pImg綁定
?f、OCIStmtExecute執行
?g、commit和釋放資源等等
?
?經過簡單的封裝,LOB操作簡單很多了:
?TDBConnection conn = TDatabase::getConnection( strConnection );
?TDBStatement dbStatement(conn.getStatement());
?dbStatement.prepareSQL( "insert into mytable (myimg) values (:pImg)" );
?dbStatement.getParam("pImg").attachInputStream( fStream );
?dbStatement.execSQL();
?conn.commite();
?
2、空字符串
?插入一個空字符串,OCIBindByPos/OCIBindByName的時候,那個字符串的數據類型,不能是SQLT_LVC,用SQLT_CHR倒是沒有問題。
?具體為什么,只有Oracle自己知道了。
?
3、指定IP連接Oracle
?通過OCIServerAttach里面指定綁定的SID連接數據庫。用SID連接Oracle當然有它的好處。奇怪的是JDBC能用IP和Port指定連接哪個Oracle。
?找遍OCI的幫助都找不到IP連接方法。google了很久,找到解決的方法,自己生成SID:
?string strSID = "(DESCRIPTION=(ADDRESS_LIST=(ADDRESS=(PROTOCOL=TCP)(HOST=%1)(PORT=%2)))(CONNECT_DATA=(SERVICE_NAME=%3)))";
?里面的%1設為Oracle服務器的IP;%2設為Oracle服務器的Port,一般是1521;%3就是database的名字。
?把這個字符串放入OCIServerAttach
?