Get function by name dynamically in Kotlin

10,585

The global functions such as fun myFunc11() { ... } defined in file named i.e. Global.kt are compiled to static methods on a class named GlobalKt as described in the documentation.

To get a function reference by name you'll need to load the class that defines it. If you know the file name that defines the function reference you're trying to find you can do:

fun getFunctionFromFile(fileName: String, funcName: String): KFunction<*>? {
    val selfRef = ::getFunctionFromFile
    val currentClass = selfRef.javaMethod!!.declaringClass
    val classDefiningFunctions = currentClass.classLoader.loadClass("${fileName}Kt")
    val javaMethod  = classDefiningFunctions.methods.find { it.name == funcName && Modifier.isStatic(it.modifiers)}
    return javaMethod?.kotlinFunction
}

Then you can find and call function defined in Global.kt file:

fun myFunc11() { println("Very useful function 11") }

like so:

val kFunction = getFunctionFromFile("Global", "myFunc11")
kFunction?.call()

However the above is pretty useless. A better solution would be to search through all classes available in classpath and suffixed with Kt to reach all global functions. However due to nature of jvm class loaders this is a bit more involved as described in this answer.

Share:
10,585

Related videos on Youtube

Amir Abiri
Author by

Amir Abiri

Updated on September 14, 2022

Comments

  • Amir Abiri
    Amir Abiri over 1 year

    How can I dynamically get a function by name in Kotlin ?

    i.e:

    fun myFunc11() { println("Very useful function 11") }
    
    val funcName = "myFunc" + 11
    val funcRef = getFunction(funcName)
    
    funcRef()
    

    Edit: The accepted answer appears to be correct, however the code is currently hitting some bug in Kotlin. Bug report submitted: https://youtrack.jetbrains.com/issue/KT-10690

  • Amir Abiri
    Amir Abiri over 2 years
    The question is about getting a function object dynamically by its name where the name isn't known at compile time (i.e "myFunc11") - not how to get the name of a known function.