inflearn logo
강의

강의

N
챌린지

챌린지

멘토링

멘토링

N
클립

클립

로드맵

로드맵

지식공유

실전! 스프링 부트와 JPA 활용1 - 웹 애플리케이션 개발

컨트롤러에서find한 객체를 받았을 때

316

jhkim593

작성한 질문수 7

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();
}

spring java 웹앱 JPA spring-boot

답변 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를 문제없이 다루려면 이 부분까지는 공부하시는 것을 권장드립니다.)

감사합니다.

sdk 설정 오류

0

48

2

오탈자 - @Transactional

0

53

1

src/test/resources 테스트 경로 문제

0

49

1

상품 등록후 H2 db 출력 순서 바꿀 수 있나요?

0

62

1

MemberRepositoryTest 실행오류

0

77

1

boot 4.x >>> trasasction rolled back log & p6spy(영한님, 수업 자료 업데이트 해주시면 감사하겠습니다!!)

1

183

2

강의 마지막 QueryDSL 사용 부분 질문있습니다

1

137

2

클라이언트에서 isbn과 author 수정 요청을 한 경우에 대해 질문드립니다.

0

50

1

도메인 모델 패턴 vs 트랜잭션 스크립트 패턴

0

71

1

기본 생성자

0

59

1

h2 DB 연결시 jdbc url 변경 이유가 궁금합니다.

0

100

1

멤버서비스테스트 부분에서 막힙니다.

0

163

4

실무에서도 EntityManager를 이용해서 많이 작업하는 편일까요?

0

115

1

초반에 h2 다운로드 과정 꼭 필요한가요?

0

117

2

자신 필드에도 get으로 접근하는 이유가 있을까요?

0

110

1

24분 27초 연관관계 편의 메서드 위치

0

112

1

단건 주문만 가능하게 한건 의도한 부분이신가요?

0

107

2

빌드 툴, Gradle

0

57

1

h2연결은 된 것 같은데 엔티티 테이블까지 작성 후 확인해보아도 테이블이 안보입니다

0

73

2

Repository에서 EntityManager 주입 방식 차이

0

87

1

롬복과 사용자 정의 setter 메서드

0

70

1

주문 목록 조회 fetch join 질문드립니다

0

80

1

dirty checking 질문드립니다.

0

80

1

동시성 관련 질문입니다

0

72

1