###前提・実現したいこと
SpringBatch+iBatisを使用し、OracleのXMLTYPE型にINSERT/UPDATE/SELECTを行いたいと思っております。
容量の小さなXMLファイルをStringとして読み込み値をINSERTした場合は、正常に登録されますが、
ある容量以上のファイルになると下記エラーが発生します。
###発生している問題・エラーメッセージ
org.springframework.jdbc.UncategorizedSQLException: SqlMapClient operation; uncategorized SQLException for SQL []; SQL state [72000]; error code [1461]; ORA-01461: LONG値はLONG列にのみバインドできます。 ; nested exception is java.sql.BatchUpdateException: ORA-01461: LONG値はLONG列にのみバインドできます。 at org.springframework.jdbc.support.AbstractFallbackSQLExceptionTranslator.translate(AbstractFallbackSQLExceptionTranslator.java:83) at org.springframework.jdbc.support.AbstractFallbackSQLExceptionTranslator.translate(AbstractFallbackSQLExceptionTranslator.java:80) at org.springframework.jdbc.support.AbstractFallbackSQLExceptionTranslator.translate(AbstractFallbackSQLExceptionTranslator.java:80) at org.springframework.orm.ibatis.SqlMapClientTemplate.execute(SqlMapClientTemplate.java:212) at org.springframework.batch.item.database.IbatisBatchItemWriter.write(IbatisBatchItemWriter.java:122) at org.springframework.batch.item.support.CompositeItemWriter.write(CompositeItemWriter.java:50) at org.springframework.batch.core.step.item.SimpleChunkProcessor.writeItems(SimpleChunkProcessor.java:171) at org.springframework.batch.core.step.item.SimpleChunkProcessor.doWrite(SimpleChunkProcessor.java:150) at org.springframework.batch.core.step.item.SimpleChunkProcessor.write(SimpleChunkProcessor.java:268) at org.springframework.batch.core.step.item.SimpleChunkProcessor.process(SimpleChunkProcessor.java:194) at org.springframework.batch.core.step.item.ChunkOrientedTasklet.execute(ChunkOrientedTasklet.java:74) at org.springframework.batch.core.step.tasklet.TaskletStep$ChunkTransactionCallback.doInTransaction(TaskletStep.java:371) at org.springframework.transaction.support.TransactionTemplate.execute(TransactionTemplate.java:128) at org.springframework.batch.core.step.tasklet.TaskletStep$2.doInChunkContext(TaskletStep.java:262) at org.springframework.batch.core.scope.context.StepContextRepeatCallback.doInIteration(StepContextRepeatCallback.java:76) at org.springframework.batch.repeat.support.RepeatTemplate.getNextResult(RepeatTemplate.java:367) at org.springframework.batch.repeat.support.RepeatTemplate.executeInternal(RepeatTemplate.java:214) at org.springframework.batch.repeat.support.RepeatTemplate.iterate(RepeatTemplate.java:143) at org.springframework.batch.core.step.tasklet.TaskletStep.doExecute(TaskletStep.java:248) at org.springframework.batch.core.step.AbstractStep.execute(AbstractStep.java:195) at org.springframework.batch.core.job.SimpleStepHandler.handleStep(SimpleStepHandler.java:135) at org.springframework.batch.core.job.flow.JobFlowExecutor.executeStep(JobFlowExecutor.java:61) at org.springframework.batch.core.job.flow.support.state.StepState.handle(StepState.java:60) at org.springframework.batch.core.job.flow.support.SimpleFlow.resume(SimpleFlow.java:144) at org.springframework.batch.core.job.flow.support.SimpleFlow.start(SimpleFlow.java:124) at org.springframework.batch.core.job.flow.FlowJob.doExecute(FlowJob.java:135) at org.springframework.batch.core.job.AbstractJob.execute(AbstractJob.java:281) at org.springframework.batch.core.launch.support.SimpleJobLauncher$1.run(SimpleJobLauncher.java:120) at org.springframework.core.task.SyncTaskExecutor.execute(SyncTaskExecutor.java:49) at org.springframework.batch.core.launch.support.SimpleJobLauncher.run(SimpleJobLauncher.java:114) at org.springframework.batch.test.JobLauncherTestUtils.launchJob(JobLauncherTestUtils.java:151) at org.springframework.batch.test.JobLauncherTestUtils.launchJob(JobLauncherTestUtils.java:140) at com.hoge.foo.JobTest.testLaunchJob(JobTest.java:138) 中略 Caused by: java.sql.BatchUpdateException: ORA-01461: LONG値はLONG列にのみバインドできます。 at oracle.jdbc.driver.OraclePreparedStatement.executeBatch(OraclePreparedStatement.java:15000) at oracle.jdbc.driver.OracleStatementWrapper.executeBatch(OracleStatementWrapper.java:591) at org.apache.commons.dbcp.DelegatingStatement.executeBatch(DelegatingStatement.java:297) at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39) at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25) at java.lang.reflect.Method.invoke(Method.java:597) at com.ibatis.common.jdbc.logging.PreparedStatementLogProxy.invoke(PreparedStatementLogProxy.java:88) at $Proxy14.executeBatch(Unknown Source) at com.ibatis.sqlmap.engine.execution.SqlExecutor$Batch.executeBatchDetailed(SqlExecutor.java:642) at com.ibatis.sqlmap.engine.execution.SqlExecutor.executeBatchDetailed(SqlExecutor.java:142) at com.ibatis.sqlmap.engine.impl.SqlMapExecutorDelegate.executeBatchDetailed(SqlMapExecutorDelegate.java:775) at com.ibatis.sqlmap.engine.impl.SqlMapSessionImpl.executeBatchDetailed(SqlMapSessionImpl.java:188) at org.springframework.batch.item.database.IbatisBatchItemWriter$1.doInSqlMapClient(IbatisBatchItemWriter.java:131) at org.springframework.orm.ibatis.SqlMapClientTemplate.execute(SqlMapClientTemplate.java:209) ... 52 more
###該当のソースコード
Job定義は
ReaderにてXMLファイルの一覧を取得し、Readで1件ずつファイル名をProcessorに引き渡し、
ProcessorにてXMLファイルをString文字列でAAAAクラスに格納、
Writer(org.springframework.batch.item.database.IbatisBatchItemWriter)を使用し、格納となっています。
Java
1package com.hoge.foo; 2public class AAAA { 3 private Integer id; 4 private String xmldata; 5 //各セッター・ゲッター 6}
XML
1SqlMap.xml 2<sqlMap namespace="SqlMap_HOGE_TABLE"> 3 <insert id="insertForXMLTYPE" parameterClass="com.hoge.foo.AAAA"> 4 INSERT INTO HOGE_TABLE (ID,XML_DATA) VALUES(#id:DECIMAL#,#xmldata:OTHER#) 5 </insert> 6</sqlMap> 7 8SqlMapConfig.xml 9<sqlMapConfig> 10 <settings useStatementNamespaces="true"/> 11 <sqlMap resource="sqlmaps/SqlMap.xml"/> 12</sqlMapConfig>
###試したこと
ファイル自体の容量を小さくした場合に、正常にINSERTされることを確認しました。
org.springframework.jdbc.support.xml.Jdbc4SqlXmlHandlerをどうにかして、利用できれば、
と試してみましたが、組み込み方法がよくわからず、下記のように記載しましたが、エラーとなりました。
XML
1SqlMapConfig.xml 2<sqlMapConfig> 3 <settings useStatementNamespaces="true"/> 4 <sqlMap resource="sqlmaps/SqlMap.xml"/> 5 <typeHandler javaType="java.lang.String" callback="Jdbc4SqlXmlHandler"/> 6</sqlMapConfig>
また、PrepareStatementを使用し、SQLXML型でINSERTができるのではないかと考え、
下記の実装を行ってみましたが、エラーが発生しました。
Java
1 ・フィールドにてデータソースを定義し、@Autowiredで注入 2 private OracleDataSource ds; 3 4 ・処理メソッド 5 AAAA aaaa = new AAAA(); 6 OracleConnection conn = null; 7 OraclePreparedStatement pre= null; 8 try { 9 conn = (OracleConnection) ds.getConnection(); 10 StringBuilder sql = new StringBuilder(); 11 sql.append("INSERT INTO HOGE_TABLE (ID,XML_DATA) VALUES (?,?)"); 12 pre = (OraclePreparedStatement) conn.prepareStatement(sql.toString()); 13 int parameterIndex = 1; 14 pre.setString(parameterIndex++, aaaa.getId()); 15 SQLXML xmlObject = conn.createSQLXML(); 16 xmlObject.setString(aaaa.getXmlData()); 17 pre.setSQLXML(parameterIndex++, xmlObject); 18 19 pre.addBatch(); 20 pre.executeUpdate(); 21 22 } catch (Exception e) { 23 e.printStackTrace(); 24 }finally{ 25 SFDBUtils.closeQuietly(conn, pre, null); 26 }
XML
1データソースに@Autowiredを行う設定 2<bean id="test.XMLDB" class="oracle.jdbc.pool.OracleDataSource"> 3 <property name="URL" value="${XMLDB.jdbc.url}" /> 4</bean> 5また、このデータソースでも同様に試しましたが、エラーが発生するのは変わらず。 6<bean id="test.XMLDB" class="org.apache.commons.dbcp.BasicDataSource"> 7 <property name="driverClassName" value="oracle.jdbc.OracleDriver" /> 8 <property name="url" value="${XMLDB.url}" /> 9 <property name="username" value="${XMLDB.username}" /> 10 <property name="password" value="${XMLDB.password}" /> 11</bean>
Java
1発生したエラー 2java.sql.SQLException: サポートされない機能です。 3 at oracle.jdbc.driver.PhysicalConnection.createSQLXML(PhysicalConnection.java:17745) 4 at com.hoge.foo.JobTest.testLaunchJob(JobTest.java:102) 5 //省略
###補足情報(言語/FW/ツール等のバージョンなど)
- Java
java version "1.6.0_24"
Java(TM) SE Runtime Environment (build 1.6.0_24-b07)
Java HotSpot(TM) 64-Bit Server VM (build 19.1-b02, mixed mode)
- Spring Batch
spring-batch-2.1.6
- Spring Beans
spring-beans-2.5
- Oracle
Oracle Database 11g Enterprise Edition Release 11.2.0.1.0 - 64bit Production
PL/SQL Release 11.2.0.1.0 - Production
CORE 11.2.0.1.0 Production
TNS for Linux: Version 11.2.0.1.0 - Production
NLSRTL Version 11.2.0.1.0 - Production

バッドをするには、ログインかつ
こちらの条件を満たす必要があります。
2016/10/31 08:19