【问题标题】:InvalidDataAccessResourceUsageException: Unexpected cursor position changeInvalidDataAccessResourceUsageException:意外的光标位置更改
【发布时间】:2013-12-04 22:55:42
【问题描述】:

在 Spring Batch 应用程序中尝试从数据库读取时遇到以下异常。任何见解都会有所帮助。这是在使用 JdbcCursorItemReader 时引起的。该应用程序使用 JdbcPagingItemReader 可以正常工作。

还需要在使用 JdbcCursorItemReader 时显式关闭任何资源(游标等)吗?如果是这样,我们该怎么做?

SEVERE: Encountered an error executing the step
    org.springframework.dao.InvalidDataAccessResourceUsageException: Unexpected cursor position change.
        at org.springframework.batch.item.database.AbstractCursorItemReader.verifyCursorPosition(AbstractCursorItemReader.java:365)
        at org.springframework.batch.item.database.AbstractCursorItemReader.doRead(AbstractCursorItemReader.java:449)
        at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
        at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:57)
        at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
        at java.lang.reflect.Method.invoke(Method.java:601)
        at org.springframework.aop.support.AopUtils.invokeJoinpointUsingReflection(AopUtils.java:317)
        at org.springframework.aop.framework.ReflectiveMethodInvocation.invokeJoinpoint(ReflectiveMethodInvocation.java:183)
        at org.springframework.aop.framework.CglibAopProxy$CglibMethodInvocation.invokeJoinpoint(CglibAopProxy.java:695)
        at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:150)
        at org.springframework.aop.support.DelegatingIntroductionInterceptor.doProceed(DelegatingIntroductionInterceptor.java:132)
        at org.springframework.aop.support.DelegatingIntroductionInterceptor.invoke(DelegatingIntroductionInterceptor.java:120)
        at org.springframework.aop.framework.ReflectiveMethodInvocation.proceed(ReflectiveMethodInvocation.java:172)
        at org.springframework.aop.framework.CglibAopProxy$DynamicAdvisedInterceptor.intercept(CglibAopProxy.java:631)
        at com.testapp.springbatchtest.io.DataExtractCursorItemReader$$EnhancerByCGLIB$$748012e7.doRead(<generated>)
        at org.springframework.batch.item.support.AbstractItemCountingItemStreamItemReader.read(AbstractItemCountingItemStreamItemReader.java:83)
        at org.springframework.batch.core.step.item.SimpleChunkProvider.doRead(SimpleChunkProvider.java:91)
        at org.springframework.batch.core.step.item.SimpleChunkProvider.read(SimpleChunkProvider.java:155)
        at org.springframework.batch.core.step.item.SimpleChunkProvider$1.doInIteration(SimpleChunkProvider.java:114)
        at org.springframework.batch.repeat.support.RepeatTemplate.getNextResult(RepeatTemplate.java:368)
        at org.springframework.batch.repeat.support.RepeatTemplate.executeInternal(RepeatTemplate.java:215)
        at org.springframework.batch.repeat.support.RepeatTemplate.iterate(RepeatTemplate.java:144)
        at org.springframework.batch.core.step.item.SimpleChunkProvider.provide(SimpleChunkProvider.java:108)
        at org.springframework.batch.core.step.item.ChunkOrientedTasklet.execute(ChunkOrientedTasklet.java:69)
        at org.springframework.batch.core.step.tasklet.TaskletStep$ChunkTransactionCallback.doInTransaction(TaskletStep.java:395)
        at org.springframework.transaction.support.TransactionTemplate.execute(TransactionTemplate.java:131)
        at org.springframework.batch.core.step.tasklet.TaskletStep$2.doInChunkContext(TaskletStep.java:267)
        at org.springframework.batch.core.scope.context.StepContextRepeatCallback.doInIteration(StepContextRepeatCallback.java:77)
        at org.springframework.batch.repeat.support.RepeatTemplate.getNextResult(RepeatTemplate.java:368)
        at org.springframework.batch.repeat.support.RepeatTemplate.executeInternal(RepeatTemplate.java:215)
        at org.springframework.batch.repeat.support.RepeatTemplate.iterate(RepeatTemplate.java:144)
        at org.springframework.batch.core.step.tasklet.TaskletStep.doExecute(TaskletStep.java:253)
        at org.springframework.batch.core.step.AbstractStep.execute(AbstractStep.java:195)
        at org.springframework.batch.core.job.SimpleStepHandler.handleStep(SimpleStepHandler.java:137)
        at org.springframework.batch.core.job.AbstractJob.handleStep(AbstractJob.java:380)
        at org.springframework.batch.core.job.SimpleJob.doExecute(SimpleJob.java:124)
        at org.springframework.batch.core.job.AbstractJob.execute(AbstractJob.java:301)
        at org.springframework.batch.core.launch.support.SimpleJobLauncher$1.run(SimpleJobLauncher.java:134)
        at org.springframework.core.task.SyncTaskExecutor.execute(SyncTaskExecutor.java:49)
        at org.springframework.batch.core.launch.support.SimpleJobLauncher.run(SimpleJobLauncher.java:127)

以下是 bean 声明的示例代码。 DataExtractCursorItemReader 扩展了 JdbcCursorItemReader。

@Bean
@StepScope
public DataExtractCursorItemReader cursorReaderBean() throws Exception {
    DataExtractCursorItemReader dataExtractReader = new DataExtractCursorItemReader();
    dataExtractReader.setDataSource(dataSource);
    dataExtractReader.setSql("SELECT * FROM SAMPLETABLE");
    dataExtractReader.setFetchSize(500);
    dataExtractReader.setRowMapper(new DataExtractRowMapper());

    return dataExtractReader;
}

【问题讨论】:

    标签: spring exception jdbc spring-batch


    【解决方案1】:

    verifyCursorPosition 属性(似乎)只是一个检查,以防止用户在子类化阅读器时对索引进行操作;也许是您的情况,或者在某些数据库平台属性上不起作用。

    您可以禁用它,一切正常 - 这只是防止子类化的测试。
    如果您修改光标位置,请检查您的代码或检查是否由于数据库驱动程序导致测试失败。

    【讨论】:

    • 你的意思是继承 AbstractCursorItemReader 吗?如问题中所述,DataExtractCursorItemReader 确实扩展了 JdbcCursorItemReader。知道为什么默认设置为 TRUE 吗?
    • IMO 在子类化阅读器时防止用户(不需要的)结果集光标操作。但由于数据库驱动程序不支持的功能也无法工作(尝试设置为false并检查结果)。你需要子类化阅读器吗?
    • 通过设置属性为false来解决。考虑验证是否可以将其设置为 false,以防没有完成光标操作[在代码中]。
    • 我相信多线程也会导致这个问题。您能否在回答中解决这种情况?
    【解决方案2】:

    如果有多个线程作用于数据库读取器并且读取器不是线程安全的,我会看到此错误 通常是 JDBCCursorItemReader 与 JDBCPagingItemReader

    【讨论】:

      【解决方案3】:

      verifyCursorPosition 检查依赖于 java.sql.ResultSet::getRow()。根据JavaDoc

      ,“对于结果集类型为 TYPE_FORWARD_ONLY 的结果集,对 getRow 方法的支持是可选的”

      您可以尝试更改驱动程序、ResultSet (Reader) 的类型,或按照上述建议禁用 verifyCursorPosition

      【讨论】:

        猜你喜欢
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 1970-01-01
        • 2011-04-16
        • 2014-10-28
        • 2018-09-11
        • 1970-01-01
        相关资源
        最近更新 更多