Kotlinのプロパティ委譲!10選の使い方と実践例

Kotlinのプロパティ委譲を詳しく解説するイメージKotlin
この記事は約25分で読めます。

 

【サイト内のコードはご自由に個人利用・商用利用いただけます】

この記事では、プログラムの基礎知識を前提に話を進めています。

説明のためのコードや、サンプルコードもありますので、もちろん初心者でも理解できるように表現してあります。

基本的な知識があればカスタムコードを使って機能追加、目的を達成できるように作ってあります。

※この記事は、一般的にプロフェッショナルの指標とされる『実務経験10,000時間以上』を凌駕する現役のプログラマチームによって監修されています。

サイト内のコードを共有する場合は、参照元として引用して下さいますと幸いです

※Japanシーモアは、常に解説内容のわかりやすさや記事の品質に注力しております。不具合、分かりにくい説明や不適切な表現、動かないコードなど気になることがございましたら、記事の品質向上の為にお問い合わせフォームにてご共有いただけますと幸いです。
(送信された情報は、プライバシーポリシーのもと、厳正に取扱い、処分させていただきます。)

はじめに

Kotlinは現代のプログラミング言語として、その簡潔さや革新的な特徴で注目を浴びています。

その中でも「プロパティ委譲」はKotlinの特色の一つとして知られています。

今回の記事では、Kotlinのプロパティ委譲について初心者から上級者まで詳しく解説していきます。

プロパティ委譲の基本から使い方、実践的なサンプルコード、さらにはカスタマイズ方法までを網羅的に紹介します。

●Kotlinのプロパティ委譲とは

Kotlinのプロパティ委譲は、プロパティの取得や設定のロジックを別のオブジェクトに委譲することができる機能です。

これにより、プロパティの振る舞いを再利用したり、コードの見通しを良くしたりすることができます。

Kotlinの公式ライブラリには、よく使われるプロパティ委譲のパターンが予め提供されており、それを利用することで簡単にプロパティ委譲を実装することができます。

○プロパティ委譲の基本

Kotlinでプロパティ委譲を行うには、プロパティの定義時に「by」キーワードを用いることで、その振る舞いを別のオブジェクトに委譲することができます。

例えば、次のような簡単なプロパティ委譲のサンプルコードを考えてみましょう。

class Example {
    var prop: String by Delegate()
}

class Delegate {
    operator fun getValue(thisRef: Any?, property: KProperty<*>): String {
        return "Hello from delegate!"
    }

    operator fun setValue(thisRef: Any?, property: KProperty<*>, value: String) {
        println("$value has been assigned to '${property.name}' in $thisRef.")
    }
}

fun main() {
    val e = Example()
    println(e.prop)
    e.prop = "New value"
}

このコードでは、ExampleクラスのpropというプロパティがDelegateクラスによって委譲されています。

DelegateクラスはgetValuesetValueの2つのオペレータ関数を持っており、これによってプロパティの取得や設定の動作が定義されています。

このコードを実行すると、まず”Hello from delegate!”という文字列が出力されます。これはgetValue関数によるものです。

次に、”New value has been assigned to ‘prop’ in Example@…”というメッセージが出力されます。

これはsetValue関数が呼び出されたときのものです。

●プロパティ委譲の使い方

Kotlinは様々なプログラム機能を備えている中、プロパティ委譲はその中でも非常に強力な機能の一つと言えます。

プロパティ委譲を適切に使用することで、コードの簡潔さや再利用性が大幅に向上します。

Kotlinのプロパティ委譲の背景として、Kotlinでは変数やプロパティの振る舞いをカスタマイズする必要が出てきた場合、毎回ゲッターやセッターを書き換えるのは非効率的でした。

そこで、Kotlinはこの問題を解決するために、プロパティのゲッターやセッターの振る舞いを他のオブジェクトに委譲する機能を導入しました。

○サンプルコード1:基本的なプロパティ委譲

下記のサンプルコードでは、Kotlinのプロパティ委譲の基本的な使い方を表しています。

class Example {
    var message: String by Delegate()
}

class Delegate {
    var value: String = ""

    operator fun getValue(thisRef: Any?, property: KProperty<*>): String {
        return value
    }

    operator fun setValue(thisRef: Any?, property: KProperty<*>, value: String) {
        this.value = value
    }
}

fun main() {
    val example = Example()
    example.message = "Hello, Kotlin!"
    println(example.message)
}

このコードでは、Exampleクラスのmessageプロパティのゲッターとセッターの振る舞いをDelegateクラスに委譲しています。

Delegateクラスには、getValuesetValueの二つのオペレータ関数が定義されており、これによりmessageプロパティの取得と設定の振る舞いがカスタマイズされます。

このコードを実行すると、コンソールにHello, Kotlin!というメッセージが表示されます。

○サンプルコード2:Lazy委譲の使用

Kotlinの標準ライブラリには、いくつかの便利なプロパティ委譲が用意されており、lazy委譲はその中でも特によく使用されるものの一つです。

lazyは、プロパティの初回アクセス時にのみ値を計算し、以降はその計算結果をキャッシュする特性を持っています。

val lazyValue: String by lazy {
    println("Computing value...")
    "Hello, Lazy Kotlin!"
}

fun main() {
    println("Before accessing lazyValue")
    println(lazyValue)
    println("After accessing lazyValue")
    println(lazyValue)
}

このコードでは、lazyValueというプロパティにlazy委譲を使用しています。

lazyValueにアクセスする前と、2回アクセスした際の振る舞いが異なります。

このコードを実行すると、初回アクセス時にのみComputing value...というメッセージが表示され、2回目以降のアクセス時にはこのメッセージは表示されません。

このようにlazyを使うことで、コンピュータリソースを節約することができます。

○サンプルコード3:Observable委譲の使用

Kotlinでは、プロパティ値の変更を監視するための特別なタイプの委譲、Observableを提供しています。

この委譲を使用することで、プロパティが変更された際に何らかのアクションを実行することができます。

Observable委譲を使用したサンプルコードを紹介します。

import kotlin.properties.Delegates

class User {
    var name: String by Delegates.observable("<no name>") { _, old, new ->
        println("名前が $old から $new に変更されました。")
    }
}

fun main() {
    val user = User()
    user.name = "Taro"
    user.name = "Jiro"
}

このコードでは、Userクラス内のnameというプロパティにDelegates.observableを使用しています。

このデリゲートは3つのパラメータを受け取ります。

最初のパラメータはデフォルト値、次の二つはラムダ式のパラメータで、古い値と新しい値です。

ラムダ式の中で、名前がどのように変更されたかを印刷しています。

このコードを実行すると、プロパティの変更が監視され、次の出力が得られます。

名前が <no name> から Taro に変更されました。
名前が Taro から Jiro に変更されました。

Observable委譲は、状態の変更をトラックする必要がある場合や、特定のアクションをトリガするための条件を設定する際に特に役立ちます。

○サンプルコード4:Mapによる委譲の使用

Kotlinには、Mapのエントリをプロパティにマッピングするための特別なタイプの委譲も提供されています。

これを使用すると、Mapのエントリを直接プロパティとして扱うことができるようになります。

Mapを使用した委譲のサンプルコードを紹介します。

class User(val map: Map<String, Any?>) {
    val name: String by map
    val age: Int     by map
}

fun main() {
    val user = User(mapOf(
        "name" to "Taro",
        "age"  to 25
    ))

    println("名前: ${user.name}, 年齢: ${user.age}")
}

このコードでは、Userクラスはmapという名前のMapを受け取ります。

このMap内のエントリは、nameageという名前のプロパティにマッピングされます。

Mapのキーとプロパティの名前が一致しているため、委譲が動作します。

このコードを実行すると、次の出力が得られます。

名前: Taro, 年齢: 25

Mapを使用した委譲は、JSONのようなキーと値のペアからオブジェクトを生成する場面などで特に有効です。

●プロパティ委譲の応用例

Kotlinのプロパティ委譲は非常に強力で、基本的な使い方からさまざまな応用例まであります。

ここでは、その中でも特に実践的な応用例を2つ、カスタムデリゲートの作成と非同期の処理を伴う委譲に焦点を当てて解説します。

○サンプルコード5:カスタムデリゲートの作成

カスタムデリゲートを作成することで、独自のロジックや機能を持ったプロパティ委譲を実現することができます。

下記のコードでは、指定された範囲内の整数のみを許可するカスタムデリゲートを作成しています。

範囲外の値が設定された場合、デフォルトの値にリセットされます。

class RangeDelegate(private val start: Int, private val end: Int, private val default: Int) {
    private var value = default

    operator fun getValue(thisRef: Any?, property: KProperty<*>): Int {
        return value
    }

    operator fun setValue(thisRef: Any?, property: KProperty<*>, value: Int) {
        if (value in start..end) {
            this.value = value
        } else {
            this.value = default
        }
    }
}

class SampleClass {
    var rangeValue: Int by RangeDelegate(1, 10, 5)
}

fun main() {
    val sample = SampleClass()

    sample.rangeValue = 7
    println(sample.rangeValue)  // 範囲内なので7が出力されます。

    sample.rangeValue = 11
    println(sample.rangeValue)  // 範囲外なのでデフォルトの5が出力されます。
}

このコードでは、RangeDelegateという名前のカスタムデリゲートを定義し、範囲内の整数のみを許可しています。

そして、SampleClassというクラスにこのデリゲートを適用しています。

このコードを実行すると、7と5が順番に出力される結果となります。

範囲外の値11が設定された時点で、デリゲートが介入し、デフォルトの5に値がリセットされているためです。

○サンプルコード6:非同期の処理を伴う委譲

非同期の処理を行う場合、coroutinesやFutureなどの非同期ライブラリと組み合わせて、プロパティの値の取得や設定を非同期に行うことが考えられます。

それでは、coroutinesを使って非同期にデータを取得するプロパティ委譲の例を紹介します。

import kotlinx.coroutines.*

class AsyncDelegate<T>(private val block: suspend () -> T) {
    private var value: T? = null

    operator fun getValue(thisRef: Any?, property: KProperty<*>): T {
        if (value == null) {
            runBlocking {
                value = block()
            }
        }
        return value!!
    }
}

class AsyncSampleClass {
    val asyncData: String by AsyncDelegate {
        delay(1000)
        "非同期で取得したデータ"
    }
}

fun main() {
    val sample = AsyncSampleClass()

    println("データ取得開始")
    println(sample.asyncData)
    println("データ取得完了")
}

このコードでは、AsyncDelegateという非同期のデリゲートを定義しています。

このデリゲートでは、指定された非同期の処理を実行し、その結果を返します。

このコードを実行すると、”データ取得開始”、1秒の遅延後に”非同期で取得したデータ”、そして”データ取得完了”という順で出力される結果となります。

○サンプルコード7:データベースアクセスの委譲

Kotlinのプロパティ委譲の特性を活かし、データベースアクセスの部分も委譲を使用してスマートに実装することができます。

ここでは、データベースからのデータ取得を委譲を用いて行う方法を詳細に解説します。

import kotlin.reflect.KProperty

class DatabaseDelegate(private val query: String) {
    private var cachedData: String? = null

    operator fun getValue(thisRef: Any?, property: KProperty<*>): String {
        if (cachedData == null) {
            cachedData = fetchDataFromDatabase(query)
        }
        return cachedData!!
    }

    private fun fetchDataFromDatabase(query: String): String {
        // こちらの部分では、実際のデータベース接続とクエリの実行を行い、
        // 結果を取得する処理を実装します。
        // 今回はサンプルなので、固定の文字列を返しています。
        return "Sample Data from Database"
    }
}

class MyClass {
    val data by DatabaseDelegate("SELECT * FROM sample_table")
}

このコードでは、DatabaseDelegateというクラスを作成しています。

このクラスはデータベースへのアクセスを担当するデリゲートです。

具体的には、初回アクセス時にデータベースからデータを取得し、それをキャッシュとして保持します。

2回目以降のアクセス時には、キャッシュされたデータを返します。

MyClassというクラスでは、dataというプロパティをDatabaseDelegateを用いて委譲しています。

その結果、dataにアクセスするたびに、デリゲートがデータベースからのデータ取得を行います。

このコードを実行すると、MyClassのインスタンスを生成し、そのdataプロパティにアクセスすることで、データベースからのデータ取得が行われ、Sample Data from Databaseという文字列が取得される結果となります。

ただし、上記のコードはデモ用のものであり、実際のデータベース接続の詳細な処理は省略しています。

実際の実装では、適切なデータベース接続処理やエラーハンドリングなどの考慮が必要です。

○サンプルコード8:委譲を使用したUIの更新

KotlinでUIの更新に関しては、通常、アクティビティやフラグメントのライフサイクルを考慮しなければなりません。

特に非同期処理を伴う場合、UIの更新を行うタイミングでそのコンポーネントがまだ生きているかどうかを確認する必要があります。

Kotlinのプロパティ委譲を使用することで、このような問題をエレガントに解決することができます。

まず、UIの更新に関する状態を持つクラスを考えます。

このクラスはプロパティ委譲を使用して、UIの状態を安全に更新することができるように設計されています。

import kotlin.properties.Delegates

class UIUpdater {
    var uiState: String by Delegates.observable("<初期状態>") { _, old, new ->
        println("UIの状態が $old から $new へ変更されました。")
        // ここでUIの更新ロジックを実装します。
    }
}

fun main() {
    val updater = UIUpdater()
    updater.uiState = "読み込み中"
    updater.uiState = "読み込み完了"
}

このコードでは、Delegates.observableを使ってuiStateプロパティの変更を監視しています。

プロパティの値が変わるたびに、監視ロジックがトリガーされ、UIの状態の変更がコンソールに表示されます。

このコードを実行すると、次のような出力が得られます。

UIの状態が <初期状態> から 読み込み中 へ変更されました。
UIの状態が 読み込み中 から 読み込み完了 へ変更されました。

このように、プロパティの変更をトリガーとしてUIの更新ロジックを実行することができます。

実際のアプリケーションでは、このロジックの部分に具体的なUI更新のコード(例えば、テキストの変更や画像の読み込みなど)を実装します。

○サンプルコード9:複数のデリゲートの組み合わせ

Kotlinにおけるプロパティ委譲は非常に強力な機能であり、多くの場面でその恩恵を受けることができます。

特に、複数のデリゲートを組み合わせることにより、更なる柔軟性と効率性を持ったコードの実装が可能となります。

複数のデリゲートを組み合わせたサンプルコードを紹介します。

import kotlin.properties.ReadOnlyProperty
import kotlin.reflect.KProperty

class CompositeDelegate<T>(
    vararg delegates: ReadOnlyProperty<Any?, T>
) : ReadOnlyProperty<Any?, T> {
    private val delegateList = delegates.toList()

    override fun getValue(thisRef: Any?, property: KProperty<*>): T {
        for (delegate in delegateList) {
            try {
                return delegate.getValue(thisRef, property)
            } catch (e: Exception) {
                // 特に処理を行わない
            }
        }
        throw IllegalArgumentException("No valid delegate found")
    }
}

class MyClass {
    val prop: String by CompositeDelegate(
        FirstDelegate(),
        SecondDelegate()
    )
}

class FirstDelegate : ReadOnlyProperty<Any?, String> {
    override fun getValue(thisRef: Any?, property: KProperty<*>): String {
        throw Exception("FirstDelegate Exception")
    }
}

class SecondDelegate : ReadOnlyProperty<Any?, String> {
    override fun getValue(thisRef: Any?, property: KProperty<*>): String {
        return "Value from SecondDelegate"
    }
}

fun main() {
    val myClass = MyClass()
    println(myClass.prop)
}

このコードでは、CompositeDelegateという新しいデリゲートを定義しています。

このデリゲートは、複数のデリゲートを受け取ることができ、それらのデリゲートを順番に試行し、最初に成功するデリゲートの結果を返します。

これにより、一つのデリゲートが失敗した場合に、次のデリゲートにフォールバックすることができます。

このコードを実行すると、FirstDelegateが例外をスローするため、次にあるSecondDelegateが実行され、”Value from SecondDelegate”という文字列が出力されます。

○サンプルコード10:エラーハンドリングを伴う委譲

Kotlinのプロパティ委譲には、様々な使い方や応用例がありますが、中でもエラーハンドリングを伴う委譲は非常に実践的な場面で役立ちます。

特に、外部リソースやAPIからのデータ取得時に発生するエラーをキャッチして適切に処理する場面では、この手法が有効です。

ここでは、エラーハンドリングを伴う委譲のサンプルコードとその解説を行います。

import kotlin.reflect.KProperty

class ErrorHandlerDelegate<T>(private val initialValue: T) {
    private var value: T = initialValue

    operator fun getValue(thisRef: Any?, property: KProperty<*>): T {
        return value
    }

    operator fun setValue(thisRef: Any?, property: KProperty<*>, value: T) {
        try {
            this.value = value
        } catch (e: Exception) {
            println("エラーが発生しました: ${e.message}")
        }
    }
}

class SampleClass {
    var data: String by ErrorHandlerDelegate("初期値")
}

fun main() {
    val sample = SampleClass()
    println(sample.data)  // 初期値を表示
    sample.data = "更新値"
    println(sample.data)  // 更新値を表示
}

このコードでは、ErrorHandlerDelegateというカスタムデリゲートを作成しています。

このデリゲートは値のセット時にエラーが発生した場合に、エラーメッセージをコンソールに出力する機能を持っています。

今回のサンプルでは特にエラーを投げるような処理は実装していませんが、このデリゲートを利用することで、値のセット時に何らかのエラーが発生した場合の処理を一元的に行うことができます。

SampleClassというクラスでは、このErrorHandlerDelegateを用いてdataというプロパティの委譲を行っています。

このコードを実行すると、次のような結果が得られます。

初めに、SampleClassのインスタンスを作成し、そのdataプロパティを表示します。

この時点での値は”初期値”ですので、この文字列が出力されます。

次に、このdataプロパティの値を”更新値”に変更し、再度その値を表示します。

この時、”更新値”という文字列が出力されることが確認できます。

今回のサンプルではエラーを意図的に発生させる処理は含まれていませんが、外部リソースやAPIの呼び出し時など、エラーが発生する可能性のある処理を行う際にこのようなエラーハンドリングを伴う委譲を利用することで、安全かつ効率的なコードを実装することができます。

●注意点と対処法

Kotlinのプロパティ委譲を利用する際には、効果的なコーディングだけでなく、注意すべき点も存在します。

特に初心者や経験が浅い開発者は、次の注意点を把握し、トラブルを回避するための対処法を理解しておくことが重要です。

○適切な委譲オブジェクトの選択

プロパティ委譲は多くのバリエーションがあります。

Lazy, Observable, Mapなど、さまざまなデリゲートが存在します。

そのため、どのデリゲートを使用するかの判断が難しく、場合によっては不適切なデリゲートを選択してしまうリスクがあります。

対処法としては、必要とする動作や性能要件をしっかりと明確にしてから、適切なデリゲートを選択することが大切です。

また、公式ドキュメントやコミュニティの情報を参考にすることで、より適切な選択が可能となります。

○過度な委譲の使用

委譲の利用が簡単であるため、過度に使用してしまいがちです。

しかし、すべてのプロパティに委譲を使用すると、コードの読みやすさや保守性が低下する可能性があります。

対処法としては、必要最低限の場所でのみプロパティ委譲を使用するよう心がけましょう。

また、コードレビューを積極的に行い、チーム全体での認識を共有することも重要です。

○カスタムデリゲートの過度なカスタマイズ

カスタムデリゲートを作成する際、過度なカスタマイズを行うと、他の開発者が理解しづらいコードとなりがちです。

対処法としては、カスタムデリゲートの作成時には、シンプルかつ一貫性のある実装を心がけること。

また、十分なコメントを記述することで、後からコードを見た人が理解しやすくなります。

○非同期処理を伴う委譲の扱い

非同期処理を伴う委譲を使用する際には、スレッドの安全性やライフサイクルの管理など、様々な課題が生じることがあります。

対処法としては、非同期処理を伴う委譲を使用する際には、coroutinesFlowなどの非同期処理ライブラリと組み合わせることを検討しましょう。

また、適切なスコープやディスパッチャを使用することで、スレッドの安全性を確保することが可能となります。

○エラーハンドリングの考慮不足

委譲を使用することで、内部で例外が発生した場合のエラーハンドリングが不十分となることがあります。

対処法としては、デリゲート内でのエラーハンドリングを十分に考慮することです。

具体的には、適切なtry-catch構文を使用する、またはエラーハンドリングを伴う委譲の実装を検討しましょう。

●カスタマイズ方法

Kotlinのプロパティ委譲は非常に柔軟性が高いため、独自のカスタマイズが可能です。

ここでは、そのカスタマイズ方法を解説します。

特に、独自のデリゲートプロパティを作成する方法や、既存のデリゲートをより効果的に活用するテクニックに焦点を当てます。

○独自のデリゲートプロパティの作成

Kotlinでは、独自のデリゲートプロパティを作成することができます。

これにより、特定のビジネスロジックや制約を持ったプロパティを簡単に実装することができます。

例として、0以上の整数しか設定できないプロパティを考えます。

class NonNegativeDelegate {
    private var realValue: Int = 0

    operator fun getValue(thisRef: Any?, property: KProperty<*>): Int {
        return realValue
    }

    operator fun setValue(thisRef: Any?, property: KProperty<*>, value: Int) {
        if (value < 0) throw IllegalArgumentException("負の数は設定できません。")
        realValue = value
    }
}

class Sample {
    var nonNegativeValue by NonNegativeDelegate()
}

このコードではNonNegativeDelegateという独自のデリゲートを使って、nonNegativeValueプロパティに0未満の値を設定しようとすると例外を投げるようにしています。

このコードを実行すると、次のような結果が得られます。

fun main() {
    val sample = Sample()

    sample.nonNegativeValue = 5
    println(sample.nonNegativeValue) // 5

    try {
        sample.nonNegativeValue = -1 // 例外が発生
    } catch (e: IllegalArgumentException) {
        println(e.message) // "負の数は設定できません。"と表示される
    }
}

○既存のデリゲートを活用するテクニック

Kotlinで提供されているlazy, observableなどのデリゲートを活用することで、さまざまなカスタマイズができます。

例えば、初めてプロパティにアクセスされたときにログを出力するような動作を実現したい場合、lazyデリゲートをカスタマイズして実現することができます。

val loggedLazyValue: String by lazy {
    println("初めてのアクセスです。")
    "Hello, Kotlin!"
}

このコードを実行すると、loggedLazyValueに初めてアクセスされた時に”初めてのアクセスです。”というログが出力されます。

次回以降のアクセスではログは出力されません。

まとめ

Kotlinのプロパティ委譲は非常に強力であり、多様な使い方が可能です。

初心者から上級者まで、この機能を理解し、上手に活用することで、Kotlinの開発がより楽しく、効率的になるでしょう。

本記事を通じて、プロパティ委譲の基本から応用、カスタマイズ方法までを習得していただけたら幸いです。