티스토리 뷰
목차
Spring MVC - Controller + Service
[Spring]Spring MVC - Controller 클래스 구조 생성 및 설계
[Spring]Spring MVC - Controller 클래스에 핸들러 메서드(Handler Method) 구현
[Spring]Spring MVC - Controller 클래스에 ResponseEntity 적용
[Spring]Spring MVC - Controller 클래스에 DTO 적용
[Spring]Spring MVC - DTO 유효성 검증(Validation)
[Spring]Spring MVC - DI를 통한 API 계층 ↔ 서비스 계층 연동
[Spring]Spring MVC - 매퍼(Mapper)를 이용한 DTO 클래스 ↔ 엔티티(Entity) 클래스 매핑
[Spring]Spring MVC - @ExceptionHandler를 이용한 예외처리
[Spring]Spring MVC - @RestControllerAdvice를 이용한 예외처리
Spring Data JDBC
[Spring]JDBC(Java DataBase Connectivity)
[Spring]Spring Data JDBC, Spring Data JDBC 사용법
[Spring]Spring Data JDBC - 도메인 엔티티&테이블 설계
Spring Data JPA
[Spring]JPA(Java Persistence API)
[Spring]JPA - Entity ↔ DB Table Mapping
[Spring]JPA - Entity ↔ Entity Mapping
[Spring]Spring Data JPA - 데이터 액세스 계층 구현
지난 글에서 트랜잭션의 정의와 규칙, 구체적인 WorkFlow에 대해 알아보았다.
2022.09.05 - [개발/Spring] - [Spring]트랜잭션(Transaction)
요약하면 트랜잭션이란 모두 성공하거나 모두 실패하는(All or Nothing), 쪼갤 수 없는 작업의 최소 단위이며
ACID Principles에 의해 안정성을 보장받고 있다.
또한 Commit과 Rollback 명령어로 제어가 가능하며
commit()의 경우 내부적으로 상당히 복잡한 과정을 거쳐 호출된다.
이번 글에서는 Spring Boot 환경에서 선언형 방식으로 트랜잭션을 적용하는 두 가지 방식에 대해 살펴본다.
관련 설정은 Spring Boot가 내부적으로 처리해주기 때문에, 여기서는 바로 트랜잭션을 적용한다.
시작하기 전에 트랜잭션 과정이 로그에 찍히도록 application.yml에 다음과 같은 설정을 추가한다.
...
logging:
level:
org:
springframework:
orm:
jpa: DEBUG
@Transactional
스프링에서 트랜잭션을 적용하는 가장 간단한 방법은 필요한 영역에 @Transactional을 추가하는 것이다.
...
import org.springframework.transaction.annotation.Transactional;
@Service
@Transactional // 추가
public class MemberService {
private final MemberRepository memberRepository;
public MemberService(MemberRepository memberRepository) {
this.memberRepository = memberRepository;
}
...
}
위와 같이 클래스 레벨에 @Transactional을 추가하면 해당 클래스에서
MemberRepository를 사용하는 모든 메서드에 트랜잭션이 적용된다.
commit 동작 여부 확인
애플리케이션을 실행시키고 Postman을 통해 회원 정보를 입력하면 아래와 같은 로그가 출력된다.
2022-09-06 11:08:29.014 DEBUG 55436 --- [nio-8080-exec-1] o.s.orm.jpa.JpaTransactionManager
: Creating new transaction[...MemberService.createMember]: PROPAGATION_REQUIRED,ISOLATION_DEFAULT
: Initiating transaction commit
: Committing JPA transaction on EntityManager [SessionImpl(36635712<open>)]
: Not closing pre-bound JPA EntityManager after transaction
: Closing JPA EntityManager in OpenEntityManagerInViewInterceptor
createMember() 메서드가 호출되며 트랜잭션이 생성, commit() 후 종료되는 것을 확인할 수 있다.
rollback 동작 여부 확인
계속해서 예외를 발생시킨 뒤 rollback이 작동하는 로그를 확인하자.
...
import org.springframework.transaction.annotation.Transactional;
@Service
@Transactional
public class MemberService {
...
public Member createMember(Member member) {
verifyExistsEmail(member.getEmail());
Member resultMember = memberRepository.save(member);
if (true) throw new RuntimeException("Rollback Test"); // 예외 발생
return resultMember;
}
...
}
회원 정보를 저장하고 결과를 리턴하기 전 강제로 RuntimeException을 발생시켰다.
동일하게 Postman을 통해 요청을 보내면 이번엔 이런 로그가 뜬다.
2022-09-06 11:24:15.884 DEBUG 56455 --- [nio-8080-exec-2] o.s.orm.jpa.JpaTransactionManager
: Initiating transaction rollback
: Rolling back JPA transaction on EntityManager [SessionImpl(1661184085<open>)]
: Not closing pre-bound JPA EntityManager after transaction
2022-09-06 11:24:15.951 ERROR 56455 --- [nio-8080-exec-2] c.c.m.s.advice.GlobalExceptionAdvice
: # handle Exception
java.lang.RuntimeException: Rollback Test
2022-09-06 11:24:16.350 DEBUG 56455 --- [nio-8080-exec-2] o.j.s.OpenEntityManagerInViewInterceptor
: Closing JPA EntityManager in OpenEntityManagerInViewInterceptor
예외에 따라 rollback이 잘 작동하는 것을 확인할 수 있다.
체크 예외(Checked Exception)의 Rollback
@Transactional은 기본적으로 예상하지 못한 Unchecked Exception, Error 만을 Rollback 한다.
즉, SQLException, DataFormatException과 같은 Checked Exception은
@Transactional 만으로는 Rollback이 동작하지 않는다.
Checked Exception의 경우 예상되는 예외이기 때문에 catch 후 예외 전략을 수립해야 하며, 별도의 전략이 필요 없을 시
@Transactional(rollbackFor = Exception.class)
과 같이 해당 예외를 지정하거나 Unchecked Exception으로 감싸주어야 한다.
Method Level에 @Transactional 적용
이번에는 클래스 레벨과 함께 메서드 레벨에도 @Transactional을 적용해보자.
...
import org.springframework.transaction.annotation.Transactional;
@Service
@Transactional // 클래스 레벨에 적용
public class MemberService {
...
@Transactional(readOnly = true) // 메서드 레벨에 적용
public Member findMember(long memberId) {
return findVerifiedMember(memberId);
}
...
}
findMember() 메서드 레벨에도 @Transactional이 적용된 것을 확인할 수 있다.
(readOnly = true) 속성은 해당 트랜잭션을 읽기 전용으로 설정한다.
2022-09-06 11:49:06.957 DEBUG 57607[...MemberService.findMember]
: PROPAGATION_REQUIRED,ISOLATION_DEFAULT,readOnly
: Committing JPA transaction on EntityManager [SessionImpl(574318919<open>)]
실행시키고 로그를 확인하면 트랜잭션 설정이 readOnly인 것을 볼 수 있다.
@Transactional(readOnly = true)
위 로그에서 또 하나 확인할 수 있는 건 읽기 전용임에도 commit을 진행한다는 점이다.
그럼에도 불구하고 DB의 상태가 변하지 않는 이유는, 간단하게도 flush를 진행하지 않기 때문이다.
flush는 이전에 JPA에 대해 처음 알아보던 글에서 등장한 적이 있는데,
2022.08.31 - [개발/Spring] - [Spring]JPA(Java Persistence API)
flush란 한 마디로 말하면 영속성 컨텍스트와 DB의 동기화 과정이다.
따라서 읽기 전용 트랜잭션은 commit을 진행하지만 DB의 상태가 변하지 않으며,
이를 대략적으로 나타내면 아래 그림과 같다.
Class Level / Method Level @Transactional 적용 순서
- 클래스 레벨에만 적용된 경우 - 해당 클래스의 메서드에 @Transactional 일괄 적용
- 클래스 레벨과 메서드 레벨에 함께 적용된 경우 - 메서드 레벨 → 클래스 레벨 순으로 @Transactional 적용
여러 작업을 하나의 트랜잭션으로 묶기
위 그림은 커피 주문 앱에서 주문 정보를 DB에 저장하는 상황이다.
주문 정보를 저장하는 작업과 스탬프를 업데이트하는 작업이 독립적으로 진행되고 있는데,
이때 updateStamp()에서 예외가 발생할 경우 주문 정보는 저장되지만 스탬프는 업데이트되지 않는 상황이 벌어질 것이다.
따라서 위 그림처럼 두 개의 작업을 하나의 트랜잭션으로 묶을 필요성이 생기는데,
이 경우 updateStamp()에서 예외가 발생하면 두 클래스의 작업이 묶여있으므로 한 번에 Rollback 처리가 된다.
코드와 동작 로그를 확인하자.
...
@Service
@Transactional
public class OrderService {
...
public Order createOrder(Order order) {
verifyOrder(order);
Order savedOrder = saveOrder(order);
updateStamp(savedOrder); // updateStamp() 호출 -> Transaction Binding
throw new RuntimeException("Rollback Test"); // 예외 발생
// return savedOrder;
}
public void updateStamp(Order order) {
Member member = memberService.findMember(order.getMember().getMemberId());
int stampCount = calculateStampCount(order);
Stamp stamp = member.getStamp();
stamp.setStampCount(stamp.getStampCount() + stampCount);
member.setStamp(stamp);
memberService.updateMember(member);
}
private int calculateStampCount(Order order) {
return order.getOrderCoffees().stream()
.map(orderCoffee -> orderCoffee.getQuantity())
.mapToInt(quantity -> quantity)
.sum();
}
private Order saveOrder(Order order) {
return orderRepository.save(order);
}
...
}
...
@Service
@Transactional
public class MemberService {
private final MemberRepository memberRepository;
...
@Transactional(propagation = Propagation.REQUIRED) // Propagation 속성 지정
public Member updateMember(Member member) {
Member findMember = findVerifiedMember(member.getMemberId());
Optional.ofNullable(member.getName())
.ifPresent(name -> findMember.setName(name));
Optional.ofNullable(member.getPhone())
.ifPresent(phone -> findMember.setPhone(phone));
Optional.ofNullable(member.getMemberStatus())
.ifPresent(memberStatus -> findMember.setMemberStatus(memberStatus));
return memberRepository.save(findMember);
}
@Transactional(readOnly = true)
public Member findMember(long memberId) {
return findVerifiedMember(memberId);
}
...
}
2022-09-06 14:24:25.020 DEBUG 63635 --- [nio-8080-exec-5] o.s.orm.jpa.JpaTransactionManager
: Initiating transaction rollback
: Rolling back JPA transaction on EntityManager [SessionImpl(142478777<open>)]
: Not closing pre-bound JPA EntityManager after transaction
: # handle Exception
java.lang.RuntimeException: Rollback Test
2022-09-06 14:24:25.090 DEBUG 63635 --- [nio-8080-exec-5] o.j.s.OpenEntityManagerInViewInterceptor
: Closing JPA EntityManager in OpenEntityManagerInViewInterceptor
앱을 실행시키고, 회원과 커피 정보를 등록한 후 주문을 넣으면 위와 같은 로그를 확인할 수 있다.
두 작업이 OrderService에서 시작된 트랜잭션으로 묶여있기 때문에 모두 Rollback 처리된다.
트랜잭션 전파(Transaction Propagation)
트랜잭션 전파란 위 경우처럼 특정 트랜잭션 내부에서 다른 트랜잭션을 실행할 경우 처리하는 방식을 의미한다.
@Transactional 애너테이션의 propagation 속성으로 지정할 수 있으며,
대표적으로 아래와 같은 유형의 전파가 존재한다.
종류 | 설명 | 다이어그램 |
REQUIRED(default) | 부모 트랜잭션이 존재하면 합류 없다면 새로운 트랜잭션 생성 예외 발생시 전체 Rollback |
|
REQUIRES_NEW | 무조건 새로운 트랜잭션 생성 기존의 트랜잭션은 새 트랜잭션 종료까지 정지 |
|
MANDATORY | 부모 트랜잭션이 존재하면 합류 없다면 예외 발생 |
|
NOT_SUPPORTED | 부모 트랜잭션이 있다면 메서드 이후까지 보류 없다면 트랜잭션 생성하지 않음 |
|
NEVER | 부모 트랜잭션이 있다면 예외 발생 없다면 트랜잭션 생성하지 않음 |
트랜잭션 고립 수준(Isolation Level)
트랜잭션 고립 수준은 ACID Princlples의 Isolation 원칙을 조정할 수 있는 속성이다.
@Transactional(isolation = Isolation.DEFAULT)과 같이 사용한다.
조금 더 정확하게는 여러 개의 트랜잭션이 실행 중일 때 특정 트랜잭션이 다른 트랜잭션에 접근할 수 있는 정도를 나타낸다.
대략 아래와 같은 종류가 존재한다.
종류 | 설명 |
REPEATABLE_READ(default) | SELECT시 현재 시점의 스냅샷을 만들고 조회한다. 즉, 반복해서 조회해도 같은 결과를 얻는다. 트랜잭션 내에서 일관성을 보장한다. |
READ_UNCOMMITTED | Commit 되지 않은 데이터에 다른 트랜잭션이 접근할 수 있다. Commit 이나 Rollback에 상관없이 현재의 데이터를 읽어온다. Rollback 될 데이터도 읽어올 수 있으므로 주의가 필요하다. |
READ_COMMITTED | Commit 된 데이터에 다른 트랜잭션이 접근할 수 있다. 구현 방식의 차이로 Query 수행 시점의 데이터와 정확하게 일치하지 않을 수 있다. 대량의 데이터를 이동하거나 복제할 때 사용된다. |
SERIALIZABLE | 동일 데이터에 대해 두 개 이상의 트랜잭션이 수행될 수 없다. 가장 단순하고 엄격한 격리수준이다. |
AOP
AOP는 위에 알아본 @Transactional을 사용하지 않고도 트랜잭션을 적용하는 방법이다.
CoffeeService에 적용하는 예를 확인하자.
상위 폴더에 config 패키지를 생성해 트랜잭션 Config 파일을 생성한다.
package com.codestates.config;
import org.springframework.aop.Advisor;
import org.springframework.aop.aspectj.AspectJExpressionPointcut;
import org.springframework.aop.support.DefaultPointcutAdvisor;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.transaction.TransactionDefinition;
import org.springframework.transaction.TransactionManager;
import org.springframework.transaction.interceptor.*;
import java.util.HashMap;
import java.util.Map;
@Configuration // @Bean 등록을 위한 애너테이션
public class TxConfig {
private final TransactionManager transactionManager;
public TxConfig(TransactionManager transactionManager) { // 트랜잭션 적용을 위해 TransactionManager 객체 DI
this.transactionManager = transactionManager;
}
@Bean
public TransactionInterceptor txAdvice() { // 트랜잭션 어드바이스용 TransactionInterceptor 빈 등록
NameMatchTransactionAttributeSource txAttributeSource =
new NameMatchTransactionAttributeSource();
RuleBasedTransactionAttribute txAttribute =
new RuleBasedTransactionAttribute(); // 조회 메서드를 제외한 공통 트랜잭션 속성
txAttribute.setPropagationBehavior(TransactionDefinition.PROPAGATION_REQUIRED);
RuleBasedTransactionAttribute txFindAttribute =
new RuleBasedTransactionAttribute(); // 조회 메서드를 위한 트랜잭션 속성
txFindAttribute.setPropagationBehavior(TransactionDefinition.PROPAGATION_REQUIRED);
txFindAttribute.setReadOnly(true);
Map<String, TransactionAttribute> txMethods = new HashMap<>(); // 트랜잭션 속성 매핑
txMethods.put("find*", txFindAttribute); // Map의 key를 메서드 이름 패턴으로 지정
txMethods.put("*", txAttribute);
txAttributeSource.setNameMap(txMethods); // 위의 Map 객체를 txAttributeSource()에게 전달
return new TransactionInterceptor(transactionManager, txAttributeSource);
// TransactionInterceptor의 생성자 파라미터로 transactionManager, txAttributeSource 지정
}
@Bean
public Advisor txAdvisor() {
AspectJExpressionPointcut pointcut = new AspectJExpressionPointcut(); // 포인트 컷 지정
pointcut.setExpression("execution(* com.codestates.coffee.service." + "CoffeeService.*(..))");
return new DefaultPointcutAdvisor(pointcut, txAdvice());
// Advisor 객체 생성 및 파라미터로 포인트컷과 어드바이스 지정
}
}
이어서 CoffeeService 클래스의 createCoffee()를 아래와 같이 수정하고
public Coffee createCoffee(Coffee coffee) {
String coffeeCode = coffee.getCoffeeCode().toUpperCase();
verifyExistCoffee(coffeeCode);
coffee.setCoffeeCode(coffeeCode);
if (true) throw new RuntimeException("Rollback Test");
return coffeeRepository.save(coffee);
}
요청을 보내면 Rollback이 실행되는 것을 쉽게 확인할 수 있다.
'Java+Spring > Spring' 카테고리의 다른 글
[Spring]Hamcrest를 적용한 단위 테스트 (2) | 2022.09.08 |
---|---|
[Spring]JUnit을 이용한 비즈니스 로직 단위 테스트 (2) | 2022.09.07 |
[Spring]단위 테스트(Unit Test) (1) | 2022.09.07 |
[Spring]트랜잭션(Transaction) (0) | 2022.09.05 |
[Spring]ORM, Hibernate, JPA, Spring Data JPA (0) | 2022.09.02 |
[Spring]Spring Data JPA - 데이터 액세스 계층 구현 (0) | 2022.09.02 |
- Total
- Today
- Yesterday
- 기술면접
- Backjoon
- 여행
- RX100M5
- 유럽
- 자바
- 남미
- 맛집
- 동적계획법
- 알고리즘
- 리스트
- BOJ
- 야경
- 지지
- 유럽여행
- 스트림
- 파이썬
- 칼이사
- Algorithm
- a6000
- 세모
- java
- spring
- 스프링
- 백준
- 면접 준비
- Python
- 세계여행
- 세계일주
- 중남미
일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
1 | 2 | 3 | 4 | |||
5 | 6 | 7 | 8 | 9 | 10 | 11 |
12 | 13 | 14 | 15 | 16 | 17 | 18 |
19 | 20 | 21 | 22 | 23 | 24 | 25 |
26 | 27 | 28 | 29 | 30 | 31 |