인프런 커뮤니티 질문&답변

장선애님의 프로필 이미지

작성한 질문수

자바 개발자를 위한 코틀린 입문(Java to Kotlin Starter Guide)

11강. 코틀린에서 접근 제어를 다루는 방법

Accessing non-final property number in constructor

해결된 질문

24.07.05 14:43 작성

·

153

·

수정됨

1

- 학습 관련 질문을 남겨주세요. 상세히 작성하면 더 좋아요!
- 먼저 유사한 질문이 있었는지 검색해보세요.
- 서로 예의를 지키며 존중하는 문화를 만들어가요.
- 잠깐! 인프런 서비스 운영 관련 문의는 1:1 문의하기를 이용해주세요.

먼저, 좋은 강의를 만들어주셔서 감사합니다. 잘 이해가 되지 않는 부분이 있습니다.

Accessing non-final property number in constructor가 발생하는 이유를 다음과 같이 정리해보았는데 맞을까요??

  1. 하위 클래스(Derived)를 인스턴스화하면서 상속받고 있는 상위 클래스(Base)가 먼저 생성된다.

  2. init 블럭이 생성자보다 먼저 호출되므로, init 블럭에서 프로퍼티에 접근하면 초기화되지 않은 기본값을 갖는다.

  3. init 블럭이 호출되고 하위 클래스에서 전달한 값이 상위 클래스 생성자에 있는 프로퍼티 값으로 초기화된다.

  4. 하위 클래스의 init 블럭이 호출될 때, number 프로퍼티 값은 상위 클래스가 갖는 number 프로퍼티 값을 출력한다.

"Accessing non-final property number in constructor 가 경고창이 드는 이유는 init 블럭에서 초기화되지 않은 number 프로퍼티에 접근했기 때문이다." 라고 정리하면 될까요?!

감사합니다.

open class Base(
    open val number: Int = 100
) {

    init {
        println("Base Class")
        /**
         * 호출 순서 : init 블럭 -> 생성자
         * 따라서 init 블럭에서 number 프로퍼티에 접근하면 초기화되지 않은 초기값인 0을 갖게 된다.
         */
        println("Base Class number ${number}")
    }
}
class Derived(
    override val number: Int
): Base(number) {

    init {
        println("Derived Class")
        println("Derived Class number : ${number}")
    }
}

답변 2

1

최태현님의 프로필 이미지
최태현
지식공유자

2024. 07. 06. 13:56

안녕하세요 선애님! 🙂

결론부터 말씀드리면

"Accessing non-final property number in constructor 가 경고창이 드는 이유는 init 블럭에서 초기화되지 않은 number 프로퍼티에 접근했기 때문이다." 라고 정리하면 될까요?!

라고 표현해주신게 맞습니다!

 

조금 더 정확히 말씀드려보면

  • Derived 클래스를 인스턴스화 하는 과정에서

  • 가장 먼저Base 클래스 (상의 클래스) init block이 호출 되었고

  • 이 안에서 필드 number 에 접근을 하게 되었는데

  • number 필드는 open 되어 있고 하위 클래스에서 이 필드를 재정의 하고 있을 수도 있으니

  • 아직 값이 초기화되지 않았고 기본값인 0이 나오게 된다

라고 할 수 있을 것 같아요! 😊

 

또 궁금하신 점 있으시면 언제든 편하게 질문 남겨주세요.

감사합니다! 🙏

0

인프런 AI 인턴님의 프로필 이미지

2024. 07. 05. 14:46

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

장선애님, 좋은 질문해주셨네요. 상속을 활용한 클래스 초기화 과정 중 발생하는 Accessing non-final property number in constructor 경고에 대해 정리해 주셨군요.

장선애님이 해주신 요약 정리는 크게 올바른 방향입니다. Kotlin에서 부모 클래스(Base)의 property를 하위 클래스(Derived)에서 오버라이딩하는 경우, 특히 생성자와 init 블록에서 해당 프로피터에 접근할 때 주의해야 합니다. 그 이유는, Kotlin에서는 클래스의 생성 과정 중 init 블록이 생성자보다 먼저 실행되기 때문입니다. 그리고 하위 클래스의 property는 하위 클래스의 생성자가 실행될 때 최종적으로 초기화되므로, 부모 클래스의 init 블록에서 해당 프로퍼티에 접근하면 오버라이드된 값이 아닌 기본값이나 초기화되지 않은 상태의 값이 사용될 수 있습니다.

이 문제를 해결하기 위해 몇 가지 접근 방법이 있습니다:
1. 최대한 생성자 내부에서만 필요한 작업을 처리하고, open 프로퍼티에 대한 접근을 자제합니다.
2. 초기화에 의존적인 로직이 있다면, 오버라이드된 프로퍼티가 아닌 메서드를 통해 접근하는 방법을 고려할 수 있습니다.

따라서, 장선애님이 정리하신 내용은 매우 중요한 Kotlin의 클래스 상속 및 초기화 순서와 관련된 개념을 잘 설명하고 있으며, “Accessing non-final property number in constructor경고가 뜨는 이유가 init 블럭에서 초기화되지 않은 number 프로퍼티에 접근했기 때문”이라는 결론 또한 정확합니다.

여러 질문 중 유사한 주제로 상위 클래스의 생성자나 init 블럭에서 final이 아닌 프로퍼티에 접근하지 말라 의미여 에 대한 설명이 도움이 될 수 있습니다. 이 게시글에서도 상속과 초기화 블록에서의 프로퍼티 접근에 대한 주의사항이 논의되어 있습니다.

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