news

Sabtu, 13 Juni 2020

Function Inside Function didalam Kotlin


Function Inside Function didalam Kotlin

Ketika mengembangkan sebuah proyek, kita pasti membuat beberapa fungsi tersendiri dengan tujuan untuk memisahkan logika program dari fungsi utama. 

Tujuannya adalah agar kode lebih terstruktur dan mudah dibaca. Namun pada praktiknya, terkadang kode yang ada pada fungsi tersebut malah lebih panjang dan susah dibaca. Salah satu penyebabnya adalah karena penulisan kode yang berulang atau lainnya.
Untuk mengatasinya, kita bisa memisahkannya lagi menjadi sebuah fungsi lokal (inner function) dengan hak akses terbatas hanya untuk fungsi terluarnya. Ini bisa dilakukan karena pada Kotlin kita bisa mendefinisikan sebuah fungsi di mana pun, bahkan di dalam sebuah fungsi (function inside function).
Berikut adalah contoh dari sebuah inner function:

  1. fun setWord(message: String) {

  2.    fun printMessage(text: String) {

  3.        println(text)

  4.    }

  5.  

  6.    printMessage(message)

  7. }


Bisa diperhatikan bahwa fungsi printMessage() didefinisikan di dalam fungsi setWord(). Mendefinisikan sebuah inner function sama halnya seperti kita mendefinisikan sebuah fungsi seperti biasanya. Menariknya, kita bisa mengakses apa yang menjadi bagian fungsi terluarnya. Contoh, parameter dari fungsi setWord() bisa diakses dari dalam fungsi print sehingga kode di atas bisa diubah menjadi seperti berikut:

  1. fun setWord(message: String) {

  2.    fun printMessage() {

  3.        println(message)

  4.    }

  5.  

  6.    printMessage()

  7. }


Lebih sederhana bukan? Perlu diperhatikan, inner function hanya bisa diakses setelah fungsi tersebut didefinisikan. Jika kita coba mengaksesnya, maka akan tampil eror seperti berikut:

  1. fun setWord(message: String) {

  2.    printMessage() // ERROR: Unresolved references

  3.  

  4.    fun printMessage() {

  5.        println(message)

  6.    }

  7.  

  8.    printMessage()

  9. }


Lalu, pada kondisi seperti apa kita bisa memanfaatkan inner function? Perhatikan deklarasi fungsi berikut di bawah ini:

  1. fun sum(valueA: Int, valueB: Int, valueC: Int): Int {

  2.    if (valueA == 0) {

  3.        throw IllegalArgumentException("valueA must be better than 0")

  4.    }

  5.  

  6.    if (valueB == 0) {

  7.        throw IllegalArgumentException("valueB must be better than 0")

  8.    }

  9.  

  10.    if (valueC == 0) {

  11.        throw IllegalArgumentException("valueC must be better than 0")

  12.    }

  13.  

  14.    return valueA + valueB

  15. }


Tidak ada yang salah dengan semua fungsi di atas. Fungsi tersebut akan berjalan dengan semestinya tanpa adanya eror selama kondisi yang berada di dalamnya tidak terpenuhi. Namun jika kita perhatikan, terdapat pengulangan kode yang sama yaitu penggunaan if expression untuk memeriksa apakah nilai dari argumen yang diberikan bernilai null.
Di sinilah kita bisa memanfaatkan inner function untuk membuat kode yang ditulis berulang tersebut menjadi fungsi tersendiri.

  1. fun sum(valueA: Int, valueB: Int, valueC: Int): Int {

  2.    fun validateNumber(value: Int) {

  3.        if (value == 0) {

  4.            throw IllegalArgumentException("value must be better than 0")

  5.        }

  6.    }

  7.  

  8.    validateNumber(valueA)

  9.    validateNumber(valueB)

  10.    validateNumber(valueC)

  11.  

  12.    return valueA + valueB

  13. }


Setelah menjadikannya sebagai sebuah fungsi tersendiri, kode yang ada di dalam fungsi sum() tersebut lebih singkat dan tentunya lebih mudah dibaca dibandingkan sebelumnya.

Selain itu, kita juga bisa menjadikan inner function sebagai extensions function. Contohnya seperti berikut:

  1. fun sum(valueA: Int, valueB: Int, valueC: Int): Int {

  2.    fun Int.validateNumber(){

  3.        if (this == 0) {

  4.            throw IllegalArgumentException("value must be better than 0")

  5.        }

  6.    }

  7.  

  8.    valueA.validateNumber()

  9.    valueB.validateNumber()

  10.    valueC.validateNumber()

  11.  

  12.    return valueA + valueB

  13. }