我正在为绑定而苦苦挣扎。我正在尝试创建一个百分比分数,当成功测试的数量发生变化(在下面的代码中反映为"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)
);