JOIN FETCH в spring-data-jpa не вызывает ленивых ассоциаций - PullRequest
0 голосов
/ 14 мая 2019

У меня проблема с получением ленивой @ManyToMany ассоциации в spring-data-jpa. У меня есть список фильмов, связанных с пользователями, которые сделали этот фильм любимым, и список жанров:

@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.getUsers() пусто после выполнения этого запроса.

Я обнаружил , что использование @EntityGraph может помочь, но это не значит, что ассоциация movie.getUsers() все еще пуста.

Я также пытался сделать эту ассоциацию как FetchType.EAGER просто для тестирования, но она все еще пуста.

У меня нет идей, поэтому буду признателен за любую помощь.

UPD: Это не могло быть проблемой с пользователями, отсутствующими в БД, поскольку у меня есть другой запрос, который извлекает только фильмы, которые отмечены как избранные пользователем в жанре избранного пользователя. Поэтому, если бы пользователь не присутствовал в БД, запрос вернул бы мне пустой результат, но я получил бы результат, хотя movie.getUsers() пусто. Вот метод / запрос:

@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 Ответ

2 голосов
/ 14 мая 2019

Да, однонаправленное отображение должно работать. У меня нет никаких проблем с базовым примером того, что вы делаете. Это должно быть хорошей отправной точкой, чтобы помочь вам понять, в чем различия. Я заметил, что не присоединяюсь к 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);
}
...