2

我得到了一项任务,我必须删除 JML 产生的每个警告。
如果我在构造函数 my 中调用一个方法requires并且ensures不再验证,尽管为被调用函数添加了相同的约束。
我被要求仅使用requires,ensures和。 这是代码:invariantloop_invariant

  /*@ non_null @*/ int[] elements;
  int n;
  //@ invariant n >= 0;
  
  //@ requires input != null;
  //@ requires input.length >= 0;
  //@ ensures elements != null;
  Class(int[] input) {
    n = input.length;
    elements = new int[n];
    arraycopy(input, 0, elements, 0, n);
    //@ assert n >= 0;
  }
  
  //@ requires srcOff >= 0;
  //@ requires destOff >= 0;
  //@ requires length >= 0;
  //@ requires dest.length >= destOff + length;
  //@ requires src.length >= srcOff + length;
  //@ ensures dest.length == \old(dest.length);
  //@ ensures length == \old(length) ==> length >= 0;
  //@ ensures dest != null;
  private static void arraycopy(/*@ non_null @*/ int[] src,
                                int   srcOff,
                                /*@ non_null @*/ int[] dest,
                                int   destOff,
                                int   length) {
     
     //@ loop_invariant destOff+i >= 0;
     //@ loop_invariant srcOff+i >= 0;
     //@ loop_invariant length >= 0;
     for(int i=0 ; i<length; i=i+1) {
        dest[destOff+i] = src[srcOff+i];
    }
  }

和产生的输出:

Class.java:25: warning: The prover cannot establish an assertion (NullField) in method Class
  /*@ non_null @*/ int[] elements;
                         ^
Class.java:32: warning: The prover cannot establish an assertion (InvariantExit: MultiSet.java:27: ) in method Class
  Class(int[] input) {
  ^
Class.java:27: warning: Associated declaration: Class.java:32: 
  //@ invariant n >= 0;
      ^
3 warnings

一种解决方案是使函数arraycopy非静态,但我不明白为什么。

4

1 回答 1

0

证明者无法确定类变量是否随着函数对 和 的可见性n而改变elements。因此它应该需要一个注释

//@ ensures n == \old(n)
//@ ensures elements == \old(elements)

这是一个问题,有两个不同的原因:

  1. 在 Java 中,静态方法无法访问非静态变量的值,因此 JML 无法证明以下规范(显示工具限制)。
// ...
//@ ensures n == \old(n)
//@ ensures elements == \old(elements)
private static void arraycopy( /* ... */ ) {
  1. 第二个ensures可能会导致一些问题 elements作为src论据arraycopy

为了避免修改函数签名,您需要在每次函数调用assume后添加规范。arraycopy

于 2020-12-24T13:11:14.747 回答