Programming Language/Kotlin

[Koltin] Object Keyword. (싱글톤 패턴, 익명 함수)

개발왕 금골드 2023. 2. 8. 22:08
반응형
  • 경우에 따라 새 하위 클래스를 명시적으로 선언하지 않고 일부 클래스를 약간 수정한 개체를 만들어야 한다. 코틀린은 객체 표현식과 객체 선언으로 이를 처리할 수 있다.

Object expressions

  • 객체 표현식은 익명 클래스, 즉 클래스가 명시적으로 선언되지 않은 클래스의 객체를 만든다. 이러한 클래스는 일회용으로 사용하기 유용하다.
  • 처음부터 정의하거나 기존 클래스에서 상속하거나 인터페이스를 구현할 수 있다.
  • 익명 클래스의 인스턴스는 이름이 아닌 식에 의해 정의되므로 익명 객체라고도 한다.


Creating anonymous objects from scratch

  • 체 표현식은 object 키워드를 사용한다.
val helloWorld = object {
    val hello = "Hello"
    val world = "World"
    // object expressions extend Any, so `override` is required on `toString()`
    override fun toString() = "$hello $world"
}

// result
// Hello World


Inheriting anonymous objects from supertypes

  • 어떤 유형을 상속받는 익명 클래스의 객체를 만들려면 object와 콜론(:) 뒤에 유형을 선언한다.
  • 그런 다음 이 클래스 구성원을 상속받는 것처럼 구현하거나 재정의한다.
window.addMouseListener(object : MouseAdapter() {
    override fun mouseClicked(e: MouseEvent) { /*...*/ }

    override fun mouseEntered(e: MouseEvent) { /*...*/ }
})
  • 슈퍼타입에 생성자가 있는 경우 적절한 생성자 매개변수를 전달한다.
  • 콜론 뒤에 여러 개의 슈퍼타입을 쉼표로 구분하여 목록으로 지정할 수 있다.
open class A(x: Int) {
    public open val y: Int = x
}

interface B { /*...*/ }

val ab: A = object : A(1), B {
    override val y = 15
}


Using anonymous objects as return and value types

  • 익명 객체가 인라인 선언이 아닌 로컬 또는 private 유형의 객체로 사용될 때, 모든 구성원은 이 함수 또는 프로퍼티를 통해 액세스 할 수 있다.
class C {
    private fun getObject() = object {
        val x: String = "x"
    }

    fun printX() {
        println(getObject().x)
    }
}
  • 만약 이러한 함수 혹은 프로퍼티가 public or private inline으로 선언되었다면, 실제 타입은 다음 상황에 따라 결정된다.
    • 만약 선언된 슈퍼타입이 없다면 익명 객체의 타입은 Any이다.
    • 하나의 슈퍼타입으로 선언되었다면 타입으로 선언된 슈퍼타입과 동일한 타입을 갖는다.
    • 둘 이상이라면 명시적으로 선언된 타입을 갖는다.
  • 이러한 경우 익명 객체에 추가된 구성원에 접근할 수 없다. 재정의된 구성원은 함수 또는 프로퍼티의 실제 유형으로 선언된 경우 접근할 수 있다.
interface A {
    fun funFromA() {}
}
interface B

class C {
    // The return type is Any. x is not accessible
    fun getObject() = object {
        val x: String = "x"
    }

    // The return type is A; x is not accessible
    fun getObjectA() = object: A {
        override fun funFromA() {}
        val x: String = "x"
    }

    // The return type is B; funFromA() and x are not accessible
    fun getObjectB(): B = object: A, B { // explicit return type is required
        override fun funFromA() {}
        val x: String = "x"
    }
}


Accessing variables from anonymous objects

  • 객체 표현식 안에 있는 코드는 일정 범위 안에 있는 변수에 접근할 수 있다.
fun countClicks(window: JComponent) {
    var clickCount = 0
    var enterCount = 0

    window.addMouseListener(object : MouseAdapter() {
        override fun mouseClicked(e: MouseEvent) {
            clickCount++
        }

        override fun mouseEntered(e: MouseEvent) {
            enterCount++
        }
    })
    // ...
}


Object declarations

  • Singleton 패턴은 매우 유용한데, 코틀린은 object 키워드를 사용하여 이를 쉽게 만들 수 있다.
  • 객체 선언은 표현식이 아니므로 할당문 오른쪽에 선언할 수 없다.
  • 객체 선언의 초기화는 스레드에 안전하며 첫 번째 액세스에서 수행된다.
  • 객체 선언은 로컬에서 선언될 수 없지만 (즉, 함수 내에 직접 중첩 불가.) 다른 객체 선언이나 내부 클래스가 아닌 클래스로 중첩될 수 있다.
object DataProviderManager {
    fun registerDataProvider(provider: DataProvider) {
        // ...
    }

    val allDataProviders: Collection<DataProvider>
        get() = // ...
}
  • 객체를 참조하려면 직접 참조한다.
DataProviderManager.registerDataProvider(...)
  • 슈퍼타입이 존재할 수 있다.
object DefaultListener : MouseAdapter() {
    override fun mouseClicked(e: MouseEvent) { ... }

    override fun mouseEntered(e: MouseEvent) { ... }
}
  • Kotlin에서 일반 객체 선언을 인쇄할 때 문자열 표현에 객체의 이름과 해시가 모두 포함되어 있다.
object MyObject

fun main() {
    println(MyObject) // MyObject@1f32e575
}
  • data class처럼 data object로 선언하면 toString 함수에 대한 구현을 수동으로 구현하지 않고도 적절한 형식의 문자열 표현을 얻을 수 있다. (Kotlin 1.9 버전에서 사용 가능하지만 실험적인 기능.)
data object MyObject

fun main() {
    println(MyObject) // MyObject
}
  • data object 기능은 sealed class와 함께 사용하기 유용하다.
sealed class ReadResult {
    data class Number(val value: Int): ReadResult()
    data class Text(val value: String): ReadResult()
    data object EndOfFile: ReadResult()
}

fun main() {
    println(ReadResult.Number(1)) // Number(value=1)
    println(ReadResult.Text("Foo")) // Text(value=Foo)
    println(ReadResult.EndOfFile) // EndOfFile
}


Companion objects

  • companion 키워드를 사용하여 멤버를 전역으로 사용할 수 있고 쉽게 접근할 수 있다.
  • 해당 클래스가 로드될 때 초기화된다.
  • const val로 선언된 상수는 static 변수라고 할 수 있다. (혹은, @JvmStatic or @JvmField 어노테이션이 붙은 함수 및 프로퍼티)
class MyClass {
    companion object Factory {
        fun create(): MyClass = MyClass()
    }
}

val instance = MyClass.create()
  • 다른 언어에서의 static 멤버와 비슷하지만 런타임에서는 실제 객체의 인스턴스 멤버이기 때문에 인터페이스 구현이 가능하다.
  • 그러나 만약 @JvmStatic 어노테이션을 사용하였다면 JVM에서 실제 정적 메서드 및 필드로 생성된 객체의 멤버를 가질 수 있다. Java와의 상호 운용성에 대한 이해가 필요하다.
interface Factory<T> {
    fun create(): T
}

class MyClass {
    companion object : Factory<MyClass> {
        override fun create(): MyClass = MyClass()
    }
}

val f: Factory<MyClass> = MyClass


Semantic difference between object expressions and declarations

  • 객체 표현식은 사용되는 곳에서 즉시 실행(초기화)된다.
  • 객체 선언은 처음 액세스되었을 때 lazy 초기화된다.
  • 객체 선언은 Java로 변환된 파일 내부에 object file name 타입의 static INSTANCE 객체가 생성된다. (Singleton)
  • companion object는 해당 클래스가 로드될 때 초기화된다.
  • companion object 내에서 선언된 함수 및 프로퍼티는 Java로 변환된 파일 내부에 생성되는 static class Companion을 통해 접근이 가능하다. 즉, 함수 및 프로퍼티 자체가 static은 아니다.

 



Kotlin Docs 번역
참고자료 :
https://kotlinlang.org/docs/object-declarations.html

 

Object expressions and declarations | Kotlin

 

kotlinlang.org

 

반응형