클래스 속성(property)에 관한 보충 설명입니다. 자세한 내용은 쉽게 다가가는 최신 프로그래밍: 코틀린 - 4.2 클래스 속성을 참고하기 바랍니다.
클래스 속성은 변수 아닌가? 아닙니다. 클래스 속성은 기본적으로 변수이지만 속성과 연관된 함수(getter 또는 setter)를 정의할 수 있습니다. 코틀린에서는 속성에 대한 getter, setter를 자동 생성한다던데... 반은 맞고 반은 틀립니다. 속성에 대한 getter와 setter도 목적에 맞게 정의할 수 있습니다. 여러분이 혹시 놓치고 있을지도 모를 클래스 속성에 대한 모든 것을 알아보겠습니다.
■ 클래스에서 속성 선언: 디폴트 getter와 디폴트 setter
클래스 Person은 2개의 속성(name과 age)을 갖습니다. 클래스 Person의 속성에 초깃값을 할당하면, 객체 kim을 생성합니다. "kim.name"을 사용해 객체 kim의 속성 name에 저장된 값을 가져올 수 있습니다. 간단한 코드이지만, 여기에 중요한 내용이 숨겨져 있습니다.
class Person (_name: String, _age: Int) {
val name: String = _name
var age: Int = _age
}
fun main() {
val kim = Person("Kim", 23)
println(kim.name)
println(kim.age)
}
클래스의 속성에 초깃값이 할당되면, 백업 필드(backing field)라 불리는 비공개 필드(field)에 초깃값을 저장합니다. "kim.name"처럼 속성 name에 저장된 값을 가져오려면, 필드에 저장된 값을 읽어오는 공개 메소드 getter가 필요합니다. 클래스의 속성 name을 val로 선언하면, 코틀린 컴파일러는 아래처럼 디폴트 getter를 자동으로 생성합니다. 이 getter를 사용해 필드에 저장된 값을 가져옵니다.
val name: String = _name
get() = field // 디폴트 게터(getter)
var로 선언한 속성 age는 읽고 쓰는 작업이 모두 필요합니다. 속성 age에 대해서는 필드에 저장된 값을 읽어오기 위한 디폴트 getter와 필드에 값(value)을 저장하기 위한 setter가 생성됩니다.
var age: Int = _age
get() = field // 디폴트 getter
set(value) {
field = value // 디폴트 setter
}
■ field(백업 필드) - 속성을 초기화해야 getter/setter에서 field를 사용할 수 있습니다.
field는 백업 필드(backing field)로써, 속성의 초깃값을 저장하고 있습니다. 위 코드와 아래 코드를 비교해 볼까요. 뭐가 달라졌을까요? (코드를 간단히 하려고 속성 age를 val로 수정했습니다)
class Person {
val name: String = "Kim"
get() = field
val age: Int = 23
get() = field
val isAdult: Boolean
get() = this.age >= 18
}
fun main() {
val kim = Person()
println(kim.name)
println(kim.isAdult)
}
코드를 보면 val name: String = "Kim"과 같이 속성을 초기화했습니다(속성 초기화 과정에서는 setter를 호출하는 것이 아니라 초깃값을 field에 저장합니다). 이렇게 속성을 초기화해야만 field를 사용할 수 있기 때문입니다. kim.name으로 속성 name을 참조하면, getter는 field에 저장된 값을 반환합니다.
그럼 속성 isAdult는 어떨까요? 속성 isAdult의 get()에서는 field를 사용할 수 없습니다. 속성 isAdult를 초기화하지 않았기 때문이죠. isAdult에 대한 백업 필드가 없기 때문에, 속성 isAdult를 참조할 때마다 get()을 다시 실행해 값을 구합니다.
■ 수식어 val을 사용한 속성 : 사용자 정의 getter
주 생성자에서 정의한 속성(name, age)에 대해서는 사용자 정의 getter (또는 setter)를 정의할 수 없습니다. 클래스 Person 블록 안에서 선언한 속성에 대해서만 getter(또는 setter)를 정의할 수 있습니다. 여기서는 속성 isAdult에 대해 사용자 정의 getter를 정의합니다. kim.isAdult처럼 속성 isAdult를 참조할 때 getter가 호출되면서 속성을 초기화합니다.
class Person (val name: String, var age: Int) {
val isAdult: Boolean
get() = age >= 18
}
fun main() {
val kim = Person("Kim", 19)
println(kim.isAdult) // true
}
■ 수식어 var를 사용한 속성 : 사용자 정의 getter와 setter
수식어 var를 사용해 선언한 속성은 사용자 정의 getter와 setter를 정의할 수 있습니다. 아래 코드에서 속성 age는 자동으로 생성된 getter(get() = field)를 사용하며, 사용자 정의 setter만 직접 구현했습니다. kim.age = -1 과 같이 속성 age 값을 변경할 때 setter가 호출됩니다.
class Person (val name: String) {
var age: Int = 0
get() = field // default getter. 생략할 수 있음.
set(value) {
if (value >= 0) {
field = value
} else {
println("Age cannot be negative")
}
}
}
fun main() {
val kim = Person("Kim")
kim.age = -1
println(kim.age) // 0
}
■ 가시성 수식어(visibility modifier)를 붙인 속성
속성에 대한 사용자 정의 getter와 setter에도 가시성 수식어를 붙일 수 있습니다. 가시성 수식어 중 private을 주로 사용하며, 가변 속성의 값을 클래스 밖에서 변경하지 못하도록 일반적으로 setter에 수식어 private 을 붙입니다. 자동으로 생성되는 getter나 setter에만 적용할 경우 get 또는 set 앞에 키워드 private를 붙이면 됩니다.
아래 예는 속성 age를 클래스 밖에서 바꾸지 못하도록 기본 setter(private set)를 정의했습니다. 올해 년도를 입력하면 속성 currentYear가 바뀌면서 속성 age도 함께 바뀝니다. 속성 age에 대한 getter는 따로 정의하지 않았지만, 기본 가시성은 public이며 외부에서 속성 age를 참조할 수 있습니다. println(kim.age)를 실행하면 현재 나이를 정확히(?) 알 수 있습니다.
import java.time.LocalDate
class Person (val name: String, val birthYear: Int) {
var age: Int = 0
private set
var currentYear: Int = 2000
set(value) {
field = value
age = currentYear - birthYear
}
}
fun main() {
val kim = Person("Kim", 2000)
kim.currentYear = LocalDate.now().year
println(kim.age)
}
'코틀린' 카테고리의 다른 글
코틀린: 4장 테스트에 도전해 보세요 (0) | 2025.01.06 |
---|---|
코틀린: enum class(updated) (0) | 2025.01.06 |
코틀린: 생성자와 속성 초기화 - 보조 생성자 (0) | 2025.01.06 |
코틀린: 생성자와 속성 초기화 - 주 생성자 (0) | 2025.01.06 |
코틀린: 확장 함수와 수신 객체를 갖는 함수 타입(3/3) (0) | 2025.01.06 |