3

简洁版本

Hibernate 正在执行多个选择查询而不是仅两个查询来检索一对一的关系

长版

人物类:

@Entity
@Table(name = "people")
public class Person{

  @Id
  @GeneratedValue
  @Column
  private Integer id;

  @Column
  private String name;

  @OneToOne(optional=false, fetch=FetchType.EAGER)
  private Job job;
}

工作类别:

@Entity
@Table(name = "jobs")
public class Job{

  @Id
  @GeneratedValue
  @Column
  private Integer id;

  @Column
  private String name;
}

显示所有人员及其工作:

Session session = HibernateUtil.getSessionFactory().getCurrentSession();
session.beginTransaction();
List<Person> people = session.createQuery("from Person").list();
for(Person p : people){
    System.out.println(p.getName() + " | " + p.getJob().getName());
}

我所期望的:两个选择,五个人一个,三个不同工作的一个(三个人有相同的工作)

我得到了什么:四个选择,一个用于人员,三个用于每个不同的工作

Hibernate: select person0_.id as id1_, person0_.job_id as job3_1_, person0_.name as name1_ from people person0_
Hibernate: select job0_.id as id0_0_, job0_.name as name0_0_ from jobs job0_ where job0_.id=?
Hibernate: select job0_.id as id0_0_, job0_.name as name0_0_ from jobs job0_ where job0_.id=?
Hibernate: select job0_.id as id0_0_, job0_.name as name0_0_ from jobs job0_ where job0_.id=?
Jim | Programmer
Andrew | DBA
Tomas | Sysadmin
Henry | Sysadmin
Jerry | Sysadmin

如果我有 1000 个人从事 1000 个不同的工作,我会看到 1000 个选择被执行。我怎样才能让 Hibernate 只做两个选择?

谢谢

4

2 回答 2

8

这是n + 1问题的一个示例, Hibernate 手册建议使用JOINfetch 模式。您可以使用特定于休眠的注释启用此功能

@Fetch(FetchMode.JOIN)
于 2012-04-25T21:26:37.700 回答
2

你没有一对一的关系,你有一个一对多的关系。一个工作可以有多个人参与其中。Hibernate 假设实际上存在一对一的关系,并且没有注意到三个人指向一份工作。然后你必须解决 N+1 选择问题,正如 beerbajay 指出的那样。

于 2012-04-25T21:27:39.750 回答