我了解到循环不变应该在循环之前,循环体的开始处,循环体的末端以及循环之后为真。如果条件没有副作用,这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中还有另一种更优雅的方式吗?