Oracle 11g (Red Hat 上) を使用しています。XMLType 列を持つ単純な通常の表があります。
CREATE TABLE PROJECTS
(
PROJECT_ID NUMBER(*, 0) NOT NULL,
PROJECT SYS.XMLTYPE,
);
Oracle SQL Developer (Windows 上) を使用して、次のことを行います。
select T1.PROJECT P1 from PROJECTS T1 where PROJECT_ID = '161';
できます。セルを1つ取得します。ダブルクリックして、XML ファイル全体をダウンロードできます。
次に、結果を CLOB として取得しようとしました。
select T1.PROJECT.getClobVal() P1 from PROJECTS T1 where PROJECT_ID = '161';
できます。セルを1つ取得します。ダブルクリックしてテキスト全体を表示し、コピーできます。しかし問題がある。クリップボードにコピーすると、最初の 4000 文字しか取得できません。4000 の位置に 0x00 文字があり、残りの CLOB はコピーされていないようです。
これを確認するために、java でチェックを書きました。
// ... create projectsStatement
Reader reader = projectsStatement.getResultSet().getCharacterStream( "P1" );
BufferedReader bf = new BufferedReader( reader );
char buffer[] = new char[ 1024 ];
int count = 0;
int globalPos = 0;
while ( ( count = bf.read( buffer, 0, buffer.length ) ) > 0 )
for ( int i = 0; i < count; i++, globalPos++ )
if ( buffer[ i ] == 0 )
throw new Exception( "ZERO at " + Integer.toString(globalPos) );
Reader は完全な XML を返しますが、位置 4000 にヌル文字があるため、例外がスローされます。この 1 バイトを削除することもできますが、これはかなり奇妙な回避策になります。
そこでは VARCHAR2 を使用していませんが、この問題は VARCHAR2 の制限 (4000 バイト) に関連している可能性があります。他のアイデアはありますか?これは Oracle のバグですか、それとも何か不足していますか?
- - - - - - - - - - 編集 - - - - - - - - - -
次のストアド プロシージャを使用して値が挿入されました。
create or replace
procedure addProject( projectId number, projectXml clob ) is
sqlstr varchar2(2000);
begin
sqlstr := 'insert into projects ( PROJECT_ID, PROJECT ) VALUES ( :projectId, :projectData )';
execute immediate sqlstr using projectId, XMLTYPE(projectXml);
end;
それを呼び出すために使用される Java コード:
try ( CallableStatement cs = connection.prepareCall("{call addProject(?,?)}") )
{
cs.setInt( "projectId", projectId );
cs.setCharacterStream( "projectXml", new StringReader(xmlStr) , xmlStr.length() );
cs.execute();
}
- - - - - - - - - - 編集。簡単なテスト --------------------
あなたの答えから学んだすべてを使用します。最も単純なテーブルを作成します。
create table T1 ( P XMLTYPE );
XML を含む 2 つの CLOB を準備します。1 つ目はヌル文字あり、2 つ目はなしです。
declare
P1 clob;
P2 clob;
P3 clob;
begin
P1 := '<a>';
P2 := '<a>';
FOR i IN 1..1000 LOOP
P1 := P1 || '0123456789' || chr(0);
P2 := P2 || '0123456789';
END LOOP;
P1 := P1 || '</a>';
P2 := P2 || '</a>';
null が最初の CLOB にあり、2 番目の CLOB にないかどうかを確認します。
DBMS_OUTPUT.put_line( DBMS_LOB.INSTR( P1, chr(0) ) );
DBMS_OUTPUT.put_line( DBMS_LOB.INSTR( P2, chr(0) ) );
期待どおりに取得します:
14
0
最初の CLOB を XMLTYPE に挿入してみてください。効果がないでしょう。そのような値を挿入することはできません:
insert into T1 ( P ) values ( XMLTYPE( P1 ) );
2 番目の CLOB を XMLTYPE に挿入してみてください。それが動作します:
insert into T1 ( P ) values ( XMLTYPE( P2 ) );
挿入された XML を 3 番目の CLOB に読み取ろうとします。それが動作します:
select T.P.getClobVal() into P3 from T1 T where rownum = 1;
null があるかどうかを確認します。null はありません:
DBMS_OUTPUT.put_line( DBMS_LOB.INSTR( P3, chr(0) ) );
データベース内に null はなく、PL/SQL コンテキストにいる限り、null はありません。しかし、SQL Developer (Windows の場合) または Java (Red Hat EE および Tomcat7 の場合) で次の SQL を使用しようとすると、返されたすべての CLOB の位置 4000 に null 文字が表示されます。
select T.P.getClobVal() from T1 T;
BR、JM