当前位置: 首页 > 面试题库 >

按类型表映射子类(多对一hibernate)

颜熙云
2023-03-14
问题内容

我必须制作一个拼贴项目的域层。我们几乎没有标准,例如我们必须使用Hibernate,数据库也已修复。

数据库的相关部分几乎如下所示:

总线实体 (表1)

  • 总线编号
  • 公交专用信息

总线类型 (表2)

  • BusTypeId
  • 座位数
  • 子类鉴别器

我遇到的问题是,在域层中有2种类型的总线通过BusType表中的标识符进行区分:

@Entity
class Bus
{
   @Id
   int _id;
   ...
}

@Entity
class BusSubClass1 extends Bus
{
   ...
}

@Entity
class BusSubClass2 extends Bus
{
   ...
}

有没有办法用Hibernate和JPA映射类似的东西?谢谢你的回答。

乔钦·莫伦特(Jochen Morent)


问题答案:

是的,几乎没有办法映射这种情况。JPA支持三种不同的数据表示形式:

  • 单表 策略
  • 联合 策略
  • 逐表 策略

换句话说,根据所使用的继承类型,您将获得不同的数据库模型。各种JPA提供程序可能支持其他继承策略。

考虑以下示例:

@Entity
@Inheritance //by default SINGLE_TABLE strategy
@DiscriminatorColumn( //not supported for TABLE_PER_CLASS strategy
    name = "BUS_TYPE", 
    discriminatorType = DiscriminatorType.INTEGER
)
public abstract class Bus {
    @Id
    protected int id;
    protected int seats;

    public Bus() {
    }
}

@Entity
@DiscriminatorValue(value = "1") //not supported for TABLE_PER_CLASS strategy
public class BusSubClass1 extends Bus {
    private String specific1;

    public BusSubClass1() {
    }
}

@Entity
@DiscriminatorValue(value = "2") //not supported for TABLE_PER_CLASS strategy
public class BusSubClass2 extends Bus {
    @Temporal
    private Data specific2;

    public BusSubClass2() {
    }
}

使用InheritanceType.SINGLE_TABLE策略将导致一个包含所有具体实体类型的数据库表:

 Bus

 ID BUS_TYPE SEATS SPECIFIC1 SPECIFIC2
 -- -------- ----- --------- ---------
 1  1        50    qwerty
 2  1        55    asdfgh
 3  2        30              2014-01-01
  • 继承层次结构中的每个具体实体实例都通过区分符值进行区分(此处由BUS_TYPE列指示)
  • 所有数据库列都必须声明为 空, 因为并非所有数据库列都可能包含值(因此,对于无法设置为null的列可能会出现问题)
  • 宽或深的层次结构可能会影响性能(表中分别有很多冗余的列或行)
  • 缺乏规范化会浪费数据库表中的空间
  • 为读取和写入操作提供良好的性能(多态查询不需要连接,仅需要鉴别符值)
  • 添加/删除实体字段可能有问题(从数据库管理的角度来看)

使用InheritanceType.JOINED策略会导致每个实体类型具有多个数据库表(来自的所有共享字段Bus都存储在相应的表中):

 Bus                        BusSubClass1          BusSubClass2

 ID BUS_TYPE SEATS          ID SPECIFIC1          ID SPECIFIC2
 -- -------- -----          -- ---------          -- ---------
 1  1        50             1  qwerty             3  2014-01-01
 2  1        55             2  asdfgh
 3  2        30
  • 继承层次结构中的每个具体实体类都通过鉴别符值进行区分(此处由BUS_TYPE列指示)
  • 标准化可改善数据存储(与SINGLE_TABLE策略相比,减少了未使用的空间)
  • 非多态查询(针对单个具体实体)需要联接
  • 多态查询(用于实体类的广泛或深入层次结构)需要多个联接,并且可能很昂贵
  • 添加/删除实体字段非常简单(从数据库管理的角度来看)

使用InheritanceType.TABLE_PER_CLASS策略可以为每种实体类型精确地提供一个数据库表(所有共享字段Bus都在具体的子类中重新定义):

 BusSubClass1                BusSubClass2

 ID SEATS SPECIFIC1          ID SEATS SPECIFIC2
 -- ----- ---------          -- ----- ---------
 1  50    qwerty             3  30    2014-01-01
 2  55    asdfgh
  • 继承层次结构中的每个具体实体类仅通过共享标识符来区分(不使用鉴别​​符)
  • 规范化改善了数据存储(与SINGLE_TABLE策略相比,未使用的空间更少,但比JOINED策略的情况要多)
  • 非多态查询(对于单个具体实体)非常有效,因为不需要连接
  • 多态查询(用于实体类的广泛或深入层次结构)需要多个联接,并且可能很昂贵
  • 添加/删除实体字段非常简单(从数据库管理的角度来看)


 类似资料:
  • 我对Flutter编程非常陌生,我正在尝试导入一个本地JSON文件,其中包含书籍数据,如标题、作者、发布年份等。 最初我使用JSON-to-DART转换器来组装一个图书数据模型,现在我正在尝试创建一个函数,在该函数中,程序获取本地JSON文件,并使用类中的方法将数据解析为地图。 我遇到的问题是返回类型

  • 我是一个新的颤振和得到类型错误。我正在尝试使用json自动序列化。 在做了一些调整后这里看起来是这样的 下面是我如何尝试从api获取数据 我的类如下所示 谁能帮我一下吗。我被困在这里了。我一直在尝试不同的方法,但都不管用。非常感谢。

  • 你好,我正在做一个api调用,在那里我得到了一系列的横幅。我正在犯错误- 横幅。飞奔 横幅。飞奔 api_base_助手。飞奔 得到你的横幅。飞奔

  • 因此,我创建了一个应用程序,通过api读取数据,并尝试解析JSON api 这是我的错误截图 我试着把它改成一个列表,但它仍然读取一个错误 这是我的密码大象。飞奔 如何纠正此错误?请帮忙

  • 我在尝试使用JSON_serializable序列化从internet获取的复杂JSON并使用FutureProvider将它们连接在一起时遇到了这个问题。这是json文件, 这是包含序列化类的代码, 获取json文件并将其转换为dart的未来提供者, 这是基因编码, 这是转换后的json文件显示的代码 当我试着运行这个项目时,我遇到了这个错误 i/flutter(23347): type'Lis

  • 我试图获得用户信息后,用户输入电子邮件和密码。我使用的api会返回我用户的额外信息,所以我尝试从中获取这些信息。 我试图解析这个json: 这些是我的模型: 这是我发送电子邮件、密码和获取用户数据的部分: 这是给我错误的部分 我打印了响应体,它给出了我想要的字符串,但是jsonDecode返回了一个列表,所以我不能使用它。我怎样才能解决这个问题?提前谢谢。