为什么我的循环不变量可能不会被任何迭代保留?

Why my loop invariant might not be preserved by any iteration?

我正在尝试在 Spark 中编写代码,使用 Horner 方法计算多项式的值。变量Result由Horner计算得到,变量Z 以经典方式计算。我做了很多不同的测试,我的循环不变量总是正确的。但是,我收到消息:

loop invariant might not be preserved by an arbitrary iteration

是否存在循环不变量不起作用的情况,或者您是否需要向不变量添加更多条件?

这是调用 Horner 函数的主要函数:

  with Ada.Integer_Text_IO;
  use Ada.Integer_Text_IO;

  with Poly;
  use Poly;

  procedure Main is
     X : Integer;
     A : Vector (0 .. 2);
  begin
     X := 2;
     A := (5, 10, 15);

     Put(Horner(X, A));
  end Main;

和霍纳函数:

package body Poly with SPARK_Mode is
function Horner (X : Integer; A : Vector) 
  return Integer 
  is

  Result : Integer := 0;
  Z : Integer := 0;

  begin
     for i in reverse A'Range loop
        
        pragma Loop_Invariant (Z=Result*(X**(i+1)));

        Result := Result*X + A(i); 
        Z := Z + A(i)*X**(i);

     end loop;

     pragma Assert (Result = Z);

     return Result;
end Horner;
end Poly;

Vector是怎么定义的?是不是有点像

type Vector is array(Integer range <>) of Float;

在这种情况下,如果 A 的某些索引为负,条件可能会失败。另外,即使 A 的第一个索引不为零,不变量是否成立?也许不变量失败的另一种情况是 A'Last = Integer'Last;在这种情况下,i+1 会溢出。

通常SPARK在无法证明某事时给出一个理由,一个反例。我建议您检查一下,它可以让您了解不变量何时失败。请记住,反例有时是一些非常极端的情况,例如 A'Last = Integer'Last。如果是这种情况,您需要告诉 SPARK A'Last = Integer'Last 永远不会发生,可能通过为 Vector 索引定义特定子类型或向您的函数添加契约。