작성
·
248
0
안녕하세요. 강의 예제대로 진행하면서
마지막에 완성된 item-service를 기초로 하여 H2 데이터베이스를 이용한 JPA 구현을 직접 해보려고했습니다.
기존 예제와 다른점은 Store라는 domain 클래스를 직접 만들어보고 Item과 매핑해보았습니다.
Item, Store 클래스에 대해 엔티티 설정을 해주었습니다.
먼저 /resources/META-INF/persistence.xml 을 생성하여 JPA 영속성 관련 설정을 해주었습니다.
그리고 EntityManagerFactory 클래스의 공유 변수를 위해 MyEntityManagerFactory 클래스를 만들었습니다.
컨트롤러, 서비스, 레포지토리 클래스에는 모두 각각
@Controller, @Service, @Repository 어노테이션을 추가하여 빈으로 등록 해주었고 빈으로 정상등록되는지도 확인했습니다.
먼저 문제점은 @Transactional 어노테이션이 정상적으로 적용이 안된다는 것입니다.
Item 객체를 추가하는 코드를 보면
이처럼 EntityManager에서 EntityTransaction을 만든 후
tx.begin()과 tx.commit()을 추가하여 실제 웹에서 테스트 해본결과 정상적으로 Item이 생성되며 DB까지 값이 들어갑니다.
위와 다르게 EntityTransaction 을 사용하지 않고 @Transactional 어노테이션을 이용하기 위해 코드를 변경했습니다.
먼저 서비스인 BasicItemService 클래스에 @Transactional 어노테이션을 추가했습니다.
그런 다음에 기존의 ItemRepository의 create 메소드의 tx 부분을 주석처리 했습니다.
제가 생각한 것은 서비스 클래스에서 @Transactional이 명시되어있기 때문에 레포지토리의 create 메소드 에서 em.persist(data)가 실행 된후 메소드가 종료 될 때 자동으로 commit이 될것이라고 생각했습니다. 하지만
실제로 아이템을 등록해보니 아래와 같은 오류가 발생했습니다.
2021-04-14 23:36:52.404 ERROR 22660 --- [nio-8081-exec-2] o.a.c.c.C.[.[.[/].[dispatcherServlet] : Servlet.service() for servlet [dispatcherServlet] in context with path [] threw exception [Request processing failed; nested exception is java.lang.IllegalArgumentException: Model has no value for key 'itemId'] with root cause
레포지토리에서 commit이 정상적으로 되지 않아서 AddForm에서 입력받은 데이터로 Item이 생성되지 않았고, 아이템이 생성되지 않았고 AddForm에서 Item 상세정보로 리다이렉트를 할때 itemId 값이 null이라서 오류가 발생했습니다.
사용하는 dependency는 다음과같습니다.
@Transactional 에 대해 여러 정보를 찾아봤는데 해결법을 찾지 못해서 질문드립니다!
답변 1
0
안녕하세요. Jungyu Choi님^^
JPA를 스프링과 함께 사용하면 스프링이 제공하는 설정들을 사용하거나 스프링 부트가 제공하는 설정을 사용해야 합니다.
JPA와 스프링을 함께 사용하는 방법에 대해서는
실전! 스프링 부트와 JPA 활용1 - 웹 애플리케이션 개발
강의를 참고해주세요.
감사합니다.