给定一个程序 c c c,由以下specification注解:
{ P } c { Q } \{P\}c\{Q\} {P}c{Q}
为了证明这个三元组,我们构造一个验证条件(verification condition, VC)的集合
给定一个断言 Q Q Q和一个程序 c c c,一个谓词转换器(predicate transformer)是一个函数,输出另一个断言
最弱前置条件(weakest precondition)谓词转换器产生一个 w p ( c , Q ) wp(c,Q) wp(c,Q),使得
我们用霍尔三元组来定义 w l p wlp wlp
比如 w l p ( y : = x + 1 , ( ∀ x . x < z → x < y ) → x + 1 ≤ y ) wlp(y:=x+1, (\forall x.x
注意,答案并不是 ( ∀ x . x < z → x < x + 1 ) → x + 1 ≤ x + 1 (\forall x.x
因为当我们用 x + 1 x+1 x+1替换 y y y以处理 ( ∀ x . x < z → x < y ) (\forall x.x
当我们扩展 P [ a / x ] P[a/x] P[a/x]时,我们需要:
数组赋值的霍尔规则可以表示为:
A s g n A r r { Q [ x ⟨ a 1 ◃ a 2 ⟩ / x ] } x [ a 1 ] : = a 2 { Q } AsgnArr~\frac{}{\{Q[x\langle a_1\triangleleft a_2\rangle /x]\}x[a_1]:=a_2\{Q\}} AsgnArr {Q[x⟨a1◃a2⟩/x]}x[a1]:=a2{Q}
相应的转换器即为
w l p ( x [ a 1 ] : = a 2 , Q ) = Q [ x ⟨ a 1 ◃ a 2 ⟩ / x ] wlp (x[a_1]:=a_2,Q)=Q[x\langle a_1\triangleleft a_2\rangle /x] wlp(x[a1]:=a2,Q)=Q[x⟨a1◃a2⟩/x]
举例:
计算 w l p ( b [ i ] : = 5 , b [ i ] = 5 ) wlp(b[i]:=5,b[i]=5) wlp(b[i]:=5,b[i]=5)
w l p ( b [ i ] : = 5 , b [ i ] = 5 ) = ( b ⟨ ◃ 5 ⟩ [ i ] = 5 ) = ( 5 = 5 ) = t r u e wlp(b[i]:=5,b[i]=5)=(b\langle\triangleleft 5\rangle [i]=5)=(5=5)=true wlp(b[i]:=5,b[i]=5)=(b⟨◃5⟩[i]=5)=(5=5)=true
计算 w l p ( b [ n ] : = x , ∀ i . 1 ≤ i < n → b [ i ] ≤ b [ i + 1 ] ) wlp(b[n]:=x,\forall i.1\le i
进行代入
w l p ( b [ n ] : = x , ∀ i . 1 ≤ i < n → b [ i ] ≤ b [ i + 1 ] ) = ∀ i . 1 ≤ i < n → ( b ⟨ ◃ x ⟩ ) [ i ] ≤ ( b ⟨ n ◃ x ⟩ ) [ i + 1 ] = ( b ⟨ n ◃ x ⟩ ) [ n − 1 ] ≤ ( b ⟨ n ◃ x ⟩ ) [ n ] ∧ ∀ i . 1 ≤ i < n − 1 → ( b ⟨ n ◃ x ⟩ ) [ i ] ≤ ( b ⟨ n ◃ x ⟩ ) [ i + 1 ] wlp(b[n]:=x,\forall i.1\le i
依据霍尔规则
S e q { P } c 1 { P ′ } { P ′ } c 2 { Q } { P } c 1 ; c 2 { Q } Seq~\frac{\{P\}c_1\{P'\}\qquad\{P'\}c_2\{Q\}}{\{P\}c_1;c_2\{Q\}} Seq {P}c1;c2{Q}{P}c1{P′}{P′}c2{Q}
相应的谓词转换器即为
w l p ( c 1 ; c 2 , Q ) = w l p ( c 1 , w l p ( c 2 , Q ) ) wlp(c_1;c_2,Q)=wlp(c_1,wlp(c_2,Q)) wlp(c1;c2,Q)=wlp(c1,wlp(c2,Q))
依据霍尔规则
I f { P ∧ b } c 1 { Q } { P ∧ ¬ b } c 2 { Q } { P } i f b t h e n c 1 e l s e c 2 { Q } If~\frac{\{P\wedge b\}c_1\{Q\}\qquad\{P\wedge\neg b\}c_2\{Q\}}{\{P\}\mathbf{if}~b~\mathbf{then}~c_1~\mathbf{else}~c_2\{Q\}} If {P}if b then c1 else c2{Q}{P∧b}c1{Q}{P∧¬b}c2{Q}
相应的转换器即为
w l p ( i f b t h e n c 1 e l s e c 2 , Q ) = ( b → w l p ( c 1 , Q ) ) ∧ ( ¬ b → w l p ( c 2 , Q ) ) wlp(\mathbf{if}~b~\mathbf{then}~c_1~\mathbf{else}~c_2,Q)\\ =(b\to wlp(c_1,Q))\wedge (\neg b\to wlp(c_2,Q)) wlp(if b then c1 else c2,Q)=(b→wlp(c1,Q))∧(¬b→wlp(c2,Q))
依据等价关系
w h i l e b d o c ≡ i f b t h e n c ; w h i l e b d o c e l s e s k i p \mathbf{while}~b~\mathbf{do}~c\equiv \mathbf{if}~b~\mathbf{then}~c;\mathbf{while}~b~\mathbf{do}~c~\mathbf{else}~\mathbf{skip} while b do c≡if b then c;while b do c else skip
相应的 w l p wlp wlp即为
此处略,最后转了个圈又回来了。
一般来说,我们无法总是算出循环的 w l p wlp wlp,比如上面的情况。
但是,我们可以借助于循环不变式来近似它
下面,我们使用这种方式表示循环:
w h i l e b d o { I } c \mathbf{while}~b~\mathbf{do}\{I\}c while b do{I}c
这里 I I I是由程序员提供的循环不变量
最为直观的想法是令
w l p ( w h i l e b d o { I } c , Q ) = I wlp(\mathbf{while}~b~\mathbf{do}\{I\}c,Q)=I wlp(while b do{I}c,Q)=I
但此时 I I I可能不是最弱的前置条件
如果我们草率地认为 w l p ( w h i l e b d o { I } c , Q ) = I wlp(\mathbf{while}~b~\mathbf{do}\{I\}c,Q)=I wlp(while b do{I}c,Q)=I,我们漏了两件事情:
所以我们需要构造一个额外的验证条件(verification condition)的集合,
v c ( w h i l e b d o { I } c , Q ) = { I ∧ ¬ b ⇒ Q I ∧ b ⇒ w l p ( c , I ) vc(\mathbf{while}~b~\mathbf{do}\{I\}c,Q)=\begin{cases}I\wedge \neg b\Rightarrow Q\\I\wedge b\Rightarrow wlp(c,I)\end{cases} vc(while b do{I}c,Q)={I∧¬b⇒QI∧b⇒wlp(c,I)
为了在执行循环后确保 Q Q Q能够实现,需要满足两个条件:
while是唯一一个引入额外条件的命令,但是其他的声明可能也包含循环,所以:
综上,我们得到验证 { P } c { Q } \{P\}c\{Q\} {P}c{Q}的通用方法:
若3,4检验均通过,那么 { P } c { Q } \{P\}c\{Q\} {P}c{Q}是永真的,但反之不一定成立,因为循环不变式可能不是最弱的前置条件。