
트랜잭션 템플릿
- 트랜잭션 사용하는 로직은 트랜잭션 시작, 성공, 실패 등과 관련된 코드가 같은 패턴으로 반복이 된다
- 템플릿 콜백 패턴을 통해서 위와 같은 문제를 해결할 수 있음
- 스프링은
TransactionTemplate 라 는 템플릿 클래스를 제공한다.
public class TransactionTemplate {
private PlatformTransactionManager transactionManager;
public <T> T execute(TransactionCallback<T> action){..}
void executeWithoutResult(Consumer<TransactionStatus> action){..} }
execute() : 응답 값이 있을 때 사용한다.
executeWithoutResult() : 응답 값이 없을 때 사용한다.
예)
- 서비스
- 트랜잭션의 시작, 성공, 실패와 관련된 코드들이 모두 사라짐
@Slf4j
@RequiredArgsConstructor
public class MemberServiceV3_2 {
private final TransactionTemplate txTemplate;
private final MemberRepositoryV3 memberRepositoryV3;
public MemberServiceV3_2(PlatformTransactionManager transactionManager, MemberRepositoryV3 memberRepositoryV3) {
this.txTemplate = new TransactionTemplate(transactionManager);
this.memberRepositoryV3 = memberRepositoryV3;
}
public void accountTransfer(String fromId, String toId, int money) throws SQLException {
txTemplate.executeWithoutResult((status) -> {
try {
Member fromMember = memberRepositoryV3.findById(fromId);
Member toMember = memberRepositoryV3.findById(toId);
memberRepositoryV3.update(fromId, fromMember.getMoney() - money);
if(toMember.getMemberId().equals("ex")) {
throw new IllegalStateException("이체중 예외 발생");
}
memberRepositoryV3.update(toId, toMember.getMoney() + money);
} catch(SQLException e) {
throw new IllegalStateException(e);
}
});
}
}