[JPA] @Lock 비관적 락, PESSIMISTIC_WRITE, 타임아웃, 트랜잭션

2024. 2. 2. 13:29·JPA
목차
  1. 낙관적 락
  2. 비관적 락
  3. JPA 락 옵션
728x90

낙관적 락

  • @Version을 사용
  • 트랜잭션을 커밋하는 시점에 충돌을 알 수 있음

 

비관적 락

  • 선점 잠금
  • 데이터베이스 트랜잭션 락 메커니즘에 의존하는 방식
  • SQL 쿼리에 select for update 구문을 사용하면서 시작
  • 버전 정보는 사용하지 않음

 

JPA 락 옵션

낙관적 락 OPTIMISTIC 낙관적 락을 사용한다.
낙관적 락 OPTIMISTIC_FORCE_INCREMENT 낙관적 락 + 버전정보를 강제로 증가한다.
비관적 락 PESSIMISTIC_READ 비관적 락, 읽기 락을 사용한다. (공유 잠금)
비관적 락 PESSIMISTIC_WRITE 비관적 락, 쓰기 락을 사용한다.(배타적 잠금)
비관적 락 PESSIMISTIC_FORCE_INCREMENT 비관적 락 + 버전 정보를 강제로 증가한다.
기타 NONE 락을 걸지 않는다.
기타 READ JPA 1.0 호환 기능으로 OPTIMISTIC과 동일
기타 WRITE JPA 1.0 호환 기능으로 OPTIMISTIC_FORCE_INCREMENT와 동일

 


Entity

@Entity
@Getter
@Setter
@NoArgsConstructor
public class Order {
    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    private Long idx;

    private int price;

    public int decreasePrice(int price) {
        if (this.price - price < 0) {
            throw new IllegalArgumentException("가격 부족");
        }
        return this.price -= price;
    }
}

 
Repository

public interface OrderRepository extends JpaRepository<Order, Long> {

    Order findById(Long idx);

    @Lock(LockModeType.PESSIMISTIC_WRITE)
    @Query("select o from Order o where o.idx = :idx")
    Order findByIdxForUpdate(@Param("idx") Long idx);
}

 
service

@Service
@RequiredArgsConstructor
@Slf4j
public class OrderService {
  private final OrderRepository repository;

  @Transactional
  public int getCurrentPrice(Long idx) {
      Order order = homeRepository.findById(idx); // 조회
      return order.getPrice();
  }

  @Transactional
  public int decreasePrice(Long idx, int price) {
      Order order = homeRepository.findByIdxForUpdate(name); // 비관적락 조회
      order.decreasePrice(price); // 가격 감소
      return order.getPrice();
  }
}

 
controller

@RestController
@Slf4j
@RequiredArgsConstructor
@RequestMapping("/order")
public class HomeController {
  private final HomeService homeService;

  @GetMapping("/decrease")
  public String decreasePrice(@RequestParam(value = "idx") Long idx, @RequestParam(value = "price") int price) {
      String result;
      try {
          homeService.decreasePrice(idx, price);
          result = "현재 가격 : " + homeService.currentPrice(idx);
      } catch (Exception e) {
          result = e.getMessage();
      }
      log.info(result);
      return result;
  }
}

 

타임아웃

비관적 락을 사용하면 락을 획득할 때까지 트랜잭션이 대기하게 되는데 무한정 기다릴 수 없으므로 타임아웃 시간을 줄 수 있음
public interface OrderRepository extends JpaRepository<Order, Long> {

    @Lock(LockModeType.PESSIMISTIC_WRITE)
    @Query("select o from Order o where o.idx = :idx")
    @QueryHints({@QueryHint(name = "javax.persistence.lock.timeout", value = "10000")})
    Order findByIdxForUpdate(@Param("idx") Long idx);
}
728x90
  1. 낙관적 락
  2. 비관적 락
  3. JPA 락 옵션
'JPA' 카테고리의 다른 글
  • [JPA] 변경감지 Dirty Checking, @Transactional, @DynamicUpdate
  • [JPA] fetch join, EntityGraph, N+1, Pagination, firstResult/maxResults specified with collection fetch; applying in memory
  • [QueryDSL] join, paging, where, dto Qclass
Karla Ko
Karla Ko
𝘾𝙤𝙣𝙩𝙞𝙣𝙪𝙤𝙪𝙨𝙡𝙮 𝙄𝙢𝙥𝙧𝙤𝙫𝙞𝙣𝙜, 𝘾𝙤𝙣𝙨𝙩𝙖𝙣𝙩𝙡𝙮 𝘿𝙚𝙫𝙚𝙡𝙤𝙥𝙞𝙣𝙜 𝙔𝙚𝙨!
    250x250
  • Karla Ko
    karlaLog
    Karla Ko
  • 전체
    오늘
    어제
    • Total (467)
      • Spring (19)
      • JPA (4)
      • Cloud & Architecture (15)
        • Kubernetes (5)
        • Docker (3)
        • MSA (2)
        • GCP (1)
        • AWS (4)
      • Devops (1)
      • Message Queue (4)
        • Kafka (2)
        • RabbitMQ (2)
      • Git (4)
      • DB (4)
      • Java (9)
      • Python (4)
      • CS (11)
        • OS (8)
        • Network (2)
        • Algorithm (1)
      • Coding Test (392)
        • programmers (156)
        • Graph (43)
        • DP (37)
        • Search (31)
        • Tree (13)
        • Data Structure (26)
        • Combination (12)
        • Implement (18)
        • Geedy (23)
        • Sort (7)
        • Math (21)
        • geometry (2)
  • 블로그 메뉴

    • 홈
  • 링크

  • 공지사항

  • 인기 글

  • 태그

    힙
    덱
    플로이드워셜
    최단거리
    재귀
    이분탐색
    구간합
    스택
    최소신장트리
    파이썬
    Algorithm
    정렬
    그리디
    프로그래머스
    조합
    큐
    BFS
    LIS
    그래프
    월간코드챌린지
    DP
    다익스트라
    백준
    DFS
    구현
    트리
    동적계획법
    자료구조
    알고리즘
    최대공약수
  • hELLO· Designed By정상우.v4.10.3
Karla Ko
[JPA] @Lock 비관적 락, PESSIMISTIC_WRITE, 타임아웃, 트랜잭션
상단으로

티스토리툴바

단축키

내 블로그

내 블로그 - 관리자 홈 전환
Q
Q
새 글 쓰기
W
W

블로그 게시글

글 수정 (권한 있는 경우)
E
E
댓글 영역으로 이동
C
C

모든 영역

이 페이지의 URL 복사
S
S
맨 위로 이동
T
T
티스토리 홈 이동
H
H
단축키 안내
Shift + /
⇧ + /

* 단축키는 한글/영문 대소문자로 이용 가능하며, 티스토리 기본 도메인에서만 동작합니다.