Oracle + dbunit 获取 AmbiguousTableNameException

Posted

技术标签:

【中文标题】Oracle + dbunit 获取 AmbiguousTableNameException【英文标题】:Oracle + dbunit gets AmbiguousTableNameException 【发布时间】:2012-04-23 14:55:16 【问题描述】:

我正在使用 dbunit 创建可以导入和导出的数据库备份。我的应用程序可以使用多种数据库引擎:mysql、PostgreSQL、SQLServer、H2 和 Oracle。

以上所有功能都可以使用以下代码:

            // Connect to the database 
        conn =BackupManager.getInstance().getConnection();
        IDatabaseConnection connection = new DatabaseConnection(conn);
        InputSource xmlSource = new InputSource(new FileInputStream(new File(nameXML)));
        FlatXmlProducer flatXmlProducer = new FlatXmlProducer(xmlSource);
        flatXmlProducer.setColumnSensing(true);

        DatabaseOperation.CLEAN_INSERT.execute(connection,new FlatXmlDataSet(flatXmlProducer));  

但是在 Oracle 上我得到了这个异常:

!ENTRY es.giro.girlabel.backup 1 0 2012-04-11 11:51:40.542
!MESSAGE Start import backup
org.dbunit.database.AmbiguousTableNameException: AQ$_SCHEDULES
    at org.dbunit.dataset.OrderedTableNameMap.add(OrderedTableNameMap.java:198)
    at org.dbunit.database.DatabaseDataSet.initialize(DatabaseDataSet.java:231)
    at org.dbunit.database.DatabaseDataSet.getTableMetaData(DatabaseDataSet.java:281)
    at org.dbunit.operation.DeleteAllOperation.execute(DeleteAllOperation.java:109)
    at org.dbunit.operation.CompositeOperation.execute(CompositeOperation.java:79)
    at es.giro.girlabel.backup.ImportBackup.createData(ImportBackup.java:39)
    at es.giro.girlabel.backup.handlers.Import.execute(Import.java:45)

【问题讨论】:

【参考方案1】:

来自docs:

public class AmbiguousTableNameException extends DataSetException

IDataSet 在多个表具有 可以访问相同的名称。这通常发生在数据库 连接可以访问包含相同表的多个模式 名字。

可能的解决方案:

1) 使用具有 只能访问一个数据库架构。

2) 为 DatabaseConnectionDatabaseDataSourceConnection 构造函数。

3) 启用限定表名支持(请参阅操作指南文档)。

【讨论】:

谢谢,这个问题我已经解决了,不过是你说的解决方案1。 我你用的是spring-dbunit,方案1)可以通过设置系统属性spring.dbunit.schema来实现。 我通过将DatabaseConfigBean#setQualifiedTableNames 设置为true 来修复它。 这篇文章更清楚地解释了解决方案:***.com/a/14932264/3396045。对于 Spring JDBC,我必须这样做:***.com/a/64576554/3396045【参考方案2】:

为谁使用 SpringDBUnit。我一直在努力解决这个非常烦人的问题。我最终通过添加com.github.springtestdbunit.bean.DatabaseConfigBeancom.github.springtestdbunit.bean.DatabaseDataSourceConnectionFactoryBean 的配置解决了这个问题。

这是 SpringDBUnit 的完整 Spring 上下文

<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource"
        destroy-method="close">
        <property name="driverClassName" value="oracle.jdbc.driver.OracleDriver" />
        <property name="url" value="jdbc:oracle:thin:@localhost:1521/XE" />
        <property name="username" value="xxxx" />
        <property name="password" value="xxxx" />
    </bean>


    <bean id="sessionFactory"
        class="org.springframework.orm.hibernate3.annotation.AnnotationSessionFactoryBean">
        <property name="dataSource">
            <ref bean="dataSource" />
        </property>
        <property name="hibernateProperties">
            <props>
                <prop key="hibernate.dialect">org.hibernate.dialect.OracleDialect</prop>
                <prop key="hibernate.show_sql">true</prop>
            </props>
        </property>
        <property name="annotatedClasses">
            <list>
                <value>xxx.example.domain.Person</value>
            </list>
        </property>
    </bean>

    <bean id="dbUnitDatabaseConfig" class="com.github.springtestdbunit.bean.DatabaseConfigBean">
        <property name="skipOracleRecyclebinTables" value="true" />
        <property name="qualifiedTableNames" value="true" />
        <!-- <property name="caseSensitiveTableNames" value="true"/> -->
    </bean>
    <bean id="dbUnitDatabaseConnection"
        class="com.github.springtestdbunit.bean.DatabaseDataSourceConnectionFactoryBean">
        <property name="dataSource" ref="dataSource"/>
        <property name="databaseConfig" ref="dbUnitDatabaseConfig" />
        <property name="schema" value="<your_schema_name>"/>
    </bean>

【讨论】:

【参考方案3】:

设置数据库架构为我修复了它:

@Bean
public DatabaseDataSourceConnectionFactoryBean dbUnitDatabaseConnection(final DataSource dataSource)
    final DatabaseDataSourceConnectionFactoryBean connectionFactory = new DatabaseDataSourceConnectionFactoryBean();
    connectionFactory.setDataSource(dataSource);
    connectionFactory.setSchema(DB_SCHEMA);
    return connectionFactory;

【讨论】:

【参考方案4】:

我在执行 Dbunits aginst Oracle DB 时遇到了同样的 AmbiguousTableNameException。它工作正常,有一天开始抛出错误。

根本原因:调用存储过程时,误将其修改为小写。当更改为大写时,它开始工作。

我也可以通过将 shema 名称设置为 IDatabaseTester 来解决这个问题,例如 iDatabaseTester.setSchema("SCHEMANAMEINCAPS")

另外请确保您的连接不仅仅访问具有相同表名的多个模式。

【讨论】:

【参考方案5】:

在 DBUnit 运行之前从 Hibernate 导入数据时,您可能会遇到问题。根据您使用的数据库,表名和列名的大小写可能很重要。

例如,在 HSQL 中,数据库名称必须以大写形式声明。 如果你通过 Hibernate 的 import.sql 导入数据,请确保那里的表名也是大写的,否则你会遇到以下问题:

Hibernate 以小写形式创建表格 DBUnit 从小写的 DB 中读取表名 DBUnit 尝试使用大写表名导入其数据集 您最终会陷入混乱,名称不明确。

请记住还要检查是否在上一次运行期间创建了多个表(大写和小写),在这种情况下您也需要清理它。

【讨论】:

以上是关于Oracle + dbunit 获取 AmbiguousTableNameException的主要内容,如果未能解决你的问题,请参考以下文章

DbUnit - 警告:AbstractTableMetaData

DBUnit 中的日志输出

DbUnit:NoSuchColumnException 和区分大小写

依赖数据库的单元测试——DBUnit

DBUnit使用案例

DBUnit 有没有办法自动创建表?