작성
·
148
0
안녕하세요 ! : )
궁금한점이있어 글을 남기게되었습니다.
------
1. persist 를 했을 때 생키는 에러
2. merge를 했을 때 id 값이 존재해도 저장이된다.
3. ForeignKeys 용도가 궁금합니다.
강의 범위를 벗어난 것 같지만,,, 궁금합니다.
-----------------------------------------------
아래 과정은 해당 질문이 생기게된 진행과정입니다.
1. entityManager.persist 할 때
id 값을 같이 입력한 경우
아래와 같은 오류가 발생하였습니다.
em.persis(new Product(1L,"상품",1000L)
new PersistentObjectException("detached entity passed to persist: ")
위 오류를 해결하기 위해서 merge를 하면 해당 오류가 나지 않았습니다.
실제 쿼리를 한번조회하는 과정이 있는데 이 과정을 통해 영속 상태가되어 저장되는가? 하는 생각이 들었습니다.
코드를 아직 다 보지 못했네요 .
-----
그래서 우선 Persist 에 관해서 파악하고자 소스를 따라가게되었습니다.
-----
SessionImpl
PersistEvent 에 저장하고자 하는 객체를 담아 이벤트를 발행합니다.
@Override
public void persist(Object object) throws HibernateException {
checkOpen();
firePersist( new PersistEvent( null, object, this ) );
}
DefaultPersistEventListener 클래스에서 해당 이벤트를 받아 처리한다는 것을 알게되었습니다.
final EntityEntry entityEntry = source.getPersistenceContextInternal().getEntry( entity );
EntityState entityState = EntityState.getEntityState( entity, entityName, entityEntry, source, true );
위 코드 부분에 해당 객체의 상태를 판단하는 것 같습니다.
EntityState의 리턴값이 DETACH라면 해당 오류가 발생하는 것을 알게되었습니다.
getEntityState 메소드를 쭉 따라가보니
아래의 메소드를 호출하여 해당 entity의 isTransient를 여부를 확인한다고 하는데 이름이 확 와닿지가 않습니다.
외래키를 왜 확인하는지 ? 이해가 되지 않습니다.
ForeignKeys.isTransient(
entityName,
entity,
assumedUnsaved,
source)
해당 메소드를 타고 쭉들어가면 결국엔 아래코드를 발견하게 되는데요
id 값이 null이면 true지만 null 아니면 null을 반환합니다.
null 정보로 인해 최종적으로 PERSIST 에서 DETACH 구문에 걸려 예외가 발생함을 알게되었습니다.
// we *always* assume an instance with a null
// identifier or no identifier property is unsaved!
if ( id == null ) {
return Boolean.TRUE;
}
답변 1
0
안녕하세요. Na Seunghoo님, 공식 서포터즈 David입니다.
영속화된 엔티티는 식별자를 가지고 있습니다.
영속화된 엔티티가 영속성 컨텍스트 바깥으로 나오게 되면 detached 엔티티가 됩니다.
detached 엔티티 또한 식별자를 가지고 있습니다.
식별자가 있는 Product를 persist하려다보니 식별자가 있음을 발견하고 detached 엔티티로 인식한 것입니다.
detached 엔티티가 다시 managed 엔티티로 상태가 변경되려면 persist가 아닌 merge를 사용하셔야 합니다.
자바 ORM 표준 JPA 프로그래밍 책 92p 엔티티의 생명주기 참고 부탁드려요.
외래키랑은 무슨 관련이 있는지는 저도 모르겠네요:)
감사합니다.