Kotlin/스터디
코틀린 스터디 - 11장
끄공
2023. 9. 24. 19:15
클래스 위임
- 특정 클래스에 자기 클래스가 할 일을 다른 클래스에 맡겨 처리하는 것을 말한다.
- 위임을 맡기는 클래스를 위탁자, 위임을 받아서 실제 처리하는 클래스를 수탁자라고 한다.
- 코틀린에서는 이 관계를 by를 사용해서 쉽게 구성할 수 있다.
위임이 좋은 것은 다양한 기능을 하나의 클래스를 통해서 받고 처리할 수 있도록 구조화할 수 있기 때문이다.
동일한 일을 나눠서 처리하려면 공통된 인터페이스가 필요하다.
클래스 위임 처리
//같은 결과를 반환하는 두 case
val b = BaseImpl(10)
Derived(b).say()
class Derived_() : Base by BaseImpl(10)
Derived_().say()
//베이스 클래스 구현 : 10
//베이스 클래스 구현 : 10
생성자의 매개변수 속성으로 위임 처리
val ps = Person("사람") // Person은 인터페이스 Sayable을 상속받는 클래스
val pt = Pet("개") // Pet은 인터페이스 Sayable을 상속받는 클래스
class Saying(val say : Sayable) : Sayable by say //매개변수로 전달한 객체로 위임 처리
Saying(ps).say()
Saying(pt).say()
//안녕하세요 : 사람
//멍멍멍 : 개
상속관계 클래스 위임 처리 가능
class Screen(val showable: Showable): Showable by showable
val view = View() //View는 Showable 인터페이스를 상속받는 클래스
val customView = CustomView() //CustomView는 View를 상속받는 클래스
Screen(view).show()
Screen(customView).show()
//View 클래스의 show()
//CustomView 클래스의 show()
계층구조를 가져도 위탁자 클래스와 동일한 인터페이스 구현이 되면 이 인터페이스를 기준으로 위임관계를 구성할 수 있다.
여러 인터페이스로 클래스 위임 처리
class Screen(val showable: Showable, val viewable: Viewable): Showable by showable, Viewable by viewable
val show = Show()
val view = View()
Screen(show, view).show() //View.show()
Screen(show, view).view() //CustomView.show()
//Show 클래스의 show()
//View 클래스의 view()
속성 위임
- 클래스 위임에도 by 예약어를 사용했듯이 속성에도 by로 위임을 처리할 수 있다.
- 보통 코틀린 내부적으로 위임을 처리할 수 있는 notNull, vetoable, observable 메서드를 제공한다. 또한, 지연처리를 하는 lazy 함수도 사용할 수 있다.
- 속성에 대한 위임은 속성을 사용할 때 특정한 값을 세팅해서 처리하는 것이다. 이때 Delefates 객체의 notNull, observable, vetoable를 사용한다.
notNull 위임처리
- 속성의 지연초기화는 lateinit 예약어로 처리할 수 있다. 이 방식에서 정수와 실수 프리미티브가 사용되는 자료형은 지연초기화를 할 수 없다.
- 속성 위임을 by로 사용하고 Delegates.notNull 메서드를 사용하면 프리미티브도 위임을 통한 지연초기화가 가능하다.
- Delegates.notNull은 val 속성으로 지정할 때는 사용할 수 없다.
import kotlin.properties.Delegates //코틀린 속성 지연 처리 object
lateinit var str : String //최상위 속성 지연 초기화는 참조 객체만 가능
//lateinit var int : Int //최상위 속성 지연 초기화할 때는 기본 자료형은 불가
var str1 : String by Delegates.notNull<String>()
var int1 : Int by Delegates.notNull<Int>()
//val str2: String by Delegates.notNull<String>() //val은 재할당 금지라서 지연초기화 사용 못함
//val int2: Int by Delegates.notNull<Int>() //val은 재할당 금지라서 지연초기화 사용 못함
str = "초기화 처리"
str1 = "초기화 처리"
int1 = 100
println("lateinit 처리 = $str")
println("notnUll String 처리 = $str1")
println("notNull Int. 처리 = $int1")
//lateinit 처리 = 초기화 처리
//notNull String 처리 = 초기화 처리
//notNull Int. 처리 = 100
속성 변경 관찰
import kotlin.properties.Delegates
var observed = false
var max: Int by Delegates.observable(0) {
_, oldValue, newValue ->
println("변경전 : $oldValue 변경후 : $newValue")
observed = true
}
println(max)
println("관찰상태 = ${observed}")
max = 10
println(max)
println("관찰상태 = ${observed}")
//0
//관찰상태 = false
//변경전 : 0 변경후 : 10
//10
//관찰상태 = true
Q. liveData와 비교해서 Delegates.observable를 썼을 때의 이점이 있을까요?
특정 조건이 일치할 때만 속성 위임 변경
- 속성 위임을 vetoable 메서드로 처리할 경우는 조건이 만족할 때만 속성값을 변경한다.
- 이 속성의 필드를 바꾸면 실제 짝수일 경우에만 값이 변경된다.
var vetoableField: Int by Delegates.vetoable(0) { //변경 거부 여부
_, old, new -> //속성 변경 전
println("변경전 : $old, 변경후 : $new") //변경 상태 확인
new % 2 == 0 //조건이 참인 경우: 짝수 값일 경우만 갱신
}
속성과 지역변수 지연초기화
- 속성 위임을 사용하는 지연처리는 val 속성과 val 변수에서 가능하다.
- 함수 내의 지역변수도 var는 lateinit, val은 by lazy로 지연초기화 처리를 할 수 있다.
Q. 맵 자료형을 사용한 속성 위임 처리는 이점, 활용 목적이 무엇인가요?