检查循环不变的优雅方式

时间:2017-12-12 16:05:37

标签: java invariants loop-invariant

我了解到循环不变应该在循环之前,循环体的开始处,循环体的末端以及循环之后为真。如果条件没有副作用,这4个点可以减少到2个点:在循环之前和循环体的末尾。这是Java中的一个示例,我在其中检查不变量assert

private static int cube(int n) {
    int t = 1;
    int s = 0;
    int i = 0;

    assert (s == i*i*i && t == (i-1)*(i-1) && i <= n);

    while (i < n) {
        t = t + 2*i - 1;
        s = s + 3*t + 3*i + 1;
        i++;
        assert (s == i*i*i && t == (i-1)*(i-1) && i <= n);
    }

    return s;
}

我认为不变表达式的冗余非常糟糕。如果Java允许我这样做,断言可以在条件之前减少到一个断言点。

while ( i < n)
       ^

所以我提出了另一种概念解决方案,并将条件分解为另一种方法:

private static int cubeElegant(int n){
    int t = 1;
    int s = 0;
    int i = 0;

    while (condition(n, t, s, i)){
        t = t + 2*i - 1;
        s = s + 3*t + 3*i + 1;
        i++;
    }

    return s;
}

private static boolean condition(int n, int t, int s, int i){
    assert (s == i*i*i && t == (i-1)*(i-1) && i <= n);
    return i < n;
}

我认为可读性非常差,代码不必要地膨胀。我的解决方案是否可以通过合同不变检查以某种方式实现设计,而无需使用第三方库? Java中还有另一种更优雅的方式吗?

0 个答案:

没有答案