java - Spring Data JPA 在两个查询中加载一对一关系

标签 java spring hibernate spring-data-jpa

我有以下类(class):

@Data
@Entity
@Table(name = "user_info")
public class UserIdentity implements UserDetails {
  @Id
  private Long id;
  private String username;
  private String password;
  private String facebookId;
  private String phoneNumber;
  private Timestamp unblockDate;
  @OneToOne(fetch = FetchType.EAGER, optional = false)
  @Fetch(FetchMode.JOIN)
  @JoinColumn(name = "role_id")
  private Role role;

  }

@Data
@Entity
@Table(name = "role")
public class Role {
  @Id
  private Long id;
  private String name;
  @ManyToMany
  @Fetch(FetchMode.JOIN)
  @JoinTable(name = "role_permission", joinColumns = @JoinColumn(name = "role_id", referencedColumnName = "id"), inverseJoinColumns = @JoinColumn(name = "permission_id", referencedColumnName = "id"))
  private List<Permission> permissions;
}

@Data
@Entity
@Table(name = "permission")
public class Permission {
  @Id
  private Long id;
  private String name;
}

当我查询用户身份时,我可以在日志中看到生成的这两个查询:

Hibernate: select useridenti0_.id as id1_3_, useridenti0_.facebook_id as facebook2_3_, useridenti0_.password as password3_3_, useridenti0_.phone_number as phone_nu4_3_, useridenti0_.role_id as role_id7_3_, useridenti0_.unblock_date as unblock_5_3_, useridenti0_.username as username6_3_ from user_info useridenti0_ where useridenti0_.facebook_id=?
Hibernate: select role0_.id as id1_1_0_, role0_.name as name2_1_0_, permission1_.role_id as role_id1_2_1_, permission2_.id as permissi2_2_1_, permission2_.id as id1_0_2_, permission2_.name as name2_0_2_ from role role0_ left outer join role_permission permission1_ on role0_.id=permission1_.role_id left outer join permission permission2_ on permission1_.permission_id=permission2_.id where role0_.id=?

但我希望只看到一个查询。你能帮我找出我做错了什么吗?

更新 我正在使用 CrudRepositor:

public interface UserIdentityRepository extends CrudRepository<UserIdentity, Long> {
  UserIdentity findByFacebookId(String facebookId);
  UserIdentity findByPhoneNumber(String phoneNumber);
}

我注意到以下事情 - 如果我使用 findById() 方法进行搜索,它会按预期工作,但如果我使用 findByPhoneNumber() 则它不起作用。

最佳答案

您在实体上指定的 FetchType 仅在 EntityManager 上使用不接受任何类型查询的方法时使用。 这就是您在调用 findById() 时有效执行的操作。

但是,如果您在使用查询方法作为 findByPhoneNumber() 时像 Spring Data 那样指定查询,则该查询用于确定获取策略。

但是您可以通过指定实体图来控制它。您将向您的实体类添加一个 @NamedEntityGraph 注释。以及引用该命名实体图的查询方法的 @EntityGraph 注释。

我已经准备好了a simple example over at Github

我使用了琐碎的实体图

@NamedEntityGraph(name = "joined", includeAllAttributes = true)

它会急切地获取所有内容,这可能正是您想要的。

关于java - Spring Data JPA 在两个查询中加载一对一关系,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48196290/

相关文章:

java - 使用 @PrePersist 和 @PreUpdate 时,创建的日期字段在更新时变为空

java - 处理用户操作的表单数据的正确方法是什么?

java - 如何在 Gradle 中更新依赖项

java - 在 String 中使用 ReplaceAll 时出现正则表达式异常

Spring - 使用 cookie 的 RESTful 身份验证

java.lang.NoSuchMethodError 因为 Tomcat 的 lib 文件夹中有重复的类

eclipse - 如何在 JBoss AS 7.1.3 中添加更新版本的 Hibernate

java - 如何在 Java 中设置 Appengine DataStore 的日志级别

java - XJC 不在 xs :union 内生成枚举

java - RabbitMQ 跨多线程多消费者负载均衡