본문 바로가기

Spring/JPA

JPA- Open Session In View

OSIV

 기본적으로 스프링 프로젝트를 실행하면, warn 로그가 뜨면서 spring.jpa.open-in-view : true 기본값이라고 알려준다.

 기본적으로 OSIV 전략은 최초 데이터베이스 커넥션 시작 시점부터 View, API 응답이 끝날 때까지 영속성 컨텍스트와 데이터베이스 커넥션을 유지한다.

@PostMapping("/api/v1/members")
public CreateMemberResponse saveMemberV1(@RequestBody @Valid Member member) {
    Long id = memberService.join(member);
    return new CreateMemberResponse(id);
}
@Transactional
public Long join(Member member) {
    validateDuplicateMember(member); // 중복 회원 가입 방지
    memberRepository.save(member);
    return member.getId();
}

 JPA에서는 서비스 계층에서 트랜잭션이 시작될 때, 데이터베이스 커넥션이 시작된다. 또한 이렇게 이어진 데이터 베이스 커넥션은 OSIV 전략을 사용하는 경우 해당 레포지터리, 서비스 계층 메서드(join)이 끝나더라도 DB 커넥션이 반환되지 않고, API의 경우 API가 종료될 때 까지 View의 경우 유저에게 완전히 반환될 때(html 렌더링 완전히 끝나서 유저에게 response가 반환될 때 까지) 까지 유지된다.

그러므로 View template이나 API에서 지금껏 지연로딩이 가능했었던 것이다.

 지연 로딩은 영속성 컨텍스트가 살아있어야 가능하고, 영속성 컨텍스트는 기본적으로 데이터베이스 커넥션을 유지한다. 이것 자체가 큰 장점이다.

OSIV 문제점

 커넥션이 오랜시간 계속해서 유지가 되기 때문에 이는 잘못하면 데이터베이스 커넥션 풀의 부족으로 이어질 수 도 있다. 예컨데 컨트롤러에서 외부 API를 호출하면 외부 API 대기 시간만큼 커넥션 리소스를 반환하지 못하고, 이를 유지해야 한다. 이와 같은 커넥션의 유지로 인한 커넥션 풀의 부족은 많은 커넥션을 요구하는 실시간 어플리케이션의 경우 장애로 이어질 수 있다.

OSIV 설정하기

application.yml

spring:
  datasource:
    url: jdbc:h2:tcp://localhost/~/jpashop;
    username: sa
    password:
    driver-class-name: org.h2.Driver

  jpa:
    hibernate:
      ddl-auto: create
    properties:
      hibernate:
        format_sql: true
        default_batch_fetch_size: 100 
    open-in-view: false  # OSIV 설정 끔
logging:
  level:
    org.hibernate.SQL: debug  
    org.hibernate.type: trace

jpa: open-in-view 설정을 false로 설정함으써 OSIV 설정을 false로 둘 수 있다.

OSIV를 false로 설정한 경우

OSIV를 끄면 트랜잭션을 종료할 때 영속성 컨텍스트를 닫고, 데이터베이스 커넥션도 반환한다. 따라서 커넥션 리소스를 낭비하지 않는다.
OSIV
를 끄면 모든 지연로딩을 트랜잭션 안에서 처리해야 한다. 따라서 지금까지 작성한 많은 지연 로딩 코드를 트랜잭션 안으로 넣어야 하는 단점이 있다. 그리고 view template에서 지연로딩이 동작하지 않는다. 결론적으로 트랜잭션이 끝나기 전에 지연 로딩을 강제로 호출해 두어야 한다.

 

 

'Spring > JPA' 카테고리의 다른 글

JPA- DISTINCT  (0) 2023.07.10
JPA- 벌크연산  (0) 2023.06.19
JPA-Join Fetch  (0) 2023.06.18
JPA- 경로 표현식  (0) 2023.06.18
JPA- 조건식의 사용  (0) 2023.06.18