当javafx中的父属性更改时,清理绑定并更改嵌套属性上的侦听器

时间:2018-07-23 08:37:48

标签: javafx

我有一个具有SimpleXXXXProperty属性的模型类。 Javafx GUI元素使用绑定或更改侦听器进行更新,例如

textField.textProperty().bind(myModel.myModelStatus());

myModel.myModelStatus().addListener((obj,oldv.newv) -> { update here });

当模型类的实例更改时,我重新绑定了控件并再次添加了侦听器。但是,通过内存使用,我可以看到旧模型仍然保留在内存中。

我该怎么做才能删除对模型的所有引用,以便可以对其进行清理?

  • 取消绑定,然后再次绑定
  • 删除听众
  • 两者

在父属性发生更改时,是否有更自动的方式来更新嵌套属性上的绑定和侦听器?

1 个答案:

答案 0 :(得分:4)

要撤消与模型的绑定(包括侦听器)时要考虑的要点:

  • 再次绑定同一属性(例如p1.bind(p2))时,无方向绑定(p1.bind(p3))将自动解除绑定,但是明确地进行绑定(p1.unbind())也不会受到损害。
  • 双向绑定(p1.bindBidirectional(p2)Bindings.bindBidirectional(p1, p2))必须明确地解除绑定(p1.unbindBidirectional(p2)Bindings.unbindBidirectional(p1, p2))。
  • 侦听器必须未注册(prop.removeListener(l))。

第三个是棘手的部分,因为侦听器通常实现为lambda表达式或方法引用。不幸的是,lambda表达式以及方法reference(!)都是不是常量:

// lambdas are not constant
InvalidationListener l1 = obs -> {};
InvalidationListener l2 = obs -> {};

assert l1 != l2; // they are NOT identical!

嗯,这对于lambda来说可能是显而易见的,但是对于方法引用也是如此,这确实很烦人:

// method references are not constant
Runnable runnable1 = this::anyMethod;
Runnable runnable2 = this::anyMethod;

assert runnable1 != runnable2; // they are NOT identical!

这意味着,如果您想取消注册lambda表达式或简单的方法引用,则不能将其注册为侦听器:

// if you register listeners on a property like that...
label.textProperty().addListener(obs -> System.out.println(obs));
label.textProperty().addListener(this::handleLabelInvalid);

// ...these calls WON'T remove them due to the inequality shown above!
label.textProperty().removeListener(obs -> System.out.println(obs));
label.textProperty().removeListener(this::handleLabelInvalid);

解决方案

您必须自己存储对lambda表达式或方法引用的引用。我通常使用final字段:

public class MyClass {
    // store references for adding/removal
    private final InvalidationListener l1 = this::handleLabelInvalid;
    private final InvalidationListener l2 = obs -> System.out.println(obs);

    ...

    public void bind() {
        label.textProperty().addListener(l1);
        label.textProperty().addListener(l2);
    }

    public void unbind() {
        label.textProperty().removeListener(l1);
        label.textProperty().removeListener(l2);
    }

    private void handleLabelInvalid(Observable observable) { ... }
}