JavaFX 8 Bindings.when and Bindings.divide creates Divide by



我正在为绑定而苦苦挣扎。我正在尝试创建一个百分比分数,当成功测试的数量发生变化(在下面的代码中反映为"successCountProperty"(或测试总数发生变化(反映在"结果"的大小属性中(时,该分数会更新。当我执行这段代码时,我得到java.lang.ArithmeticException:/by zero。当我最初遇到异常时,我发现了 Bindings.when((.then((.else((,在我的脑海中应该可以解决问题。不幸的是,在执行此代码时,尽管"when"返回 false,但"then"仍在执行。谁能帮忙?

public void foo()
{
    DoubleProperty scoreProperty = new SimpleDoubleProperty(0);
    ListProperty<String> results = new SimpleListProperty<>(FXCollections.observableArrayList());
    IntegerProperty successCountProperty = new SimpleIntegerProperty(0);
    scoreProperty.bind(Bindings.when(results.sizeProperty().greaterThan(0))
            .then(Bindings.divide(successCountProperty, results.sizeProperty())).otherwise(0));
}

使用 Bindings 方法或属性本身的更复杂的绑定很容易变得难以阅读且难以维护。

在这种情况下,我建议使用自定义评估方法创建一个DoubleBinding

scoreProperty.bind(Bindings.createDoubleBinding(() -> {
    int size = results.size();
    return size == 0 ? 0d : ((double) successCountProperty.get()) / size;
}, results.sizeProperty(), successCountProperty));

你的问题是

Bindings.divide(successCountProperty, results.sizeProperty())

立即进行评估,您似乎期待懒惰评估。

您可以编写自己的惰性除法器,该除法仅在实际查询值时执行除法:

class LazyDivision extends DoubleBinding
{
    private IntegerProperty dividend;
    private ReadOnlyIntegerProperty divisor;
    LazyDivision(final IntegerProperty dividend, final ReadOnlyIntegerProperty divisor)
    {
        this.dividend = dividend;
        this.divisor = divisor;
    }
    @Override
    protected double computeValue()
    {
        return dividend.getValue() / divisor.getValue().doubleValue();
    }
}

然后像这样使用它:

scoreProperty.bind(
    Bindings
        .when(results.sizeProperty().greaterThan(0))
        .then(
            new LazyDivision(successCountProperty, results.sizeProperty())
        )
        .otherwise(0)
);

最新更新