inflearn logo
강의

강의

N
챌린지

챌린지

멘토링

멘토링

N
클립

클립

로드맵

로드맵

지식공유

스프링 DB 2편 - 데이터 접근 활용 기술

트랜잭션 AOP 주의 사항 - 프록시 내부 호출2

트랜잭션 AOP주의사항 - 해결방법 문의

698

고래밥

작성한 질문수 9

0

[질문 템플릿]
1. 강의 내용과 관련된 질문인가요? 예
2. 인프런의 질문 게시판과 자주 하는 질문에 없는 내용인가요? 예
3. 질문 잘하기 메뉴얼을 읽어보셨나요? 예

[질문 내용]
영한님께서 이번 강의에서 트랜잭션 프록시 객체에서 외부메서드를 호출하고, 진짜 객체의 외부 메서드에서 자기 자신의 내부 메서드를 호출할 때 내부 메서드에는 @Transactional을 설정해도 트랜잭션이 적용되지 않는 점 이해 했습니다.

 

강의에서 이 부분을 해결하기 위해 별도의 클래스를 만들고, Bean으로 주입(이 경우 프록시객체가 Bean으로 등록)해서 프록시 객체가 @Transactional 메서드를 호출하는 방식으로 해결하는 점도 이해했습니다.

 

스프링 MVC강의에서 빈 스코프 문제(프로토타입과 싱글톤 Bean을 같이 사용할 때 문제점)을 해결하기 위해 Dependency Lookup을 사용하신 방법이 떠올라 V3테스트코드를 작성해보았습니다.

 

@Slf4j
@SpringBootTest
public class InternalCallV3Test {

    @Autowired
    CallServie callServie;

    @Test
    void printProxy() {
        log.info("callService class={}", callServie.getClass());
    }

    @Test
    void internalCall() {
        callServie.internal();
    }

    @Test
    void externalCall() {
        callServie.external();
    }

    @TestConfiguration
    static class InternalCallV1TestConfig {
        @Bean
        CallServie callServie() {
            return new CallServie();
        }
    }

    @Slf4j
    static class CallServie {

        @Autowired
        private ObjectProvider<CallServie> selfProvider;

        public void external() {
            log.info("call external");
            printTxInfo();
            CallServie self = selfProvider.getObject();
            log.info("self class={}", self.getClass());
            self.internal();
        }

        @Transactional
        public void internal() {
            log.info("call internal");
            printTxInfo();
        }

        private void printTxInfo() {
            boolean txActive = TransactionSynchronizationManager.isActualTransactionActive();
            log.info("tx active={}", txActive);
            boolean readOnly = TransactionSynchronizationManager.isCurrentTransactionReadOnly();
            log.info("tx readOnly={}", readOnly);
        }
    }
}

 

실행 결과는 다음과 같습니다.

DL을 이용해서 스프링 컨테이너에서 프록시를 가져와서, 프록시의 메서드를 호출하면 트랜잭션이 적용되는 것 처럼 보입니다.

 

1) 이 방법으로 트랜잭션이 적용되는게 맞을까요?

2) 이 방법을 혹시 스프링-고급편에서 다루나요?

3) 1번 질문이 맞다면 클래스를 새로 만들고, 메서드를 옮기는 비용보다 DL로 프록시객체를 찾아오는 비용이 더 쌀 것 같은데, 둘 중 어느 방법이 선호되는 편인가요?

 

spring

답변 2

0

김영한

안녕하세요. 고래밥님^^

해당 방법도 해결방안 중에 하나입니다.

하지만 너무 복잡해지니, 이번 강의에서 설명드린 방법을 더 권장드립니다.

참고로 해당 내용은 고급편에서 더 자세히 설명드립니다.

감사합니다.

0

고래밥

감사합니다 선생님 ^-^

0

인프런 AI 인턴

안녕하세요, 인프런 AI 입니다.

해당 내용을 보면 트랜잭션 AOP를 사용할 때 외부메서드와 내부메서드를 구분해서 AOP가 적용되는 문제점이 있었다는 것 같습니다. 이러한 문제점은 외부 메서드에서 내부 메서드를 호출할 때 내부 메서드의 @Transactional 어노테이션이 적용되지 않아 발생합니다.

해당 문제를 해결하기 위한 방법으로는 별도의 클래스를 만들고, @Bean으로 등록하는 방법이 있었습니다.

또한, 트랜잭션 AOP 문제와 관련하여 Dependency Lookup 방법을 사용하신 것 같습니다. 이 방법은 스프링 MVC 강의에서 빈 스코프 문제를 해결하는데 사용되는 방법입니다.

V3 테스트 코드를 작성하시면서 프록시 객체가 생성되는지 확인하신 것 같습니다. 이 부분에 대한 세부적인 내용은 제공되지 않아 정확한 답변을 드리지는 못하는 점 양해 부탁드립니다.

더 자세한 내용은 해당 강의의 커뮤니티 게시판에 질문을 남겨보시는 것이 좋을 것 같습니다. 감사합니다.

설정 정보 없이 임베디드 데이터베이스 생성

0

15

1

RepositoryTest의 패키지 위치가 domain인 이유

0

37

2

REQUIRES_NEW 해결 방법에 대해서 질문있습니다!!

0

32

1

update()에 사용하는 setter 질문드립니다.

0

49

1

SQL 중심적 개발의 문제점에 대한 질문

0

78

1

혹시 Containing 을 안쓰신 이유가 있을까요?

0

89

2

[공유] 스프링부트 4.x 버전 mybatis 연동

0

184

1

@repository 어노테이션

0

93

3

ItemService

0

59

1

논리 커밋, 물리 커밋 질문드립니다.

0

54

1

내부 트랜잭션 커밋은 필수인가요?

0

57

1

프록시 커넥션 객체를 반환할 때 생성하는건가요?

0

55

1

Transaction readOnly 성능 개선 (김영한님의 대한 감사인사)

2

180

2

JPQL 대신 네이티브 쿼리를 사용해야 하는 경우

0

81

1

@EventListener(ApplicationReadyEvent.class) 관련

0

90

1

트랜잭션 동기화 매니저와 데이터 소스

0

77

1

DB 관련 강의 개설 계획은 없으신건가요?

0

133

2

물리 트랜잭션 과 논리트랜잭션 용어를 맞게 이해한걸까요

0

95

1

스프링 3 버전 이상 rollbackFor 변경된듯요

1

114

1

트랜잭션 전파 질문.

0

87

1

프로젝트 오픈 에러

0

126

1

외부 트랜잭션에서 isNewTransaction이 false로 나오는거에 대해 질문드립니다

0

84

2

같은 스레드를 사용하면 트랜잭션 동기화 매니저는 같은 커넥션을 반환

0

74

1

h2 인메모리 테스트중 예약어 충돌날 경우 대처방법

0

105

1