Malkaviano
Malkaviano

Reputation: 134

Kotlin Reflection operator get implementation

I'm learning Kotlin, current using Fedora 25 OpenJDK 8 and Kotlin 1.1.

I copied the example from the Kotlin website: https://kotlinlang.org/docs/reference/delegated-properties.html and changed the get operator.

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

class Delegate {
    operator fun getValue(thisRef: Any?, property: KProperty<*>): String {
        // My implementation
        return property.getter.call(thisRef) as String
    }

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

Reading the Reflection documentation the getter expects the object instance and no other parameter, but I only achieved the following error. (Error is abbreviate because it's too big, it's in recursion.)

.
.
.
at info.malk.Example.getP(Delegation.kt)
at sun.reflect.GeneratedMethodAccessor1.invoke(Unknown Source)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
at kotlin.reflect.jvm.internal.FunctionCaller$Method.callMethod(FunctionCaller.kt:98)
at kotlin.reflect.jvm.internal.FunctionCaller$InstanceMethod.call(FunctionCaller.kt:115)
at kotlin.reflect.jvm.internal.KCallableImpl.call(KCallableImpl.kt:107)
at info.malk.Delegate.getValue(Delegation.kt:32)
at info.malk.Example.getP(Delegation.kt)
.
.
.
Caused by: java.lang.reflect.InvocationTargetException
    ... 1024 more
Caused by: java.lang.StackOverflowError
    ... 1024 more

Process finished with exit code 1

Help.

Upvotes: 0

Views: 699

Answers (1)

holi-java
holi-java

Reputation: 30686

Translation Rule says:

For instance, for the property prop the hidden property prop$delegate is generated, and the code of the accessors(getter/setter) simply delegates to this additional property.

so the kotlin property will dispatch the getter/setter to the delegator. when you get/set the value on a property around in the delegate handlers (getValue/setValue) will result in recursive call.

your Delegate should more like this:

class Delegate<T> {

    private var value: T? = null;
    //           ^--- store the proeprty value internal

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

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

Upvotes: 2

Related Questions