Почему @Basic (fetch = lazy) не работает в моем случае? - PullRequest
0 голосов
/ 03 сентября 2018

Я знаю, что подобные вопросы уже задавались много раз, но я не нашел такого, который мог бы мне помочь.

Итак, могу ли я попросить вас помочь мне найти причину, по которой Book s title выбирается EAGERly?

У меня очень простая кодовая база, вот моя сущность:

@Entity
@NoArgsConstructor
@AllArgsConstructor
@Data
@Builder
public class Book {

    @Id
    @GeneratedValue
    private int id;

    @Lob
    @Basic(fetch = FetchType.LAZY, optional = false)
    private String title;

}

А вот клиентский фрагмент кода, в данном случае представленный как psvm():

public static void main(String[] args) {

    final ApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);
    final SessionFactory sessionFactory = context.getBean(SessionFactory.class);
    Session session = sessionFactory.openSession();

    Book book = Book.builder().title("Peace and War").build();

    Transaction tx = session.beginTransaction();
    session.save(book);
    tx.commit();
    session.close();

    session = sessionFactory.openSession();
    book = session.get(Book.class, book.getId());
}

Я также добавил плагин в maven для улучшения байт-кода:

   <build>
        <plugins>
            <plugin>
                <groupId>org.hibernate.orm.tooling</groupId>
                <artifactId>hibernate-enhance-maven-plugin</artifactId>
                <version>5.3.6.Final</version>
                <executions>
                    <execution>
                        <configuration>
                            <enableLazyInitialization>true</enableLazyInitialization>
                        </configuration>
                        <goals>
                            <goal>enhance</goal>
                        </goals>
                    </execution>
                </executions>
            </plugin>
        </plugins>
   </build>

Но, тем не менее, title охотно выбирается с помощью следующего запроса:

Hibernate: select book0_.id as id1_0_0_, book0_.title as title2_0_0_ from Book book0_ where book0_.id=?

, который я вижу из-за hibernate.show_sql=true

Не могли бы вы помочь мне понять, что я делаю неправильно? Кажется, что ответ на поверхности, но я не могу его найти.

1 Ответ

0 голосов
/ 03 сентября 2018
 @Basic(fetch = FetchType.LAZY, optional = false)

В соответствии со спецификацией JPA @Basic Ленивая загрузка не гарантируется в зависимости от того, какую реализацию вы используете. Вы можете проверить это здесь https://en.wikibooks.org/wiki/Java_Persistence/Basic_Attributes

Поддержка извлечения LAZY на Basic не является обязательной в JPA, поэтому некоторые поставщики JPA могут ее не поддерживать.

В любом случае, один из обходных путей - создать новую отдельную сущность, скажем, например, BookTitle, затем установить отношение один-к-одному и Lazily загрузить его из сущности книги:

public class BookTitle {
    @Id
    @GeneratedValue
    private Long id;

    @Lob
    @Basic(fetch = FetchType.LAZY, optional = false)
    private String title;
}


public class Book {

    @Id
    @GeneratedValue
    private int id;

    @OneToOne (fetch = FetchType.LAZY)
    private BookTitle bookTitle;

}
...