인프런 커뮤니티 질문&답변
컨트롤러에서find한 객체를 받았을 때
작성
·
312
0
강의 2번째 듣는중 궁금해져서 질문드립니다.
컨트롤러에서 find한 객체를 사용하게되면 더이상 영속상태가 아니기 때문에 id값만 받아서 서비스 transactional에서 find 하라고 말씀하셔서 궁금중이 생겨 컨트롤러에서find한 객체를 받아보았습니다. order()에서 find한 객체를 받아도 정상작동 되는건 이해가 되지만 set Name으로 이름을 바꿨을때 영속 상태가 아닌 member의 이름이 바뀌어서 db에 저장되는 이유가 무엇인가요? 파라미터로 받아도 영속상태는 유지되는 건가요?
@PostMapping("/order")
public String order(@RequestParam("memberId") Long memberId,@RequestParam("itemId")Long itemId
,@RequestParam("count")int count){
Member member = memberService.find(memberId);
orderService.order(itemId,count,member);
return "redirect:/orders";
}
@Transactional
public Long order(Long itemId, int count,Member member){
member.setName("change");
Item item = itemRepository.findOne(itemId);
Delivery delivery=new Delivery();
delivery.setAddress(findMember.getAddress());
OrderItem orderItem = OrderItem.createOrderItem(item, item.getPrice(), count);
Order order = Order.createOrder(member, delivery, orderItem);
orderRepository.Order(order);
return order.getId();
}
답변 1
0
안녕하세요. 김진현님 좋은 질문입니다.
컨트롤러에서 다음 로직을 실행하면 분명 트랜잭션이 없기 때문에 member가 영속 상태가 아니어야 합니다.
Member member = memberService.find(memberId);
그런데 이상하게 다음 로직을 실행하면 실제 update 쿼리가 나가면서 데이터가 변경이 되어 버리지요.
@Transactional
public Long order(Long itemId, int count,Member member){
member.setName("change");
}
그 이유는 바로 JPA의 다음 퍼즐 조각중 하나인 OSIV(Open Session In View)라고 불리는 기능 때문입니다.
스프링 부트는 이 기능을 기본으로 활성화 해두는데요.
이 기능은 고객의 HTTP 요청이 오는 시점부터, 요청이 종료될 때 까지 하나의 영속성 컨텍스트를 강제로 유지하는 방법입니다.
원래대로라면, 트랜잭션이 있는 곳에서만 영속성 컨텍스트가 유지되어야 하는데, 이 방법 때문에 트랜잭션과 무관하게 고객의 요청이 시작되는 시점부터 끝날 때 까지 하나의 영속성 컨텍스트가 살아있게 됩니다.
그래서 처음 컨트롤러에서 member를 조회한 시점에 member가 영속성 컨텍스트에서 관리되어 버립니다. 그리고 트랜잭션이 끝나는 시점에 영속성 컨텍스트에 플러시가 일어나면서, member에 변경감지가 동작합니다. member 엔티티가 변경되었기 때문에 최종적으로 update 쿼리가 실행되는 것이지요.
이런 OSIV 덕분에 트랜잭션이 없는 곳에서도 지연 로딩이 가능합니다. 하지만 데이터베이스 커넥션을 너무 오래 가지고 있는 등 실무에서는 문제가 많아서, 심사숙고해서 사용해야 하는 기능입니다.
그래서 최근 스프링 부트도 osiv가 기본으로 켜져있으니 조심하라고 경고를 남깁니다.
2020-12-03 22:06:12.868 WARN 12040 --- [ main] JpaBaseConfiguration$JpaWebConfiguration : spring.jpa.open-in-view is enabled by default. Therefore, database queries may be performed during view rendering. Explicitly configure spring.jpa.open-in-view to disable this warning
osiv를 끄려면 다음과 같이 설정하면 됩니다. yml에서는 . 단위로 2칸씩 띄워쓰기로 적어주시면 됩니다. 그러면 원하는 대로 동작하실 거에요.
spring.jpa.open-in-view=false
OSIV에 대한 더 자세한 내용은 활용2편, OSIV와 성능 최적화에서 다룹니다^^ (실무에서 JPA를 문제없이 다루려면 이 부분까지는 공부하시는 것을 권장드립니다.)
감사합니다.





