forked from dlang/dlang.org
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcontracts.dd
247 lines (207 loc) · 7.09 KB
/
contracts.dd
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
Ddoc
$(SPEC_S Contract Programming,
$(P Contracts are a breakthrough technique to reduce the programming effort
for large projects. Contracts are the concept of preconditions, postconditions,
errors, and invariants.
Contracts can be done in C++ without modification to the language,
but the result is
clumsy and inconsistent.)
$(P Building contract support into the language makes for:)
$(OL
$(LI a consistent look and feel for the contracts)
$(LI tool support)
$(LI it's possible the compiler can generate better code using information gathered
from the contracts)
$(LI easier management and enforcement of contracts)
$(LI handling of contract inheritance)
)
$(COMMENT <img src="images/d4.gif" alt="Contracts make D bug resistant" border=0>)
$(P The idea of a contract is simple - it's just an expression that must evaluate to true.
If it does not, the contract is broken, and by definition, the program has a bug in it.
Contracts form part of the specification for a program, moving it from the documentation
to the code itself. And as every programmer knows, documentation tends to be incomplete,
out of date, wrong, or non-existent. Moving the contracts into the code makes them
verifiable against the program.)
$(H2 Assert Contract)
The most basic contract is the
$(GLINK2 expression, AssertExpression).
An $(B assert) inserts a checkable expression into
the code, and that expression must evaluate to true:
------
assert(expression);
------
C programmers will find it familiar. Unlike C, however, an <code>assert</code>
in function bodies
works by throwing an <code>AssertError</code>,
which can be caught and handled. Catching the contract violation is useful
when the code must deal with errant uses by other code, when it must be
failure proof, and as a useful tool for debugging.
$(P When compiling for release, the assert code is not generated.
The special $(D assert(0)) expression, however, is generated even in
release mode. See the $(GLINK2 expression, AssertExpression) documentation
for more information.)
$(H2 Pre and Post Contracts)
The pre contracts specify the preconditions before a statement is executed. The most
typical use of this would be in validating the parameters to a function. The post
contracts validate the result of the statement. The most typical use of this
would be in validating the return value of a function and of any side effects it has.
The syntax is:
------
in
{
...contract preconditions...
}
out (result)
{
...contract postconditions...
}
body
{
...code...
}
------
$(P By definition, if a pre contract fails, then the body received bad
parameters.
An AssertError is thrown. If a post contract fails,
then there is a bug in the body. An AssertError is thrown.)
$(P Either the <code>in</code> or the <code>out</code> clause can be omitted.
If the <code>out</code> clause is for a function
body, the variable <code>result</code> is declared and assigned the return
value of the function. For example, let's implement a square root function:)
------
long square_root(long x)
in
{
assert(x >= 0);
}
out (result)
{
assert((result * result) <= x && (result+1) * (result+1) > x);
}
body
{
return cast(long)std.math.sqrt(cast(real)x);
}
------
$(P The assert's in the in and out bodies are called <dfn>contracts</dfn>.
Any other D
statement or expression is allowed in the bodies, but it is important
to ensure that the
code has no side effects, and that the release version of the code
will not depend on any effects of the code.
For a release build of the code, the in and out code is not
inserted.)
$(P If the function returns a void, there is no result, and so there can be no
result declaration in the out clause.
In that case, use:)
------
void func()
out
{
...contracts...
}
body
{
...
}
------
In an out statement, $(I result) is initialized and set to the
return value of the function.
$(H2 In, Out and Inheritance)
$(P If a function in a derived class overrides a function in its
super class, then only one of
the $(D in) contracts of the function and its base functions
must be satisfied.
Overriding
functions then becomes a process of $(I loosening) the $(D in)
contracts.
)
$(P A function without an $(D in) contract means that any values
of the function parameters are allowed. This implies that if any
function in an inheritance hierarchy has no $(D in) contract,
then $(D in) contracts on functions overriding it have no useful
effect.
)
$(P Conversely, all of the $(D out) contracts needs to be satisfied,
so overriding functions becomes a processes of $(I tightening) the
$(D out)
contracts.
)
$(H2 Invariants)
$(P Invariants are used to specify characteristics of a class or struct that
always must be true (except while executing a member function).
For example, a class representing a date might have an invariant that the
day must be 1..31 and the hour must be 0..23:
)
------
class Date
{
int day;
int hour;
invariant() {
assert(1 <= day && day <= 31);
assert(0 <= hour && hour < 24);
}
}
------
$(P The invariant is a contract saying that the asserts must hold true.
The invariant is checked when a class or struct constructor completes,
at the start of the class or struct destructor. For public or exported
functions, the order of execution is:
)
$(OL
$(LI preconditions)
$(LI invariant)
$(LI body)
$(LI invariant)
$(LI postconditions)
)
$(P The code in the invariant may not call any public non-static members
of the class or struct, either directly or indirectly.
Doing so will result in a stack overflow, as the invariant will wind
up being called in an infinitely recursive manner.
)
$(P Invariants are implicitly const.)
$(P Since the invariant is called at the start of public or
exported members, such members should not be called from
constructors.
)
------
class Foo {
public void f() { }
private void g() { }
invariant() {
f(); // error, cannot call public member function from invariant
g(); // ok, g() is not public
}
}
------
The invariant can be checked with an <code>assert()</code> expression:
$(OL
$(LI classes need to pass a class object)
$(LI structs need to pass the address of an instance)
)
------
auto mydate = new Date(); //class
auto s = S(); //struct
...
assert(mydate); // check that class Date invariant holds
assert(&s); // check that struct S invariant holds
------
$(P Invariants contain assert expressions, and so when they fail,
they throw a $(D AssertError)s.
Class invariants are inherited, that is,
any class invariant is implicitly anded with the invariants of its base
classes.)
$(P There can be only one $(I Invariant) per class or struct.)
$(P When compiling for release, the invariant code is not generated, and the compiled program runs at maximum speed.)
$(H2 References)
$(LIST
$(XLINK2 http://people.cs.uchicago.edu/~robby/contract-reading-list/, Contracts Reading List),
$(XLINK2 http://jan.newmarch.name/java/contracts/paper-long.html, Adding Contracts to Java)
)
)
Macros:
TITLE=Contract Programming
WIKI=DBC
CATEGORY_SPEC=$0