物化视图中的“缺少表格”

时间:2015-09-17 07:56:38

标签: java hibernate postgresql jpa materialized-views

我有以下实体:

@Entity
@Table(name = "full_address")
public class FullAddress {

  @Id
  @Column(name = "guid")
  private String id;

  @Column(name = "address")
  private String address;

  //getters and setters omitted
}

我还创建了物化视图,如下所示:

CREATE TABLE address_table (
  -- address fields,
  aoid VARCHAR(36),
  CONSTRAINT address_pk PRIMARY KEY (aoid)
);
CREATE MATERIALIZED VIEW full_address AS
  SELECT buildFullAddressById(addrobj.aoid) AS address, addrobj.aoid AS guid FROM address_table AS addrobj;

-- buildFullAddressById is an sql function which is not important here

当我启动Tomcat应用程序时,我总是得到

  

org.hibernate.HibernateException:缺少表:full_address

为什么会这样?怎么解决?

UPD:看起来像是hibernate中的一个bug:https://hibernate.atlassian.net/browse/HHH-9602

3 个答案:

答案 0 :(得分:2)

这绝对是Hibernate的一个错误,并在此处进行了描述:https://hibernate.atlassian.net/browse/HHH-9602

因此,我建议删除hibernate.hbm2ddl.auto属性作为解决方法。

答案 1 :(得分:1)

过去几天我遇到了同样的错误。正如this answer所述,可以在hibernate.hbm2ddl.auto中禁用persistence.xml属性,但如果您的项目正在快速发展,则不是一个好主意。

TL; DR:将属性hibernate.hbm2dll.extra_physical_table_types设置为MATERIALIZED VIEW

或者将-Dhibernate.hbm2dll.extra_physical_table_types="MATERIALIZED VIEW"添加到VM选项。但配置文件的这些选项更好。

目前,我们正在使用PostgreSQL 9.6和Hibernate 5.2.12.Final。然而,所有物化视图验证都失败了,但有以下例外:

  

引起:org.hibernate.tool.schema.spi.SchemaManagementException:   架构验证:缺少表[our_project_schema.mv_one_of_views]

成功通过验证的所有实体都是简单的表或视图。

这似乎是通用数据库的默认行为。在79-81行的here来源中,他们只添加以下类型:

final List<String> tableTypesList = new ArrayList<>();
tableTypesList.add( "TABLE" );
tableTypesList.add( "VIEW" );

85-87告诉我们有可能使用自定义值扩展这些硬编码值:

if ( extraPhysicalTableTypes != null ) {
    Collections.addAll( tableTypesList, extraPhysicalTableTypes );
}

在第56行,它被声明为private String[] extraPhysicalTableTypes;, 在71-77行上,还有一些值添加到此数组中:

if ( !"".equals( extraPhysycalTableTypesConfig.trim() ) ) {
    this.extraPhysicalTableTypes = StringHelper.splitTrimmingTokens(
        ",;",
        extraPhysycalTableTypesConfig,
        false
    );
}

它们来自行66-70,在键EXTRA_PHYSICAL_TABLE_TYPES下编码为String,默认值为空:

final String extraPhysycalTableTypesConfig = configService.getSetting(
    AvailableSettings.EXTRA_PHYSICAL_TABLE_TYPES,
    StandardConverters.STRING,
    ""
);

1545here是该密钥的声明:

/**
 * Identifies a comma-separate list of values to specify extra table types,
 * other than the default "TABLE" value, to recognize as defining a physical table
 * by schema update, creation and validation.
 *
 * @since 5.0
 */
String EXTRA_PHYSICAL_TABLE_TYPES = "hibernate.hbm2dll.extra_physical_table_types";

因此,添加此属性将向tableTypesList添加另一个条目,该条目用于过滤数据库中的许多其他实体,例如序列,索引,临时表和其他可能具有与您的物化视图类似的名称

如果您有兴趣,这就是persistence.xml的样子:

<?xml version="1.0" encoding="UTF-8"?>
<persistence xmlns="http://xmlns.jcp.org/xml/ns/persistence"
             xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
             xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/persistence http://xmlns.jcp.org/xml/ns/persistence/persistence_2_1.xsd"
             version="2.1">
    <persistence-unit name="project-pu">
        <jta-data-source>java:jboss/datasources/project-pu</jta-data-source>
        <properties>
            <property name="hibernate.dialect" value="org.hibernate.spatial.dialect.postgis.PostgisPG95Dialect"/>
            <property name="hibernate.hbm2ddl.auto" value="validate"/>
            <property name="hibernate.hbm2dll.extra_physical_table_types" value="MATERIALIZED VIEW"/>
            <property name="hibernate.show_sql" value="false"/>
            <property name="hibernate.format_sql" value="false"/>
            <property name="hibernate.use_sql_comments" value="false"/>
            <property name="hibernate.connection.url" value="jdbc:postgresql://localhost:5432/mgt"/>
            <property name="hibernate.connection.driver_class" value="org.postgresql.Driver"/>
        </properties>
    </persistence-unit>
</persistence>

P.S。我知道这是一个非常古老的帖子,但我和这个问题争了几天。我找不到答案,所以我决定把它放在互联网上的某个地方。而这个地方就在这里。 :)

答案 2 :(得分:0)

我发现最简单的解决方案(假设您仍然希望使用Hibernate来验证架构)是在表定义中添加一个子选择。

使用hbm.xml

<hibernate-mapping>
    <class name="com.initech.MvObject" table="MV_OBJ">
        <!-- This is necessary when using older Hibernate versions because of 
            Hibernate bugs. See
            https://hibernate.atlassian.net/browse/HHH-1329
            https://hibernate.atlassian.net/browse/HHH-9602
         -->
        <subselect>select * from MV_OBJ</subselect>

使用注释

import org.hibernate.annotations.Subselect;
import javax.persistence.Entity;

@Entity
@Subselect("select * from MV_OBJ")
public class MvObj{

}