Hibernate中将基本类型的一对一映射到单个类

7
因此,情况如下:有一个实体需要与字典连接。想象一下以下的结构。
create table Address (
    addressId bigint not null,
    addressLine1 varchar(255),
    city varchar(255),
    country varchar(255),
    state varchar(255),
    zipCode varchar(255),
    primary key (addressId)
)

create table STATES_DICT (
    state_code varchar(255),
    state_fullname varchar(255), 
    primary key (state_code)
)

我希望将地址和州字典映射到一个实体中。
@Entity
@Table(name = "ADDRESS")
public class Address implements Serializable {

    @Id
    @Column(name = "ADDRESSID")
    private int addressId;

    @Column(name = "ADDRESSLINE1")
    private String addressLine1;

    @Column(name = "STATE")
    private String state;

    //??? annotations
    private String fullStateName;

    @Column(name = "ZIPCODE")
    private String zipCode;

    @Column(name = "CITY")
    private String city;

    @Column(name = "COUNTRY")
    private String country;
    //... getters and setters
}

对于纯 SQL,我会运行

select a.ADDRESSID, a.ADDRESSLINE1, a.CITY, a.ZIPCODE, a.STATE, 
    d.STATE_FULLNAME, a.COUNTRY 
from ADDRESS a, STATES_DICT d where a.STATE = d.STATE_CODE

但是我在将其与JPA相映射时遇到了严重的问题。

  1. 我无法使用@SecondaryTable,因为这些表不是由主键映射的
  2. 我能得到的最好结果是:
    @ElementCollection
    @JoinTable(name="STATES_DICT", 
            joinColumns=@JoinColumn(name="STATE_CODE", referencedColumnName="STATE"))
    @Column(name = "STATE_FULLNAME")
    private Collection<String> fullStateName;

缺点是 - 映射始终是一对一的,集合带来了混乱,关系更多的是一对一(多对一)而不是一对多。
有什么想法吗?是否有与一个对一个映射等效的@ElementCollection? 删除@ElementCollection并没有帮助。预计fullStateName字段将在ADDRESS列中出现 - 这不是情况。
一些注意事项: * 我需要这两个内容在一个实体中保持在一起。 * 我正在扩展现有的解决方案,只需要添加这个字典列 * 实体稍后会被一些其他服务处理,该服务仅运行基本类型。我宁愿不更改服务,这就是为什么添加@OneToOne关系不可取的原因
非常感谢
我通过@SecondaryTable示例扩展了问题 - 这对我没有用。
@Entity
@Table(name = "ADDRESS")
@SecondaryTable(name="STATES_DICT", 
        pkJoinColumns=@PrimaryKeyJoinColumn(columnDefinition="STATE_CODE", referencedColumnName="STATE"))
public class Address implements Serializable {

    @Id
    @Column(name = "ADDRESSID")
    private int addressId;

    @Column(name = "ADDRESSLINE1")
    private String addressLine1;

    @Column(name = "STATE")
    private String state;

    @Column(table="STATES_DICT", name = "STATE_FULLNAME")
    private String fullStateName;

    @Column(name = "ZIPCODE")
    private String zipCode;

    @Column(name = "CITY")
    private String city;

    @Column(name = "COUNTRY")
    private String country;

    //... getters and setters
}

造成了一个讨厌的异常: Caused by: org.hibernate.AnnotationException: SecondaryTable JoinColumn 不能引用非主键
为了记录 - 我找不到一种方法来做到这一点(并且认为这不是应该完成的方式)。我已经使用关系注释(@ManyToOne,@OneToOne)和 @JoinColumn - 所以是正确的方式。我已经调整了进一步处理逻辑,以将 @JoinColumn 注释与处理 @Column 相同的方式进行处理。它起作用了。
进一步处理是安全功能,基于用户角色和原始数据库列名抑制值。这就是为什么坚持使用 @Column 注释对我如此重要的原因。

嗯,“@SecondaryTable” 应该适用于此。您能详细说明为什么它不起作用吗? - Mac
@SecondaryTable期望通过主键进行映射 - 我已经扩展了这个问题,附上了未能正常工作的SecondaryTable示例。 - Jakub Marchwicki
3个回答

1
用户2601805的答案是正确的,使用保留HTML的方式。
@PrimaryKeyJoinColumn(name="STATE_CODE", referencedColumnName="STATE")

在您的情况下,这应该足够了,因为您想要的只是从STATES_DICT获得属性。
我还提出了一个关于JPA的问题,展示了使用@SecondaryTable和@Embeddable实现类似于@ElementCollection但针对@OneToOne的方式的示例。
此外,请参阅此博客的示例http://www.bagdemir.com/2013/03/03/mapping-embeddable-objects-whichve-no-identities-using-multiple-tables-with-jpahibernate/

0

@pkJoinColumns中你应该使用:

@PrimaryKeyJoinColumn(name="STATE_CODE", referencedColumnName="STATE")

我遇到了同样的问题,但是你提供的使用@SecondaryTable的提示解决了我的问题。请参阅将一个实体映射到多个表以获取更多信息。


0

你应该能够创建一个@Embeddable状态类来映射关系,但提供一个委托方法来将状态名称公开为属性。这样做是否可以实现你想要的功能?


不完全是这样;使用@Embeddable类会产生嵌套关系,而我想要避免这种情况。我希望能够直接映射到原始类型,就像在集合示例中所示(但不使用集合)。委托属性可能是一种选择(我考虑过),但会影响其他服务中的更改(处理我的实体)。这是我想要避免的...你知道,遗留系统、复杂的回归测试等等。 - Jakub Marchwicki

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接