软件构造笔记 3.2 Designing Specification

1.函数与方法

参数类型和返回类型匹配在静态检查阶段完成。

“方法”是程序的“积木”,可以被独立开发、测试、复用;

 使用“方法”的客户端,无需了解方法内部具体如何工作——“抽象”。

2.方法的规约

需要规约的原因:很多bug来自于双方的误解;不写清楚规约,不同开发者的理解会不同;没有规约难以定位错误。

规约的优点:精确的规划,有助于区分责任;客户端无需阅读调用函数的代码,只需阅读spec。


规约可以隔离“变化”,无需通知客户端;扮演防火墙的角色。

3.行为等价性

行为等价性的关键是确定两个实现间是否可以相互替换,需要站在客户端的角度看待。具体可以通过规约来判断,若两个函数符合一致的规约,那么它们是等价的。


4.前置/后置条件

前置条件:对客户端的约束,在使用方法时必须满足的条件。

后置条件:对开发者的约束,方法结束时必须满足的条件。

契约:如果前置条件满足了,后置条件必须满足。

前置条件不满足, 那么方法可以做任何事情,“你违约在先,我自然不遵守承诺”。

静态类型声明是一种规约,可据此进行静态类型检查static checking。方法前的注释也是一种规约,但需人工判定其是否满足。


注意:

除非在后置条件里声明过,否则方法内部不应该改变输入参数。

应尽量遵循此规则,尽量不设计 mutatingspec,否则就容易引发bugs

程序员之间应达成的默契:除非spec必须如此,否则不 应修改输入参数。

尽量避免使用mutable的对象。

避免使用可变的全局变量!但是为了性能原因,有时候却不得不用,这对程序的安全性造成了巨大破坏。

5.欠定规约、非确定规约

确定的规约:给定一个满足precondition的输入,其输出是唯一的、明确的。

欠缺的规约:同一个输入可能有多个输出。

非确定的规约:同一个输入,多次执行时得到的结果可能不同。

6.陈述式、操作式规约

操作式规约,例如伪代码。

声明式规约:没有内部实现的描述,只有“初-终”状态。声明式规约更有价值。


7.规约的强度及其比较

当判断两个规约间的替换关系时,引入规约的强度:若S2的前置条件更弱且后置条件更强,那么S2的强度更强,可以用S2替代S1。


8.图像规约(Diagramingspecification

对于一个实现,若满足规约则落在范围内,否则在范围外。程序员可以在规约的范围内自由选择实现方式,客户端无需了解使用了哪个实现。


更弱的前置条件意味着实现时要处理的输入更多,更强的后置条件意味着实现的自由度更低,都会使得面积减小。更强的规约表示为更小的区域。



9.设计好的规约

1)spec描述的功能应单一、简单、易理解。

2)不能让客户端产生理解的歧义。

3)具有强度,即为客户端强有力的保证。

4)在不能完全保证功能时需要足够弱。

5)在规约里面使用抽象类型,可以给方法的实现体与客户端更大的自由度。

6)注意前置条件和后置条件。不写前置条件,那么就需要在代码内部check,但客户端也不喜欢太强的前置条件。惯用做法是不限定太强的前置条件,而是在后置条件中抛出异常,尽可能的在错误的根源处fail,避免大规模扩散。

是否使用前置条件取决于方法的使用范围;check的代价;

如果只在类的内部使用该方法(private),那么可以不使用前置条件,在使用该方法的各个位置进行check,责任交给内部client。

如果在其他地方使用该方法(public),那么必须要使用前置条件,若client端不满足则方法抛出异常。






  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值