Notice
Recent Posts
Recent Comments
Link
일 | 월 | 화 | 수 | 목 | 금 | 토 |
---|---|---|---|---|---|---|
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 |
Tags
- http
- Exception
- springdatajpa
- Thymeleaf
- 그리디
- transaction
- 김영한
- spring
- Android
- 스프링 핵심 기능
- 스프링
- 백준
- 스프링 핵심 원리
- AOP
- Proxy
- 인프런
- db
- QueryDSL
- JDBC
- SpringBoot
- java
- Greedy
- Spring Boot
- kotlin
- Servlet
- jpa
- 자바
- pointcut
- JPQL
- 알고리즘
Archives
- Today
- Total
개발자되기 프로젝트
트랜잭션 - DB 예제4 - 계좌이체 본문
세 가지 상황을 가정해보자.
- 계좌이체 정상
- 계좌이체 문제 상황 - 커밋
- 계좌이체 문제 상황 - 롤백
1. 계좌이체 정상
기본 데이터 입력 - SQL
set autocommit true;
delete from member;
insert into member(member_id, money) values ('memberA',10000);
insert into member(member_id, money) values ('memberB',10000);
- memberA 10000원
- memberB 10000원
계좌이체 실행
- memberA 의 돈을 memberB 에게 2000원 계좌이체하는 트랜잭션을 실행해보자.
- 다음과 같은 2번의 update 쿼리가 수행되어야 한다.
- set autocommit false 로 설정한다.
- 아직 커밋하지 않았으므로 다른 세션에는 기존 데이터가 조회된다.
계좌이체 실행 SQL - 성공
set autocommit false;
update member set money=10000 - 2000 where member_id = 'memberA';
update member set money=10000 + 2000 where member_id = 'memberB';
- commit 명령어를 실행하면 데이터베이스에 결과가 반영된다.
- 다른 세션에서도 memberA 의 금액이 8000원으로 줄어들고,
- memberB 의 금액이 12000원으로 증가한 것을 확인할 수 있다.
commit 전
commit 후
2. 계좌이체 문제 상황 - 커밋
계좌이체 중 문제가 발생했으나 commit하는 상황
기본 데이터 입력 - SQL
set autocommit true;
delete from member;
insert into member(member_id, money) values ('memberA',10000);
insert into member(member_id, money) values ('memberB',10000);
- memberA 10000원
- memberB 10000원
계좌이체 실행
- 계좌이체를 실행하는 도중에 SQL에 문제가 발생한다.
- 그래서 memberA 의 돈을 2000원 줄이는 것에는 성공했지만, memberB 의 돈을 2000원 증가시키는 것에 실패한다.
- 두 번째 SQL은 member_iddd 라는 필드에 오타가 있다.
- 두 번째 update 쿼리를 실행하면 SQL 오류가 발생하는 것을 확인할 수 있다.
계좌이체 실행 SQL - 오류
set autocommit false;
update member set money=10000 - 2000 where member_id = 'memberA'; //성공
update member set money=10000 + 2000 where member_iddd = 'memberB'; //쿼리 예외발생
두 번째 SQL 실행시 발생하는 오류 메시지
- 여기서 문제는 memberA 의 돈은 2000원 줄어들었지만,
- memberB 의 돈은 2000원 증가하지 않았다는 점이다.
- 결과적으로 계좌이체는 실패하고 memberA 의 돈만 2000원 줄어든 상황이다.
강제 커밋
- 만약 이 상황에서 강제로 commit 을 호출하면 어떻게 될까?
- 계좌이체는 실패하고 memberA 의 돈만 2000원 줄어드는 아주 심각한 문제가 발생한다.
- 돈 증발됨. ㄷ
- 이렇게 중간에 문제가 발생했을 때는 커밋을 호출하면 안된다.
- 롤백을 호출해서 데이터를 트랜잭션 시작 시점으로 원복해야 한다.
3. 계좌이체 문제 상황 - 롤백
중간에 문제가 발생했을 때 롤백을 호출해서 트랜잭션 시작 시점으로 데이터를 원복해보자.
기본 데이터 입력 - SQL
set autocommit true;
delete from member;
insert into member(member_id, money) values ('memberA',10000);
insert into member(member_id, money) values ('memberB',10000);
계좌이체 실행하나, B는 실패
set autocommit false;
update member set money=10000 - 2000 where member_id = 'memberA'; //성공
update member set money=10000 + 2000 where member_iddd = 'memberB'; //쿼리 예외발생
세션1에서 rollback
- 이럴 때는 롤백을 호출해서 트랜잭션을 시작하기 전 단계로 데이터를 복구해야 한다.
- 롤백을 사용한 덕분에 계좌이체를 실행하기 전 상태로 돌아왔다.
- memberA 의 돈도 이전 상태인 10000 원으로 돌아오고,
memberB 의 돈도 10000 원으로 유지되는 것을 확인할 수 있다.
정리
- 원자성
트랜잭션 내에서 실행한 작업들은 마치 하나의 작업인 것처럼 모두 성공 하거나 모두 실패해야 한다.
트랜잭션의 원자성 덕분에 여러 SQL 명령어를 마치 하나의 작업인 것 처럼 처리할 수 있었다.
성공하면 한번에 반영하고, 중간에 실패해도 마치 하나의 작업을 되돌리는 것 처럼 간단히 되돌릴 수 있다. - 오토 커밋
만약 오토 커밋 모드로 동작하는데, 계좌이체 중간에 실패하면 어떻게 될까?
쿼리를 하나 실행할 때 마다 바로바로 커밋이 되어버리기 때문에
memberA 의 돈만 2000원 줄어드는 심각한 문제가 발생한다. - 트랜잭션 시작
따라서 이런 종류의 작업은 꼭 수동 커밋 모드를 사용해서 수동으로 커밋, 롤백 할 수 있도록 해야 한다.
보통 이렇게 자동 커밋 모드에서 수동 커밋 모드로 전환 하는 것을 트랜잭션을 시작한다고 표현한다.
'인프런 > [인프런] 스프링 DB 1편 - 데이터 접근 핵심 원리' 카테고리의 다른 글
DB 락 - 변경 (0) | 2022.06.02 |
---|---|
DB 락 - 개념 이해 (0) | 2022.06.01 |
트랜잭션 - DB 예제3 - 트랜잭션 실습 (0) | 2022.06.01 |
트랜잭션 - DB 예제2 - 자동 커밋, 수동 커밋 (0) | 2022.06.01 |
트랜잭션 - DB 예제1 - 개념 이해 (0) | 2022.06.01 |
Comments