如何在Hibernate/javax.persistence中将一个类映射到多个表?

24

我想使用一个类来映射三个表。我知道javax.persistence提供了@SecondaryTable注解,可以将两个表映射到一个类上。

以下是代码,其中我使用了@SecondaryTable。它只允许我定义一个次要表。但我需要将同一个类用于3个表。

@Entity
@Table(name = "table1")
@SecondaryTable(name="table2")
public class TableConfig
    implements Serializable {

    /**
     * 
     */
    private static final long serialVersionUID = 1L;

    @Id
    @Column(name = "mac", table= "table1")
    private String uniqueIdentifier;

我认为在这里你可以找到答案:http://stackoverflow.com/questions/5215471/how-to-map-2-identical-tables-same-properties-to-1-entity - Phil_Charly
如果表格结构不同且没有共享相同的主键,请参考此文章:https://www.thoughts-on-java.org/hibernate-tips-how-to-map-an-entity-to-multiple-tables/ - Guillaume Husta
3个回答

27
使用 @SecondaryTables 来映射多个表。
您可以使用 @SecondaryTables 类级别的注释将单个实体 bean 映射到多个表。要表示某个列在特定的表中,请使用 @Column@JoinColumn 的 table 参数。
例如,有三个实体,分别是:NameAddressStudentName 实体看起来像这样:
@Entity
@Table(name="name")
public class Name implements Serializable {

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

    public Name(){}

    public Name(int id,String name){
        this.id=id;
        this.name=name;
    }
        //getters and setters
}

地址 实体将如下所示:

@Entity
@Table(name="address")
public class Address implements Serializable {

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

    public Address(){}

    public Address(int id, String address) {
        super();
        this.id = id;
        this.address = address;
    }
        //getters and setters
}

Student实体将如下所示:

@Entity
@Table(name="student")
@SecondaryTables({
    @SecondaryTable(name="name", pkJoinColumns={
        @PrimaryKeyJoinColumn(name="id", referencedColumnName="student_id") }),
    @SecondaryTable(name="address", pkJoinColumns={
        @PrimaryKeyJoinColumn(name="id", referencedColumnName="student_id") })
})
public class Student implements Serializable {

    @Id
    @Column(name="student_id")
    private int studentId;

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

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

    public Student(){}

    public Student(int studentId){
        this.studentId=studentId;
    }
        //getters and setters
}

店铺类似于:

    Student s= new Student(1);
    session.save(s);

    Name n=new Name(s.getStudentId(),"Bilal Hasan");
    session.save(n);    

    Address address = new Address(s.getStudentId(), "India");
    session.save(address);

    Student ob = (Student)session.get(Student.class, s.getStudentId());

    System.out.println(ob.getStudentId());
    System.out.println(ob.getName());
    System.out.println(ob.getAddress());

输出:

1
Bilal Hasan
India

1
嗨Rembo,谢谢你的回答,但在这个例子中,你将三个表连接成一个。在我的情况下,我只想映射它。就像数据库中的三个表几乎相同,我只想用一个类来表示它们。 - Muhammad Bilal Hasan
1
@BilalHasan 我认为这是不可能的,如果你找到了解决方案,请告诉我。 - Abhishek Nayak

12

您可以定义一个类,如下所示:

@Entity
@Table(name="table1")
@SecondaryTables({
      @SecondaryTable(name="table2", pkColumnJoins={@PrimaryKeyJoinColumn(name = "id")}),
      @SecondaryTable(name="table3", pkColumnJoins={@PrimaryKeyJoinColumn(name = "id")})
  })
public class TestEntity {
      @Id
      @GeneratedValue
      private int id;

      private String field1;

      @Column(name="column2", table="table2")
      private String field2;

      @Column(name="column3", table="table3")
      private String field3;

      getter and setter...
}

你的数据库应该拥有三张表,每张表都应该有相同的主键"id"。

然后,可以像这样进行测试:

TestEntity test = new TestEntity();
test.setField1("field1");
test.setField2("field2");
test.setField3("field3");

em.merge(test);

测试完成后,在您的数据库中,每个表格将会存在一条记录:

表格1:

 1, field1

表格2:

 1, field2

table3:

 1, field3

他们所有的记录都将共享相同的主键值。希望这能帮到你。

0
在 Hibernate 映射文件中,您可以使用虚拟名称指定实体名称映射,并配合 polymorphism="explicit" 使用物理类名。您可以进行多个映射。在加载对象时,请使用实体名称(虚拟名称)。

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