6

我知道您不能在 Java 中使用委托属性语法,并且不会像在 Kotlin 中那样获得“覆盖”set/get 运算符的便利,但我仍然想在 Java 中使用现有的属性委托。

例如,一个简单的 int 委托:

class IntDelegate {
    operator fun getValue(thisRef: Any?, property: KProperty<*>) = 0
}

当然,在 Kotlin 中我们可以这样使用:

val x by IntDelegate()

但是我们如何IntDelegate在 Java 中以某种形式使用呢?这是开始,我相信:

final IntDelegate x = new IntDelegate();

然后直接使用函数。但是我怎样才能使用这个getValue功能呢?我为它的参数传递什么?如何获得KPropertyfor Java 字段?

4

2 回答 2

4

如果您真的想知道 Kotlin 委托属性在 Java 中的外观,这里是:在此示例x中,Java 类的属性JavaClass被委托给Delegates.notNull标准委托。

// delegation implementation details
import kotlin.jvm.JvmClassMappingKt;
import kotlin.jvm.internal.MutablePropertyReference1Impl;
import kotlin.jvm.internal.Reflection;
import kotlin.reflect.KProperty1;

// notNull property delegate from stdlib
import kotlin.properties.Delegates;
import kotlin.properties.ReadWriteProperty;


class JavaClass {
    private final ReadWriteProperty<Object, String> x_delegate = Delegates.INSTANCE.notNull();
    private final static KProperty1 x_property = Reflection.mutableProperty1(
            new MutablePropertyReference1Impl(
                JvmClassMappingKt.getKotlinClass(JavaClass.class), "x", "<no_signature>"));

    public String getX() {
        return x_delegate.getValue(this, x_property);
    }

    public void setX(String value) {
        x_delegate.setValue(this, x_property, value);
    }
}

class Usage {
    public static void main(String[] args) {
        JavaClass instance = new JavaClass();
        instance.setX("new value");
        System.out.println(instance.getX());
    }
}

但是我不建议使用这个解决方案,不仅因为需要样板文件,而且因为它严重依赖于委托属性和 kotlin 反射的实现细节。

于 2017-07-19T02:52:10.873 回答
1

我知道您不能在 Java 中使用委托属性语法,并且不会像在 Kotlin 中那样获得“覆盖”set/get 运算符的便利,但我仍然想在 Java 中使用现有的属性委托。

不,就像你在开始时所说的那样,它在 Java 中不存在。但如果你坚持这样做,你可以做类似的事情。

public interface Delegate<T> {
    T get();
    void set(T value);
}

public class IntDelegate implements Delegate<Integer> {
    private Integer value = null;

    @Override
    public void set(Integer value) {
        this.value = value;
    }

    @Override
    public Integer get() {
        return value;
    }
}

final Delegate<Integer> x = new IntDelegate();

Delcare x in interface 允许你有不同的实现。

于 2017-07-18T06:20:30.577 回答