inflearn logo
강의

강의

N
챌린지

챌린지

멘토링

멘토링

N
클립

클립

로드맵

로드맵

지식공유

스프링 핵심 원리 - 고급편

로그 추적기 V2 - 파라미터로 동기화 개발

createNextId()시 새로운 객체를 반환하는 이유가 있나요?

해결된 질문

190

nove1080

작성한 질문수 8

0

public TraceStatus beginSync(TraceId beforeTraceId, String message) {
    TraceId nextId = beforeTraceId.createNextId();
    Long startTimeMs = System.currentTimeMillis();
    log.info("[{}] {}{}", nextId.getId(), addSpace(START_PREFIX,
        nextId.getLevel()), message);
    return new TraceStatus(nextId, startTimeMs, message);
}

위 코드를 아래와 같이 사용하면 어떤 단점이 있나요?

public TraceStatus beginSync(TraceId traceId, String message) {
    traceId.increaseId();
    Long startTimeMs = System.currentTimeMillis();
    log.info("[{}] {}{}", traceId.getId(), addSpace(START_PREFIX,
        traceId.getLevel()), message);
    return new TraceStatus(traceId, startTimeMs, message);
}

객체를 새로 만들어서 반환하지 않고 id만 증가시켜서 그대로 사용하는 것은 지양되어야 하나요?

spring 디자인-패턴

답변 1

2

인프런 AI 인턴

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

nove1080님께서 질문하신 createNextId()에서 새로운 객체를 반환하는 이유와 객체를 새로 만들지 않고 id만 증가시켜 사용하는 것에 대한 단점에 대하여 설명드립니다.

객체를 새로 생성하여 반환하는 이유는 불변 객체(Immutable Object) 패턴을 따르기 위해서입니다. 불변 객체는 객체가 생성된 후 그 상태를 변경할 수 없는 객체를 의미합니다. 해당 패턴은 다음과 같은 이접에서 유용합니다:
- 예측 가능성: 객체의 상태가 변경되지 않기 때문에, 코드를 이해하고 디버깅하기가 더 쉬워집니다.
- 스레드 안전성: 여러 스레드에서 동시에 해당 객체에 접근해도 상태가 변하지 않아, 동기화 처리 없이도 안전하게 사용할 수 있습니다.
- 실패 원자성: 객체의 상태가 변하지 않으므로, 연산 중 예외가 발생해도 객체의 일관성을 유지할 수 있습니다.

따라서, beginSync 메서드에서 기존 TraceId 객체의 id 값을 직접 증가시키는 방식 대신에 새로운 객체를 생성해서 반환하는 접근법은 위와 같은 이점을 취하기 위함입니다. id만 증가시켜 같은 객체를 사용하는 경우, 객체의 상태가 여러 위치에서 변경될 수 있어 예측하기 어렵고, 여러 스레드에서 동시에 접근했을 때 안전하지 않을 수 있습니다.

비슷한 맥락의 질문과 답변이 “createNextId() 메서드에서 항상 새로운 객체를 생성하는 이유가 궁금합니다.“라는 링크에 포함되어 있습니다. 이 답변은 값 객체의 불변성에 대해 설명하고 있는데, 질문자님의 경우와 유사한 상황에서 불변 객체를 사용하는 이유를 설명합니다.

저는 질문자님의 이해를 돕고자 지식공유자님의 기존 답변을 참고하여 유사한 답변 링크를 추천해드리고 있습니다.
현재 베타 기능으로 답변이 만족스럽지 않을 수 있는 점 양해 부탁드립니다. 🙏
추가적으로 궁금한 점이 있으시면, 이어서 질문해 주세요. 곧 지식공유자께서 답변해 주실 것입니다.

어드바이스 순서 디폴트 기준이 궁금합니다.

0

50

1

AspectV1 예제를 @Configuration 수동 등록으로도 가능한가요?

0

77

2

구체 클래스를 상속받아 확장한 형태도 클래스패턴/데코레이터 패턴이라고 칭하나요?

0

49

1

TraceTemplate 을 미리 빈으로 등록해서 사용할때 이렇게 설정하는게 맞는건가요?

0

53

1

Decorator 에서 추상메서드로 뺄때 질문 있습니다.

0

55

1

대상 클래스에 기본 생성자가 없을 때

0

74

1

스프링 부트 버전 4.0 aop 의존성 명칭 변경

1

343

2

final 키워드 사용 관련 질문

0

74

1

안녕하세요

0

64

1

ThradLocal 실무 사례

0

97

2

실무에서의 동시성 문제

0

62

1

로그 등 부가 기능에 대한 테스트코드

0

50

1

ProceedingJoinPoint와 MethodInvocation에 대하여

0

67

1

실무 멀티스레딩에 대한 궁금증

0

69

1

안녕하세요 질문있습니다

0

43

1

AspectV3 aop 실행 순서 문의 드립니다

0

37

1

MDC vs AOP 중에 고민중입니다.

0

74

1

순환참조

0

158

2

this를 단독으로 사용할때랑 파라미터 바인딩 할때 차이

0

77

1

빈 후처리에 등록 질문 있습니다.

0

75

1

AppConfig 설정하는 설명중에 이해가 안되는게 있습니다.

0

88

1

ThreadLocal을 지역변수로 선언하면 remove가 필요할까요?

0

117

2

@Aspect 어노테이션으로 생성된 Advisor의 Bean 저장 여부

0

67

1

ThreadLocal 사용시 부작용이 있을수 있을까요? (ThreadLocal.remove()를 잘한다고 해도..)

0

73

2