강의

멘토링

로드맵

Inflearn brand logo image

인프런 커뮤니티 질문&답변

nickheatlhy님의 프로필 이미지
nickheatlhy

작성한 질문수

자바 ORM 표준 JPA 프로그래밍 - 기본편

영속성 컨텍스트 2

em.find 1차 캐시 SELECT 2번 발생

작성

·

63

·

수정됨

0

[SELECT 2번 발생]
강의에 설명된 내용과 달리 저는 em.find(Member.class, 100L); 두 번 호출하였을 때 SELECT 문이 두 번 발생하였습니다.

혹시 설정 차이로 다른 결과가 나온 것일까요?

 

추가로 지금 확인해보니 tx.commit 이후에도 데이터베이스에 저장도 되지 않고, insert 쿼리가 나가지 않네요.. 어떤 문제에서 잘못된 것일까요?

전체 코드 보내드립니다.

package hellojpa;

import jakarta.persistence.EntityManager;
import jakarta.persistence.EntityManagerFactory;
import jakarta.persistence.EntityTransaction;
import jakarta.persistence.Persistence;

import java.util.List;

public class JpaMain {

    public static void main(String[] args) {
        EntityManagerFactory emf = Persistence.createEntityManagerFactory("hello");
        EntityManager em = emf.createEntityManager();

        EntityTransaction tx = em.getTransaction();

        /**
         * 영속성 컨텍스트1 - 1차 캐시
         */
        try {
//            // 비영속
            Member member = new Member();
            member.setId(100L);
            member.setName("HelloJPA");

            System.out.println("=== BEFORE ===");
            em.persist(member); // 영속
            System.out.println("=== AFTER ===");

            Member member1 = em.find(Member.class, 100L);
            Member member2 = em.find(Member.class, 100L);

            System.out.println("result = " + (member1 == member2));

            tx.commit();
        } catch (Exception e) {
            tx.rollback();
        } finally {
            em.close();
            emf.close();
        }
    }
}

답변 2

0

nickheatlhy님의 프로필 이미지
nickheatlhy
질문자

질문 드리다 보니 begin 메서드가 빠져 있어서 데이터베이스에 반영되지 않은 것을 확인했습니다..😅

  1. 그래도 여전히 궁금한 부분은.. em.find()를 두 번 했을 때 영속성 컨텍스트에 저장되어 있는 값을 불러오니까 아예 select 문이 발생하지 않아야 할 것 같은데 왜 두번이 발생하게 되는지 잘 모르겠습니다.

 

  1. 그리고 이미 데이터베이스에 저장되어 있는 값(100L)을 여러 번 커밋해도 유니크 에러가 발생하지 않는 이유도 궁금해요

김영한님의 프로필 이미지
김영한
지식공유자

안녕하세요. nickheatlhy님

  1. 트랜잭션 없이 읽는 경우에 JPA에서는 트랜잭션 없이 읽기라는 특별한 기능이 사용됩니다.

    이 경우 영속성 컨텍스트에 엔티티를 보관하지 않습니다. 따라서 계속 조회됩니다.

  2. 이 부분은 뭔가 잘못된 것 같아요. 유니크 제약조건이 정상적으로 걸려있는지 확인하세요. 추가로 catch 부분에 e.printStacktrace()를 출력해보시면 오류 메시지를 확인할 수 있습니다.

감사합니다.

 

 

0

안녕하세요, 인프런 AI 인턴입니다.

nickheatlhy님이 경험하고 계신 문제는 JPA의 영속성 컨텍스트와 관련된 설정이나 운영 방식의 차이에서 기인할 수 있습니다. 주의해야 할 몇 가지 사항을 살펴보겠습니다.

문제 원인 및 해결 방안

  1. 1차 캐시와 em.find() 결과:

    • JPA는 em.find()를 호출할 때 먼저 1차 캐시에서 엔티티를 조회합니다. 1차 캐시에 엔티티가 이미 존재하면 데이터베이스에 쿼리를 날리지 않습니다. 그러나 두 번의 SELECT 쿼리가 발생한다면, 아마도 1차 캐시가 제대로 사용되지 않았기 때문일 수 있습니다.
    • 여러 원인이 있을 수 있지만, persistence.xml에서 JPA의 구성이 올바르게 되어 있는지를 확인해볼 필요가 있습니다.
  2. 데이터베이스 저장 문제:

    • 데이터가 저장되지 않고 INSERT 쿼리가 실행되지 않는다면, 트랜잭션이 적절하게 설정되지 않았거나 데이터베이스 설정(특히 hibernate의 ddl-auto 설정)에 문제가 있을 수 있습니다.
    • Hibernate의 hbm2ddl.auto 옵션이 createcreate-drop으로 설정되어 있으면, 애플리케이션이 실행될 때마다 기존의 테이블을 삭제하고 새로 생성할 수 있습니다. 이 경우 애플리케이션이 실행하기 전에 데이터가 저장되지 않게 됩니다. update로 설정하면 테이블의 기존 데이터를 유지할 수 있습니다.

관련된 이전 유사 질문들

  1. 쿼리가 두번 나가는 이유가 뭘까요? ㅠ

    • persistence.xml 설정 문제로 인해 동일한 결과가 발생한 사례입니다. 설정 확인의 중요성에 대해 논의됩니다.
  2. em.find 시 쿼리문 2번 수행

    • em.find 쿼리의 2번 실행과 테이블 생성 옵션의 관계성에 대해 설명하고 있습니다.

위의 정보를 참고하여 persistence.xml 파일 설정을 점검하시고, 여전히 문제가 지속된다면 설정 파일의 전체 내용을 공유해 주시면 문제 해결에 도움이 될 것입니다.

저는 질문자님의 이해를 돕고자 지식공유자님의 기존 답변을 참고하여 유사한 답변 링크를 추천해드리고 있습니다.
현재 베타 기능으로 답변이 만족스럽지 않을 수 있는 점 양해 부탁드립니다. 🙏
추가적으로 궁금한 점이 있으시면, 이어서 질문해 주세요. 곧 지식공유자께서 답변해 주실 것입니다.

nickheatlhy님의 프로필 이미지
nickheatlhy

작성한 질문수

질문하기