当前位置: 首页 > 知识库问答 >
问题:

spring数据中的JOIN FETCH jpa不获取惰性关联

阚英睿
2023-03-14

我在spring数据jpa中获取lazy@manytomy关联时遇到问题。我有一个电影列表,它与使这部电影最受欢迎的用户和类型列表有关联:

@Entity
@Table("movie")
public class Movie {
...

  @ManyToMany(fetch = FetchType.EAGER)
  @JoinTable(...)
  private List<Genre> genres;

  @ManyToMany
  @JoinTable(name = "users_movies",
            joinColumns = @JoinColumn(name = "movie_id", referencedColumnName = "id"),
            inverseJoinColumns = @JoinColumn(name = "user_id", referencedColumnName = "id"))
  private List<User> users = new ArrayList<>;
}

体裁还与将该体裁标记为喜爱的用户有关联:

@Entity
@Table("genre")
public class Genre {
....
  @ManyToMany
  @JoinTable(name = "users_genres",
            joinColumns = @JoinColumn(name = "genre_id", referencedColumnName = "id"),
            inverseJoinColumns = @JoinColumn(name = "user_id", referencedColumnName = "id"))
  private List<User> users;
}

我有一个存储库,具有以下方法/查询,必须按最喜欢的流派返回所有电影,并加载电影的用户,以便我可以显示用户是否将电影标记为最喜欢的电影:

@Query(value = "SELECT movie FROM Movie movie " +
               "JOIN movie.genres genre JOIN genre.users grenreUser " +
               "LEFT JOIN FETCH movie.users " +
               "WHERE grenreUser.id = :userId",
      countQuery = "SELECT COUNT(movie) FROM Movie movie " +
                   "JOIN movie.genres genre JOIN genre.users grenreUser " +
                   "LEFT JOIN movie.users " +
                   "WHERE grenreUser.id = :userId")
Page<Movie> getAllMoviesByFavoriteGenres(@Param("userId") String userId, Pageable);

但是这里我有一个问题,movie.get用户()在执行此查询后是空的。

我发现使用@EntityGgraph可能会有所帮助,但它没有movie.get用户()关联仍然是空的。

我还尝试将此关联设为FetchType。EAGER只是为了测试,但它仍然是空的。

我没有主意了,所以我会感谢你的帮助。

UPD:这不可能是用户不在数据库中的问题,因为我有另一个查询,它只提取用户在收藏夹用户类型中标记为收藏夹的电影。因此,如果用户没有出现在数据库中,查询将返回空结果,但是我确实得到了结果,尽管movie.get用户()是空的。下面是方法/查询:

@Query(value = "SELECT movie FROM Movie movie " +
                   "JOIN movie.genres genre " +
                   "JOIN FETCH movie.users user " +
                   "WHERE user.id = :userId AND :userId MEMBER OF genre.users",
          countQuery = "*same but with count*")
Page<Movie> getAllFavoriteMoviesByFavoriteGenres(@Param("userId") String userId, Pageable);

共有1个答案

那宏大
2023-03-14

是的,单向映射应该可以工作。我对您正在做的事情的基本示例没有任何问题。这应该是帮助您找出差异的一个很好的起点。我注意到我没有将movie.users加入CountQuery

@Entity
public class Movie {
    @Id @GeneratedValue private Long id;
    @ManyToMany
    private Set<Genre> genres;
    @ManyToMany
    private Set<User> users;

@Entity
public class Genre {
    @Id @GeneratedValue private Long id;        
    @ManyToMany
    private Set<User> users;

@Entity
public class User {
    @Id @GeneratedValue private Long id;

以及存储库:

public interface MovieRepository extends JpaRepository<Movie, Long> {
    @Query(value = "SELECT movie FROM Movie movie " +
            "JOIN movie.genres genres JOIN genres.users users LEFT JOIN FETCH movie.users " +
            "WHERE users.id = :userId", 
    countQuery = "SELECT COUNT(movie) FROM Movie movie " +
            "JOIN movie.genres genres JOIN genres.users users " +
            "WHERE users.id = :userId")
    Page<Movie> getAllMoviesByFavoriteGenres(@Param("userId") Long userId, Pageable page);

使用它:

@Override
public void run(String... args) throws Exception {
    create();
    System.out.println("something");
    Page<Movie> movies = movieRepo.getAllMoviesByFavoriteGenres(1L, PageRequest.of(0, 10));
    movies.forEach(System.out::println);
}

private void create() {
    User u1 = new User();
    userRepo.save(u1);

    Set<User> users = Collections.singleton(u1);

    Genre g1 = new Genre();
    g1.setUsers(users);
    genreRepo.save(g1);

    Set<Genre> genres = Collections.singleton(g1);

    Movie m1 = new Movie();
    m1.setGenres(genres);
    m1.setUsers(users);
    movieRepo.save(m1);
}
 类似资料:
  • 我正在使用Springjpa和龙目岛来定义爪哇豆主题。每个主题都会有很多评论。我的一吨配置是 我创建的restful api是这样的。序列化似乎是问题所在,它总是获取注释。正如Chris所说,我添加了,它似乎解决了问题。但是如果我想加载注释怎么办,不会返回序列化中的注释。

  • 我的Spring数据存储库被配置为默认@RepositoryRestResources,没有任何自定义。 JPA实体: ..... 我希望House obkect以JSON格式返回,作为Flat对象的嵌入部分,但只获取House的URL /仓库/公寓/442991: 同时,User-Role OneTo多项关系被取好,角色名为: 请求:/repository/users/5 除了关系类型,我不知道

  • 如果我有一个实体和一些惰性集合(、、,...),并且希望编写一个JpaRepository-method,该方法为我提供所有的人,这些人都渴望获取,这可能吗? 我知道人们可以在单个物体上做到这一点,但这在某种程度上可以在人的集合上做到吗?

  • 我的理解是,默认情况下,Hibernate将所有关系类型的FetchType设置为lazy。 在我的例子中,我有一个双向的OneToMany-ManyToOne关系,如下所示: 另一个问题是我如何得到一个有特定名字的孩子?是否可以通过存储库进行操作?还是需要getChildren(parentId)并迭代直到找到以特定方式命名的那个? 编辑:在一些建议之后,我继续这样实现了我的ChildRepos

  • 当然,这些列表有setter和getter。现在我想实现以用户身份不参加这些事件的可能性。因此,我尝试了以下方法: 我得到的例外情况如下: 尽管我使用了一些方法来确保执行惰性提取:

  • 我在尝试获取数据时收到异常,懒惰(最后异常) //应用程序通过以下DAO获取数据。 //Junit测试调用serviceX.getById //使用JUnit //某个对象类具有以下属性。 当我尝试访问jUnit中的另一个对象时,我得到了以下异常 方法已在额外配置中尝试 我们使用TransactionManager的spring注释。 还有,我尝试添加@ Transaction(Propagati