Thinkcspy 3

Download as pdf or txt
Download as pdf or txt
You are on page 1of 415
At a glance
Powered by AI
The document discusses Python programming concepts such as variables, expressions, statements, functions and more.

Some of the main topics covered include variables and data types, expressions, statements, functions, loops, strings, errors and debugging.

Comments are used to explain or annotate code and are ignored by the Python interpreter. They help other programmers understand the code.

How to Think Like a Computer Scientist: Learning with Python 3 Documentation

Release 3rd Edition

Peter Wentworth, Jeffrey Elkner, Allen B. Downey and Chris Meyers

February 09, 2012

CONTENTS

The way of the program 1.1 The Python programming language 1.2 What is a program? . . . . . . . . . 1.3 What is debugging? . . . . . . . . 1.4 Syntax errors . . . . . . . . . . . . 1.5 Runtime errors . . . . . . . . . . . 1.6 Semantic errors . . . . . . . . . . . 1.7 Experimental debugging . . . . . . 1.8 Formal and natural languages . . . 1.9 The rst program . . . . . . . . . . 1.10 Comments . . . . . . . . . . . . . 1.11 Glossary . . . . . . . . . . . . . . 1.12 Exercises . . . . . . . . . . . . . . Variables, expressions and statements 2.1 Values and data types . . . . . . . 2.2 Variables . . . . . . . . . . . . . 2.3 Variable names and keywords . . 2.4 Statements . . . . . . . . . . . . 2.5 Evaluating expressions . . . . . . 2.6 Operators and operands . . . . . 2.7 Type converter functions . . . . . 2.8 Order of operations . . . . . . . . 2.9 Operations on strings . . . . . . . 2.10 Input . . . . . . . . . . . . . . . 2.11 Composition . . . . . . . . . . . 2.12 The modulus operator . . . . . . 2.13 Glossary . . . . . . . . . . . . . 2.14 Exercises . . . . . . . . . . . . . Hello, little turtles! 3.1 Our rst turtle program . . . . . . 3.2 Instances a herd of turtles . . . 3.3 The for loop . . . . . . . . . . . 3.4 Flow of Execution of the for loop

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

1 1 3 3 3 4 4 4 5 6 7 7 8 11 11 13 14 15 16 16 17 18 19 20 20 21 22 23 25 25 27 29 30

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

3.5 3.6 3.7 3.8 4

The loop simplies our turtle program A few more turtle methods and tricks Glossary . . . . . . . . . . . . . . . Exercises . . . . . . . . . . . . . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

31 32 34 35 39 39 41 43 44 47 47 48 50 53 53 54 54 55 55 57 58 59 60 60 62 63 66 67 71 71 73 75 75 76 77 77 79 80 85 85 86 87 87 89 90

Functions 4.1 Functions . . . . . . . . . . . . . 4.2 Functions can call other functions 4.3 Flow of execution . . . . . . . . 4.4 Functions that return values . . . 4.5 Variables and parameters are local 4.6 Turtles Revisited . . . . . . . . . 4.7 Glossary . . . . . . . . . . . . . 4.8 Exercises . . . . . . . . . . . . . Conditionals 5.1 Boolean values and expressions . 5.2 Logical operators . . . . . . . . . 5.3 Truth Tables . . . . . . . . . . . 5.4 Simplifying Boolean Expressions 5.5 Conditional execution . . . . . . 5.6 Omitting the else clause . . . . . 5.7 Chained conditionals . . . . . . . 5.8 Nested conditionals . . . . . . . 5.9 The return statement . . . . . 5.10 Logical opposites . . . . . . . . . 5.11 Type conversion . . . . . . . . . 5.12 A Turtle Bar Chart . . . . . . . . 5.13 Glossary . . . . . . . . . . . . . 5.14 Exercises . . . . . . . . . . . . . Fruitful functions 6.1 Return values . . . . . . 6.2 Program development . 6.3 Debugging with print 6.4 Composition . . . . . . 6.5 Boolean functions . . . 6.6 Programming with style 6.7 Unit testing . . . . . . . 6.8 Glossary . . . . . . . . 6.9 Exercises . . . . . . . . Iteration 7.1 Reassignment . . . . . 7.2 Updating variables . . 7.3 The for loop revisited 7.4 The while statement 7.5 The 3n + 1 sequence . 7.6 Tracing a program . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

ii

7.7 7.8 7.9 7.10 7.11 7.12 7.13 7.14 7.15 7.16 7.17 7.18 7.19 7.20 7.21 7.22 7.23 7.24 7.25 7.26 8

Counting digits . . . . . . . . . . . . . . Abbreviated assignment . . . . . . . . . Help and meta-notation . . . . . . . . . Tables . . . . . . . . . . . . . . . . . . . Two-dimensional tables . . . . . . . . . Encapsulation and generalization . . . . More encapsulation . . . . . . . . . . . Local variables . . . . . . . . . . . . . . The break statement . . . . . . . . . . Other avours of loops . . . . . . . . . . An example . . . . . . . . . . . . . . . . The continue statement . . . . . . . . More generalization . . . . . . . . . . . Functions . . . . . . . . . . . . . . . . . Paired Data . . . . . . . . . . . . . . . . Nested Loops for Nested Data . . . . . . Newtons method for nding square roots Algorithms . . . . . . . . . . . . . . . . Glossary . . . . . . . . . . . . . . . . . Exercises . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . .

92 92 93 95 96 96 97 97 98 99 101 102 102 104 104 105 106 107 108 109 113 113 113 115 115 116 117 118 118 119 120 120 121 122 123 123 124 127 128 128 131 131 132 133 133 134

Strings 8.1 A compound data type . . . . . . . . 8.2 Working with strings as single things 8.3 Working with the parts of a string . . 8.4 Length . . . . . . . . . . . . . . . . 8.5 Traversal and the for loop . . . . . 8.6 Slices . . . . . . . . . . . . . . . . . 8.7 String comparison . . . . . . . . . . 8.8 Strings are immutable . . . . . . . . 8.9 The in and not in operators . . . 8.10 A find function . . . . . . . . . . . 8.11 Looping and counting . . . . . . . . 8.12 Optional parameters . . . . . . . . . 8.13 The built-in find method . . . . . . 8.14 The split method . . . . . . . . . 8.15 Cleaning up your strings . . . . . . . 8.16 The format method for strings . . . . 8.17 Summary . . . . . . . . . . . . . . . 8.18 Glossary . . . . . . . . . . . . . . . 8.19 Exercises . . . . . . . . . . . . . . . Tuples 9.1 Tuples are used for grouping data 9.2 Tuple assignment . . . . . . . . . 9.3 Tuples as return values . . . . . . 9.4 Composability of Data Structures 9.5 Glossary . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . .

. . . . .

. . . . .

. . . . .

. . . . .

. . . . .

. . . . .

. . . . .

. . . . .

. . . . .

. . . . .

. . . . .

. . . . .

. . . . .

. . . . .

. . . . .

. . . . .

. . . . .

. . . . .

. . . . .

. . . . .

. . . . .

. . . . .

. . . . .

. . . . .

iii

9.6

Exercises . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 134 135 135 136 138 139 141 141 143 143 144 144 145 146 146 146 147 148 149 149 150 151 152 153 154 154 155 156 157 157 158 163 163 166 167 168 168 170 171 172 172 173 174

10 Event-Driven Programming 10.1 Keypress events . . . . . . . 10.2 Mouse events . . . . . . . . . 10.3 Automatic events from a timer 10.4 An example: state machines . 10.5 Glossary . . . . . . . . . . . 10.6 Exercises . . . . . . . . . . . 11 Lists 11.1 11.2 11.3 11.4 11.5 11.6 11.7 11.8 11.9 11.10 11.11 11.12 11.13 11.14 11.15 11.16 11.17 11.18 11.19 11.20 11.21 11.22

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

List values . . . . . . . . . Accessing elements . . . . List length . . . . . . . . . List membership . . . . . . List operations . . . . . . . List slices . . . . . . . . . . Lists are mutable . . . . . . List deletion . . . . . . . . Objects and references . . . Aliasing . . . . . . . . . . . Cloning lists . . . . . . . . Lists and for loops . . . . List parameters . . . . . . . List methods . . . . . . . . Pure functions and modiers Functions that produce lists Strings and lists . . . . . . . list and range . . . . . Nested lists . . . . . . . . . Matrices . . . . . . . . . . Glossary . . . . . . . . . . Exercises . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . . . . . . . .

12 Modules 12.1 Random numbers . . . . . . . . . . 12.2 The time module . . . . . . . . . 12.3 The math module . . . . . . . . . 12.4 Creating your own modules . . . . 12.5 Namespaces . . . . . . . . . . . . 12.6 Scope and lookup rules . . . . . . . 12.7 Attributes and the dot operator . . . 12.8 Three import statement variants . 12.9 Turn your unit tester into a module 12.10 Glossary . . . . . . . . . . . . . . 12.11 Exercises . . . . . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

13 Files 179 13.1 About les . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 179 iv

13.2 13.3 13.4 13.5 13.6 13.7 13.8 13.9 13.10 13.11

Writing our rst le . . . . . . . . . . . . . . Reading a le line-at-a-time . . . . . . . . . . Turning a le into a list of lines . . . . . . . . Reading the whole le at once . . . . . . . . . Working with binary les . . . . . . . . . . . An example . . . . . . . . . . . . . . . . . . . Directories . . . . . . . . . . . . . . . . . . . What about fetching something from the web? Glossary . . . . . . . . . . . . . . . . . . . . Exercises . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

179 180 181 182 182 183 183 184 185 186 187 187 187 189 191 195 196 197 199 203 204 205 209 209 209 211 212 213 214 214 215 216 217 217 218 221 221 222 223 224 225 225

14 List Algorithms 14.1 Test-driven development . . . . . . . . . 14.2 The linear search algorithm . . . . . . . 14.3 A more realistic problem . . . . . . . . . 14.4 Binary Search . . . . . . . . . . . . . . 14.5 Removing adjacent duplicates from a list 14.6 Merging sorted lists . . . . . . . . . . . 14.7 Alice in Wonderland, again! . . . . . . . 14.8 Eight queens puzzle, part 1 . . . . . . . . 14.9 Eight queens puzzle, part 2 . . . . . . . . 14.10 Glossary . . . . . . . . . . . . . . . . . 14.11 Exercises . . . . . . . . . . . . . . . . . 15 Classes and Objects - the Basics 15.1 Object-oriented programming . . . . . 15.2 User-dened compound data types . . . 15.3 Attributes . . . . . . . . . . . . . . . . 15.4 Improving our initializer . . . . . . . . 15.5 Adding other methods to our class . . . 15.6 Instances as arguments and parameters 15.7 Converting an instance to a string . . . 15.8 Instances as return values . . . . . . . 15.9 A change of perspective . . . . . . . . 15.10 Objects can have state . . . . . . . . . 15.11 Glossary . . . . . . . . . . . . . . . . 15.12 Exercises . . . . . . . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

16 Classes and Objects - Digging a little deeper 16.1 Rectangles . . . . . . . . . . . . . . . . 16.2 Objects are mutable . . . . . . . . . . . 16.3 Sameness . . . . . . . . . . . . . . . . . 16.4 Copying . . . . . . . . . . . . . . . . . 16.5 Glossary . . . . . . . . . . . . . . . . . 16.6 Exercises . . . . . . . . . . . . . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

17 PyGame 227 17.1 The game loop . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 227 17.2 Displaying images and text . . . . . . . . . . . . . . . . . . . . . . . . . . . 230 v

17.3 17.4 17.5 17.6 17.7 17.8 17.9 17.10

Drawing a board for the N queens puzzle Sprites . . . . . . . . . . . . . . . . . . Events . . . . . . . . . . . . . . . . . . A wave of animation . . . . . . . . . . . Aliens - a case study . . . . . . . . . . . Reections . . . . . . . . . . . . . . . . Glossary . . . . . . . . . . . . . . . . . Exercises . . . . . . . . . . . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

232 236 239 241 245 246 246 247 249 249 252 252 254 255 256 256 261 261 263 264 264 265 265 267 268 268 270 270 271 273 273 274 277 277 277 279 279 280 281 282 283 285 287 287

18 Recursion 18.1 Drawing Fractals . . . . . . . . . . . . . . 18.2 Recursive data structures . . . . . . . . . . 18.3 Recursion . . . . . . . . . . . . . . . . . . 18.4 Case study: Fibonacci numbers . . . . . . 18.5 Example with recursive directories and les 18.6 Glossary . . . . . . . . . . . . . . . . . . 18.7 Exercises . . . . . . . . . . . . . . . . . . 19 Exceptions 19.1 Catching exceptions . . . . . . . . . . . . 19.2 Raising your own exceptions . . . . . . . . 19.3 Revisiting an earlier example . . . . . . . 19.4 The finally clause of the try statement 19.5 Glossary . . . . . . . . . . . . . . . . . . 19.6 Exercises . . . . . . . . . . . . . . . . . . 20 Dictionaries 20.1 Dictionary operations 20.2 Dictionary methods . 20.3 Aliasing and copying . 20.4 Sparse matrices . . . . 20.5 Memos . . . . . . . . 20.6 Counting letters . . . 20.7 Glossary . . . . . . . 20.8 Exercises . . . . . . .

. . . . . . .

. . . . . . .

. . . . . . .

. . . . . . .

. . . . . . .

. . . . . . .

. . . . . . .

. . . . . . .

. . . . . . .

. . . . . . .

. . . . . . .

. . . . . . .

. . . . . . .

. . . . . . .

. . . . . . .

. . . . . . .

. . . . . . .

. . . . . . .

. . . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

21 Even more OOP 21.1 MyTime . . . . . . . . . . . . . . . 21.2 Pure functions . . . . . . . . . . . . 21.3 Modiers . . . . . . . . . . . . . . . 21.4 Converting increment to a method 21.5 An Aha! insight . . . . . . . . . . 21.6 Generalization . . . . . . . . . . . . 21.7 Another example . . . . . . . . . . . 21.8 Operator overloading . . . . . . . . . 21.9 Polymorphism . . . . . . . . . . . . 21.10 Glossary . . . . . . . . . . . . . . . 21.11 Exercises . . . . . . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

. . . . . . . . . . .

vi

22 Collections of objects 22.1 Composition . . . . . . . . . . . . . . . . 22.2 Card objects . . . . . . . . . . . . . . . . 22.3 Class attributes and the __str__ method 22.4 Comparing cards . . . . . . . . . . . . . . 22.5 Decks . . . . . . . . . . . . . . . . . . . . 22.6 Printing the deck . . . . . . . . . . . . . . 22.7 Shufing the deck . . . . . . . . . . . . . 22.8 Removing and dealing cards . . . . . . . . 22.9 Glossary . . . . . . . . . . . . . . . . . . 22.10 Exercises . . . . . . . . . . . . . . . . . . 23 Inheritance 23.1 Inheritance . . . . . . 23.2 A hand of cards . . . . 23.3 Dealing cards . . . . . 23.4 Printing a Hand . . . . 23.5 The CardGame class 23.6 OldMaidHand class 23.7 OldMaidGame class 23.8 Glossary . . . . . . . 23.9 Exercises . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

289 289 289 290 291 293 293 295 296 297 297 299 299 299 300 301 302 303 304 307 308 309 309 309 310 311 312 312 313 314 315 316 316 317 319 319 319 320 321 321 321 322 323 323 324

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

24 Linked lists 24.1 Embedded references . . . . . . . . . 24.2 The Node class . . . . . . . . . . . . 24.3 Lists as collections . . . . . . . . . . 24.4 Lists and recursion . . . . . . . . . . 24.5 Innite lists . . . . . . . . . . . . . . 24.6 The fundamental ambiguity theorem . 24.7 Modifying lists . . . . . . . . . . . . 24.8 Wrappers and helpers . . . . . . . . 24.9 The LinkedList class . . . . . . . 24.10 Invariants . . . . . . . . . . . . . . . 24.11 Glossary . . . . . . . . . . . . . . . 24.12 Exercises . . . . . . . . . . . . . . . 25 Stacks 25.1 Abstract data types . . . . . . . . . . 25.2 The Stack ADT . . . . . . . . . . . . 25.3 Implementing stacks with Python lists 25.4 Pushing and popping . . . . . . . . . 25.5 Using a stack to evaluate postx . . . 25.6 Parsing . . . . . . . . . . . . . . . . 25.7 Evaluating postx . . . . . . . . . . 25.8 Clients and providers . . . . . . . . . 25.9 Glossary . . . . . . . . . . . . . . . 25.10 Exercises . . . . . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

. . . . . . . . . .

vii

26 Queues 26.1 The Queue ADT . . . . . . 26.2 Linked Queue . . . . . . . 26.3 Performance characteristics 26.4 Improved Linked Queue . . 26.5 Priority queue . . . . . . . 26.6 The Golfer class . . . . . 26.7 Glossary . . . . . . . . . . 26.8 Exercises . . . . . . . . . . 27 Trees 27.1 27.2 27.3 27.4 27.5 27.6 27.7 27.8 27.9

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

. . . . . . . .

325 325 325 326 327 328 329 330 331 333 333 334 334 335 336 339 340 341 342 343 343 344 344 345 345 345 346 346 347 347 348 348 349 350 350 350

Building trees . . . . . . . . Traversing trees . . . . . . Expression trees . . . . . . Tree traversal . . . . . . . . Building an expression tree Handling errors . . . . . . . The animal tree . . . . . . . Glossary . . . . . . . . . . Exercises . . . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

. . . . . . . . .

A Debugging A.1 Syntax errors . . . . . . . . . . . . . . . . . . . . . . . . . . . A.2 I cant get my program to run no matter what I do. . . . . . . . A.3 Runtime errors . . . . . . . . . . . . . . . . . . . . . . . . . . A.4 My program does absolutely nothing. . . . . . . . . . . . . . . A.5 My program hangs. . . . . . . . . . . . . . . . . . . . . . . . . A.6 Innite Loop . . . . . . . . . . . . . . . . . . . . . . . . . . . A.7 Innite Recursion . . . . . . . . . . . . . . . . . . . . . . . . A.8 Flow of Execution . . . . . . . . . . . . . . . . . . . . . . . . A.9 When I run the program I get an exception. . . . . . . . . . . . A.10 I added so many print statements I get inundated with output. A.11 Semantic errors . . . . . . . . . . . . . . . . . . . . . . . . . . A.12 My program doesnt work. . . . . . . . . . . . . . . . . . . . . A.13 Ive got a big hairy expression and it doesnt do what I expect. . A.14 Ive got a function or method that doesnt return what I expect. . A.15 Im really, really stuck and I need help. . . . . . . . . . . . . . A.16 No, I really need help. . . . . . . . . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . .

. . . . . . . . . . . . . . . .

B An odds-and-ends Workbook 353 B.1 The Five Strands of Prociency . . . . . . . . . . . . . . . . . . . . . . . . . 353 B.2 Sending Email . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 354 C Conguring Ubuntu for Python Development C.1 Vim . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . C.2 GASP . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . C.3 $HOME environment . . . . . . . . . . . . . . . . . . . . . . C.4 Making a python script executable and runnable from anywhere 357 357 358 358 359

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

. . . .

viii

D Customizing and Contributing to the Book 361 D.1 Getting the Source . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 361 D.2 Making the HTML Version . . . . . . . . . . . . . . . . . . . . . . . . . . . 362 E Some Tips, Tricks, and Common Errors 363 E.1 Functions . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 363 E.2 String handling . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 367 E.3 Looping and lists . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 368 F GNU Free Documentation License F.1 0. PREAMBLE . . . . . . . . . . . . . . . . . . . . . . . . F.2 1. APPLICABILITY AND DEFINITIONS . . . . . . . . . F.3 2. VERBATIM COPYING . . . . . . . . . . . . . . . . . . F.4 3. COPYING IN QUANTITY . . . . . . . . . . . . . . . . F.5 4. MODIFICATIONS . . . . . . . . . . . . . . . . . . . . F.6 5. COMBINING DOCUMENTS . . . . . . . . . . . . . . F.7 6. COLLECTIONS OF DOCUMENTS . . . . . . . . . . . F.8 7. AGGREGATION WITH INDEPENDENT WORKS . . . F.9 8. TRANSLATION . . . . . . . . . . . . . . . . . . . . . F.10 9. TERMINATION . . . . . . . . . . . . . . . . . . . . . . F.11 10. FUTURE REVISIONS OF THIS LICENSE . . . . . . F.12 11. RELICENSING . . . . . . . . . . . . . . . . . . . . . F.13 ADDENDUM: How to use this License for your documents Index 371 371 371 373 373 374 375 376 376 376 377 377 378 378 381

. . . . . . . . . . . . .

. . . . . . . . . . . . .

. . . . . . . . . . . . .

. . . . . . . . . . . . .

. . . . . . . . . . . . .

. . . . . . . . . . . . .

. . . . . . . . . . . . .

. . . . . . . . . . . . .

. . . . . . . . . . . . .

. . . . . . . . . . . . .

ix

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Version date: February 2012 by Peter Wentworth, Jeffrey Elkner, Allen B. Downey, and Chris Meyers (based on 2nd edition by Jeffrey Elkner, Allen B. Downey, and Chris Meyers) Corresponding author: p.wentworth@ru.ac.za Source respository is team/thinkcspy/thinkcspy3-rle at at https://code.launchpad.net/~thinkcspy-rle-

For ofine use, download a zip le of the html or a pdf version (updated less often) from http://www.ict.ru.ac.za/Resources/cspw/thinkcspy3/ search Copyright Notice Foreword Preface Preface-3 This Rhodes Local Edition (RLE) of the book Contributor List Chapter 1 The way of the program Chapter 2 Variables, expressions, and statements Chapter 3 Hello, little turtles! Chapter 4 Functions Chapter 5 Conditionals Chapter 6 Fruitful functions Chapter 7 Iteration Chapter 8 Strings Chapter 9 Tuples Chapter 10 Event handling Chapter 11 Lists Chapter 12 Modules

CONTENTS

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Chapter 13 Files Chapter 14 List Algorithms Chapter 15 Classes and Objects - the Basics Chapter 16 Classes and Objects - Digging a little deeper Chapter 17 PyGame Chapter 18 Recursion Chapter 19 Exceptions Chapter 20 Dictionaries Chapter 21 Even more OOP Chapter 22 Collections of Objects Chapter 23 Inheritance Chapter 24 Linked Lists Chapter 25 Stacks Chapter 26 Queues Chapter 27 Trees Appendix A Debugging Appendix B An odds-and-ends Workbook Appendix C Conguring Ubuntu for Python Development Appendix D Customizing and Contributing to the Book Appendix E Some Tips, Tricks, and Common Errors GNU Free Document License

Copyright Notice
Copyright (C) Peter Wentworth, Jeffrey Elkner, Allen B. Downey and Chris Meyers. Permission is granted to copy, distribute and/or modify this document under the terms of the GNU Free Documentation License, Version 1.3 or any later version published by the Free Software Foundation; with Invariant Sections being Foreword, Preface, and Contributor List, no Front-Cover Texts, and no Back-Cover Texts. A copy of the license is included in the section entitled GNU Free Documentation License.

CONTENTS

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

ii

CONTENTS

FOREWORD

By David Beazley As an educator, researcher, and book author, I am delighted to see the completion of this book. Python is a fun and extremely easy-to-use programming language that has steadily gained in popularity over the last few years. Developed over ten years ago by Guido van Rossum, Pythons simple syntax and overall feel is largely derived from ABC, a teaching language that was developed in the 1980s. However, Python was also created to solve real problems and it borrows a wide variety of features from programming languages such as C++, Java, Modula-3, and Scheme. Because of this, one of Pythons most remarkable features is its broad appeal to professional software developers, scientists, researchers, artists, and educators. Despite Pythons appeal to many different communities, you may still wonder why Python? or why teach programming with Python? Answering these questions is no simple taskespecially when popular opinion is on the side of more masochistic alternatives such as C++ and Java. However, I think the most direct answer is that programming in Python is simply a lot of fun and more productive. When I teach computer science courses, I want to cover important concepts in addition to making the material interesting and engaging to students. Unfortunately, there is a tendency for introductory programming courses to focus far too much attention on mathematical abstraction and for students to become frustrated with annoying problems related to low-level details of syntax, compilation, and the enforcement of seemingly arcane rules. Although such abstraction and formalism is important to professional software engineers and students who plan to continue their study of computer science, taking such an approach in an introductory course mostly succeeds in making computer science boring. When I teach a course, I dont want to have a room of uninspired students. I would much rather see them trying to solve interesting problems by exploring different ideas, taking unconventional approaches, breaking the rules, and learning from their mistakes. In doing so, I dont want to waste half of the semester trying to sort out obscure syntax problems, unintelligible compiler error messages, or the several hundred ways that a program might generate a general protection fault. One of the reasons why I like Python is that it provides a really nice balance between the practical and the conceptual. Since Python is interpreted, beginners can pick up the language and start doing neat things almost immediately without getting lost in the problems of compilation and linking. Furthermore, Python comes with a large library of modules that can be used to do all sorts of tasks ranging from web-programming to graphics. Having such a practical focus is a great way to engage students and it allows them to complete signicant projects. iii

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

However, Python can also serve as an excellent foundation for introducing important computer science concepts. Since Python fully supports procedures and classes, students can be gradually introduced to topics such as procedural abstraction, data structures, and object-oriented programming all of which are applicable to later courses on Java or C++. Python even borrows a number of features from functional programming languages and can be used to introduce concepts that would be covered in more detail in courses on Scheme and Lisp. In reading Jeffreys preface, I am struck by his comments that Python allowed him to see a higher level of success and a lower level of frustration and that he was able to move faster with better results. Although these comments refer to his introductory course, I sometimes use Python for these exact same reasons in advanced graduate level computer science courses at the University of Chicago. In these courses, I am constantly faced with the daunting task of covering a lot of difcult course material in a blistering nine week quarter. Although it is certainly possible for me to inict a lot of pain and suffering by using a language like C++, I have often found this approach to be counterproductiveespecially when the course is about a topic unrelated to just programming. I nd that using Python allows me to better focus on the actual topic at hand while allowing students to complete substantial class projects. Although Python is still a young and evolving language, I believe that it has a bright future in education. This book is an important step in that direction. David Beazley University of Chicago Author of the Python Essential Reference

iv

CONTENTS

PREFACE

By Jeffrey Elkner This book owes its existence to the collaboration made possible by the Internet and the free software movement. Its three authorsa college professor, a high school teacher, and a professional programmernever met face to face to work on it, but we have been able to collaborate closely, aided by many other folks who have taken the time and energy to send us their feedback. We think this book is a testament to the benets and future possibilities of this kind of collaboration, the framework for which has been put in place by Richard Stallman and the Free Software Foundation.

0.0.1 How and why I came to use Python


In 1999, the College Boards Advanced Placement (AP) Computer Science exam was given in C++ for the rst time. As in many high schools throughout the country, the decision to change languages had a direct impact on the computer science curriculum at Yorktown High School in Arlington, Virginia, where I teach. Up to this point, Pascal was the language of instruction in both our rst-year and AP courses. In keeping with past practice of giving students two years of exposure to the same language, we made the decision to switch to C++ in the rst year course for the 1997-98 school year so that we would be in step with the College Boards change for the AP course the following year. Two years later, I was convinced that C++ was a poor choice to use for introducing students to computer science. While it is certainly a very powerful programming language, it is also an extremely difcult language to learn and teach. I found myself constantly ghting with C++s difcult syntax and multiple ways of doing things, and I was losing many students unnecessarily as a result. Convinced there had to be a better language choice for our rst-year class, I went looking for an alternative to C++. I needed a language that would run on the machines in our GNU/Linux lab as well as on the Windows and Macintosh platforms most students have at home. I wanted it to be free software, so that students could use it at home regardless of their income. I wanted a language that was used by professional programmers, and one that had an active developer community around it. It had to support both procedural and object-oriented programming. And most importantly, it

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

had to be easy to learn and teach. When I investigated the choices with these goals in mind, Python stood out as the best candidate for the job. I asked one of Yorktowns talented students, Matt Ahrens, to give Python a try. In two months he not only learned the language but wrote an application called pyTicket that enabled our staff to report technology problems via the Web. I knew that Matt could not have nished an application of that scale in so short a time in C++, and this accomplishment, combined with Matts positive assessment of Python, suggested that Python was the solution I was looking for.

0.0.2 Finding a textbook


Having decided to use Python in both of my introductory computer science classes the following year, the most pressing problem was the lack of an available textbook. Free documents came to the rescue. Earlier in the year, Richard Stallman had introduced me to Allen Downey. Both of us had written to Richard expressing an interest in developing free educational materials. Allen had already written a rst-year computer science textbook, How to Think Like a Computer Scientist. When I read this book, I knew immediately that I wanted to use it in my class. It was the clearest and most helpful computer science text I had seen. It emphasized the processes of thought involved in programming rather than the features of a particular language. Reading it immediately made me a better teacher. How to Think Like a Computer Scientist was not just an excellent book, but it had been released under the GNU public license, which meant it could be used freely and modied to meet the needs of its user. Once I decided to use Python, it occurred to me that I could translate Allens original Java version of the book into the new language. While I would not have been able to write a textbook on my own, having Allens book to work from made it possible for me to do so, at the same time demonstrating that the cooperative development model used so well in software could also work for educational materials. Working on this book for the last two years has been rewarding for both my students and me, and my students played a big part in the process. Since I could make instant changes whenever someone found a spelling error or difcult passage, I encouraged them to look for mistakes in the book by giving them a bonus point each time they made a suggestion that resulted in a change in the text. This had the double benet of encouraging them to read the text more carefully and of getting the text thoroughly reviewed by its most important critics, students using it to learn computer science. For the second half of the book on object-oriented programming, I knew that someone with more real programming experience than I had would be needed to do it right. The book sat in an unnished state for the better part of a year until the open source community once again provided the needed means for its completion. I received an email from Chris Meyers expressing interest in the book. Chris is a professional programmer who started teaching a programming course last year using Python at Lane Community College in Eugene, Oregon. The prospect of teaching the course had led Chris to the book, and he started helping out with it immediately. By the end of the school year he had created a companion project on our Website at http://openbookproject.net called *Python for Fun* and was working with some of my most advanced students as a master teacher, guiding them beyond where I could take them. vi CONTENTS

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

0.0.3 Introducing programming with Python


The process of translating and using How to Think Like a Computer Scientist for the past two years has conrmed Pythons suitability for teaching beginning students. Python greatly simplies programming examples and makes important programming ideas easier to teach. The rst example from the text illustrates this point. It is the traditional hello, world program, which in the Java version of the book looks like this:
class Hello { public static void main (String[] args) { System.out.println ("Hello, world."); } }

in the Python version it becomes:


print("Hello, World!")

Even though this is a trivial example, the advantages of Python stand out. Yorktowns Computer Science I course has no prerequisites, so many of the students seeing this example are looking at their rst program. Some of them are undoubtedly a little nervous, having heard that computer programming is difcult to learn. The Java version has always forced me to choose between two unsatisfying options: either to explain the class Hello, public static void main, String[] args, {, and }, statements and risk confusing or intimidating some of the students right at the start, or to tell them, Just dont worry about all of that stuff now; we will talk about it later, and risk the same thing. The educational objectives at this point in the course are to introduce students to the idea of a programming statement and to get them to write their rst program, thereby introducing them to the programming environment. The Python program has exactly what is needed to do these things, and nothing more. Comparing the explanatory text of the program in each version of the book further illustrates what this means to the beginning student. There are seven paragraphs of explanation of Hello, world! in the Java version; in the Python version, there are only a few sentences. More importantly, the missing six paragraphs do not deal with the big ideas in computer programming but with the minutia of Java syntax. I found this same thing happening throughout the book. Whole paragraphs simply disappear from the Python version of the text because Pythons much clearer syntax renders them unnecessary. Using a very high-level language like Python allows a teacher to postpone talking about lowlevel details of the machine until students have the background that they need to better make sense of the details. It thus creates the ability to put rst things rst pedagogically. One of the best examples of this is the way in which Python handles variables. In Java a variable is a name for a place that holds a value if it is a built-in type, and a reference to an object if it is not. Explaining this distinction requires a discussion of how the computer stores data. Thus, the idea of a variable is bound up with the hardware of the machine. The powerful and fundamental concept of a variable is already difcult enough for beginning students (in both computer science and algebra). Bytes and addresses do not help the matter. In Python a variable is a name that refers to a thing. This is a far more intuitive concept for beginning students and is much closer to the meaning of variable that they learned in their math courses. I had much CONTENTS vii

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

less difculty teaching variables this year than I did in the past, and I spent less time helping students with problems using them. Another example of how Python aids in the teaching and learning of programming is in its syntax for functions. My students have always had a great deal of difculty understanding functions. The main problem centers around the difference between a function denition and a function call, and the related distinction between a parameter and an argument. Python comes to the rescue with syntax that is nothing short of beautiful. Function denitions begin with the keyword def, so I simply tell my students, When you dene a function, begin with def, followed by the name of the function that you are dening; when you call a function, simply call (type) out its name. Parameters go with denitions; arguments go with calls. There are no return types, parameter types, or reference and value parameters to get in the way, so I am now able to teach functions in less than half the time that it previously took me, with better comprehension. Using Python improved the effectiveness of our computer science program for all students. I saw a higher general level of success and a lower level of frustration than I experienced teaching with either C++ or Java. I moved faster with better results. More students left the course with the ability to create meaningful programs and with the positive attitude toward the experience of programming that this engenders.

0.0.4 Building a community


I have received email from all over the globe from people using this book to learn or to teach programming. A user community has begun to emerge, and many people have been contributing to the project by sending in materials for the companion Website at http://openbookproject.net/pybiblio. With the continued growth of Python, I expect the growth in the user community to continue and accelerate. The emergence of this user community and the possibility it suggests for similar collaboration among educators have been the most exciting parts of working on this project for me. By working together, we can increase the quality of materials available for our use and save valuable time. I invite you to join our community and look forward to hearing from you. Please write to me at jeff@elkner.net.

Jeffrey Elkner Governors Career and Technical Academy in Arlington Arlington, Virginia

viii

CONTENTS

THE RHODES LOCAL EDITION (RLE) (VERSION OF OCTOBER, 2011)

By Peter Wentworth A word of thanks ... We switched from Java to Python in our introductory courses in 2010. So far we think the results look positive. More time will tell. This predecessor to this book was a great starting point for us, especially because of the liberal permission to change things. Having our own in-house course notes or handouts allows us to adapt and stay fresh, rearrange, see what works, and it gives us agility. We can also ensure that every student in the course gets a copy of the handouts something that doesnt always happen if we prescribe costly textbooks. Many thanks to all the contributors and the authors for making their hard work available to the Python community and to our students. A colleague and friend, Peter Warren, once made the remark that learning introductory programming is as much about the environment as it is about the programming language. Im a big fan of IDEs (Integrated Development Environments). I want help to be integrated into my editor, as a rst-class citizen, available at the press of a button. I want syntax highlighting. I want immediate syntax checking, and sensible autocompletion. Im especially keen on having a single-stepping debugger and breakpoints with code inspection built in. Were trying to build a conceptual model of program execution in the students mind, so I nd most helpful for teaching to have the call stack and variables explicitly visible, and to be able to immediately inspect the result of executing a statement. My philosophy, then, is not to look for a language to teach, but to look for a combination of IDE and language that are packaged together, and evaluated as a whole. Ive made some quite deep changes to the original book to reect this (and various other opinionated views that I hold), and I have no doubt that more changes will follow as we bed down our course. Here are some of the key things Ive approached differently: ix

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Our local situation demands that we have a large number of service course students in an introductory course of just 3 or 4 weeks, and then we get another semester of teaching with those going into our mainstream program. So the book is in two parts: well do the rst ve chapters in the big get your toes wet course, and the rest of the material in a separate semester. Were using Python 3. It is cleaner, more object oriented, and has fewer ad-hoc irregularities than earlier versions of Python. Were using PyScripter as our IDE, on Windows. And it is hardwired into parts of these notes, with screenshots, etc. Ive dropped GASP. For graphics we start with the Turtle module. As things move along, we use PyGame for more advanced graphics. Ive introduced some event-driven programming using the turtle. I have tried to push more object-oriented notions earlier, without asking students to synthesize objects or write their own classes. So, for example, in the chapter about the turtle, we create multiple instances of turtles, talk about their attributes and state (color, position, etc), and favour method-call style to move them around, i.e. tess.forward(100). Similarly, when we use random numbers, we avoid the hidden singleton generator in the random module we rather create an instance of a generator, and invoke methods on the instance. The ease of constructing lists and the for loop seem to be winners in Python, so rather than use the traditional command-line input for data, Ive favoured using loops and lists right up front, like this:
1 2 3 4

friends = ["Amy", "Joe", "Bill"] for f in friends: invitation = "Hi " + f + ". Please come to my party on Saturday!" print(invitation)

This also means that I bumped range up for early exposure. I envisage that over time well see more opportunities to exploit early lists, early iteration in its most simple form. I dumped doctest : it is a bit too quirky for my liking. For example, it fails a test if the spacing between list elements is not precisely the same as the output string, or if Python prints a string with single quotes, but you wrote up the test case with double quotes. Cases like this also confused students (and instructors) quite badly:
1 2 3 4 5 6 7

def addlist(xs): """ >>> xs = [2,3,4] >>> addlist(xs) 9 """ return

If you can explain the difference in scope rules and lifetimes between the parameter xs x CONTENTS

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

and the doctest variable xs elegantly, please let me know. Yes, I know doctest creates its own scope behind our back, but it is exactly this black magic that were trying to avoid. From the usual indentation rules, also looks like the doctests are nested inside the function scope, but they are not. Students thought that the parameter had been given its value by the assignment to xs in the doctest! I also think that keeping the test suite separate from the functions under test leads to a cleaner relationship between caller and callee, and gives a better chance of getting argument passing / parameter concepts taught accurately. There is a good unit testing module in Python, (and PyScripter offers integrated support for it, and automated generation of skeleton test modules), but it looked too advanced for beginners, because it requires multi-module concepts. So Ive favoured my own test scaffolding in Chapter 6 (about 10 lines of code) that the students must insert into whatever le theyre working on. Ive played down command-line input / process / output where possible. Many of our students have never seen a command-line shell, and it is arguably quite intimidating. Weve gone back to a more classic / static approach to writing our own classes and objects. Python (in company with languages like Javascript, Ruby, Perl, PHP, etc.) dont really emphasize notions of sealed classes or private members, or even sealed instances. So one teaching approach is to allocate each instance as an empty container, and subsequently allow the external clients of the class to poke new members (methods or attributes) into different instances as they wish to. It is a very dynamic approach, but perhaps not one that encourages thinking in abstractions, layers, contracts, decoupling, etc. It might even be the kind of thing that one could write one of those x,y,z ... considered harmful papers about. In our more conservative approach, we put an initializer into every class, we determine at object instantiation time what members we want, and we initialize the instances from within the class. So weve moved closer in philosophy to C# / Java on this one. Were moving towards introducing more algorithms earlier into the course. Python is an efcient teaching language we can make fast progress. But the gains we make there wed like to invest into deeper problem solving, and more complex algorithms with the basics, rather than cover more Python features. Some of these changes have started to nd their way in this version, and Im sure well see more in future. Were interested in issues around teaching and learning. Some research indicates that intellectual playfulness is important. The study referenced in the Odds-and-ends workbook at the end just didnt seem to have anywhere sensible to go in the book, yet I wanted it included. It is quite likely that well allow more issues like this to creep into the book, to try to make it more than just about programming in Python.

CONTENTS

xi

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

xii

CONTENTS

CONTRIBUTOR LIST

To paraphrase the philosophy of the Free Software Foundation, this book is free like free speech, but not necessarily free like free pizza. It came about because of a collaboration that would not have been possible without the GNU Free Documentation License. So we would like to thank the Free Software Foundation for developing this license and, of course, making it available to us. We would also like to thank the more than 100 sharp-eyed and thoughtful readers who have sent us suggestions and corrections over the past few years. In the spirit of free software, we decided to express our gratitude in the form of a contributor list. Unfortunately, this list is not complete, but we are doing our best to keep it up to date. It was also getting too large to include everyone who sends in a typo or two. You have our gratitude, and you have the personal satisfaction of making a book you found useful better for you and everyone else who uses it. New additions to the list for the 2nd edition will be those who have made on-going contributions. If you have a chance to look through the list, you should realize that each person here has spared you and all subsequent readers from the confusion of a technical error or a less-than-transparent explanation, just by sending us a note. Impossible as it may seem after so many corrections, there may still be errors in this book. If you should stumble across one, we hope you will take a minute to contact us. The email address is jeff@elkner.net . Substantial changes made due to your suggestions will add you to the next version of the contributor list (unless you ask to be omitted). Thank you!

0.0.5 Second Edition


An email from Mike MacHenry set me straight on tail recursion. He not only pointed out an error in the presentation, but suggested how to correct it. It wasnt until 5th Grade student Owen Davies came to me in a Saturday morning Python enrichment class and said he wanted to write the card game, Gin Rummy, in Python that I nally knew what I wanted to use as the case study for the object oriented programming chapters. A special thanks to pioneering students in Jeffs Python Programming class at GCTAA during the 2009-2010 school year: Safath Ahmed, Howard Batiste, Louis Elkner-Alfaro, and Rachel Hancock. Your continual and thoughtfull feedback led to changes in most of the chapters of the book. You set the standard for the active and engaged learners that xiii

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

will help make the new Governors Academy what it is to become. Thanks to you this is truly a student tested text. Thanks in a similar vein to the students in Jeffs Computer Science class at the HBWoodlawn program during the 2007-2008 school year: James Crowley, Joshua Eddy, Eric Larson, Brian McGrail, and Iliana Vazuka. Ammar Nabulsi sent in numerous corrections from Chapters 1 and 2. Aldric Giacomoni pointed out an error in our denition of the Fibonacci sequence in Chapter 5. Roger Sperberg sent in several spelling corrections and pointed out a twisted piece of logic in Chapter 3. Adele Goldberg sat down with Jeff at PyCon 2007 and gave him a list of suggestions and corrections from throughout the book. Ben Bruno sent in corrections for chapters 4, 5, 6, and 7. Carl LaCombe pointed out that we incorrectly used the term commutative in chapter 6 where symmetric was the correct term. Alessandro Montanile sent in corrections for errors in the code examples and text in chapters 3, 12, 15, 17, 18, 19, and 20. Emanuele Rusconi found errors in chapters 4, 8, and 15. Michael Vogt reported an indentation error in an example in chapter 6, and sent in a suggestion for improving the clarity of the shell vs. script section in chapter 1.

0.0.6 First Edition


Lloyd Hugh Allen sent in a correction to Section 8.4. Yvon Boulianne sent in a correction of a semantic error in Chapter 5. Fred Bremmer submitted a correction in Section 2.1. Jonah Cohen wrote the Perl scripts to convert the LaTeX source for this book into beautiful HTML. Michael Conlon sent in a grammar correction in Chapter 2 and an improvement in style in Chapter 1, and he initiated discussion on the technical aspects of interpreters. Benoit Girard sent in a correction to a humorous mistake in Section 5.6. Courtney Gleason and Katherine Smith wrote horsebet.py, which was used as a case study in an earlier version of the book. Their program can now be found on the website. Lee Harr submitted more corrections than we have room to list here, and indeed he should be listed as one of the principal editors of the text. James Kaylin is a student using the text. He has submitted numerous corrections. David Kershaw xed the broken catTwice function in Section 3.10.

xiv

CONTENTS

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Eddie Lam has sent in numerous corrections to Chapters 1, 2, and 3. He also xed the Makele so that it creates an index the rst time it is run and helped us set up a versioning scheme. Man-Yong Lee sent in a correction to the example code in Section 2.4. David Mayo pointed out that the word unconsciously in Chapter 1 needed to be changed to subconsciously . Chris McAloon sent in several corrections to Sections 3.9 and 3.10. Matthew J. Moelter has been a long-time contributor who sent in numerous corrections and suggestions to the book. Simon Dicon Montford reported a missing function denition and several typos in Chapter 3. He also found errors in the increment function in Chapter 13. John Ouzts corrected the denition of return value in Chapter 3. Kevin Parks sent in valuable comments and suggestions as to how to improve the distribution of the book. David Pool sent in a typo in the glossary of Chapter 1, as well as kind words of encouragement. Michael Schmitt sent in a correction to the chapter on les and exceptions. Robin Shaw pointed out an error in Section 13.1, where the printTime function was used in an example without being dened. Paul Sleigh found an error in Chapter 7 and a bug in Jonah Cohens Perl script that generates HTML from LaTeX. Craig T. Snydal is testing the text in a course at Drew University. He has contributed several valuable suggestions and corrections. Ian Thomas and his students are using the text in a programming course. They are the rst ones to test the chapters in the latter half of the book, and they have make numerous corrections and suggestions. Keith Verheyden sent in a correction in Chapter 3. Peter Winstanley let us know about a longstanding error in our Latin in Chapter 3. Chris Wrobel made corrections to the code in the chapter on le I/O and exceptions. Moshe Zadka has made invaluable contributions to this project. In addition to writing the rst draft of the chapter on Dictionaries, he provided continual guidance in the early stages of the book. Christoph Zwerschke sent several corrections and pedagogic suggestions, and explained the difference between gleich and selbe. James Mayer sent us a whole slew of spelling and typographical errors, including two in the contributor list. Hayden McAfee caught a potentially confusing inconsistency between two examples.

CONTENTS

xv

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Angel Arnal is part of an international team of translators working on the Spanish version of the text. He has also found several errors in the English version. Tauhidul Hoque and Lex Berezhny created the illustrations in Chapter 1 and improved many of the other illustrations. Dr. Michele Alzetta caught an error in Chapter 8 and sent some interesting pedagogic comments and suggestions about Fibonacci and Old Maid. Andy Mitchell caught a typo in Chapter 1 and a broken example in Chapter 2. Kalin Harvey suggested a clarication in Chapter 7 and caught some typos. Christopher P. Smith caught several typos and is helping us prepare to update the book for Python 2.2. David Hutchins caught a typo in the Foreword. Gregor Lingl is teaching Python at a high school in Vienna, Austria. He is working on a German translation of the book, and he caught a couple of bad errors in Chapter 5. Julie Peters caught a typo in the Preface.

CONTENTS

CHAPTER

ONE

THE WAY OF THE PROGRAM


The goal of this book is to teach you to think like a computer scientist. This way of thinking combines some of the best features of mathematics, engineering, and natural science. Like mathematicians, computer scientists use formal languages to denote ideas (specically computations). Like engineers, they design things, assembling components into systems and evaluating tradeoffs among alternatives. Like scientists, they observe the behavior of complex systems, form hypotheses, and test predictions. The single most important skill for a computer scientist is problem solving. Problem solving means the ability to formulate problems, think creatively about solutions, and express a solution clearly and accurately. As it turns out, the process of learning to program is an excellent opportunity to practice problem-solving skills. Thats why this chapter is called, The way of the program. On one level, you will be learning to program, a useful skill by itself. On another level, you will use programming as a means to an end. As we go along, that end will become clearer.

1.1 The Python programming language


The programming language you will be learning is Python. Python is an example of a highlevel language; other high-level languages you might have heard of are C++, PHP, Pascal, C#, and Java. As you might infer from the name high-level language, there are also low-level languages, sometimes referred to as machine languages or assembly languages. Loosely speaking, computers can only execute programs written in low-level languages. Thus, programs written in a high-level language have to be translated into something more suitable before they can run. Almost all programs are written in high-level languages because of their advantages. It is much easier to program in a high-level language so programs take less time to write, they are shorter and easier to read, and they are more likely to be correct. Second, high-level languages are portable, meaning that they can run on different kinds of computers with few or no modications. The engine that translates and runs Python is called the Python Interpreter: There are two ways to use it: immediate mode and script mode. In immediate mode, you type Python expressions into the Python Interpreter window, and the interpreter immediately shows the result: 1

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

The >>> is called the Python prompt. The interpreter uses the prompt to indicate that it is ready for instructions. We typed 2 + 2, and the interpreter evaluated our expression, and replied 4, and on the next line it gave a new prompt, indicating that it is ready for more input. Alternatively, you can write a program in a le and use the interpreter to execute the contents of the le. Such a le is called a script. Scripts have the advantage that they can be saved to disk, printed, and so on. In this Rhodes Local Edition of the textbook, we use a program development environment called PyScripter. (It is available at http://code.google.com/p/pyscripter.) There are various other development environments. If youre using one of the others, you might be better off working with the authors original book rather than this edition. For example, we created a le named firstprogram.py using PyScripter. By convention, les that contain Python programs have names that end with .py To execute the program, we can click the Run button in PyScripter:

Most programs are more interesting than this one. Working directly in the interpreter is convenient for testing short bits of code because you get immediate feedback. Think of it as scratch paper used to help you work out problems. Anything longer than a few lines should be put into a script. 2 Chapter 1. The way of the program

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1.2 What is a program?


A program is a sequence of instructions that species how to perform a computation. The computation might be something mathematical, such as solving a system of equations or nding the roots of a polynomial, but it can also be a symbolic computation, such as searching and replacing text in a document or (strangely enough) compiling a program. The details look different in different languages, but a few basic instructions appear in just about every language: input Get data from the keyboard, a le, or some other device. output Display data on the screen or send data to a le or other device. math Perform basic mathematical operations like addition and multiplication. conditional execution Check for certain conditions and execute the appropriate sequence of statements. repetition Perform some action repeatedly, usually with some variation. Believe it or not, thats pretty much all there is to it. Every program youve ever used, no matter how complicated, is made up of instructions that look more or less like these. Thus, we can describe programming as the process of breaking a large, complex task into smaller and smaller subtasks until the subtasks are simple enough to be performed with sequences of these basic instructions. That may be a little vague, but we will come back to this topic later when we talk about algorithms.

1.3 What is debugging?


Programming is a complex process, and because it is done by human beings, it often leads to errors. Programming errors are called bugs and the process of tracking them down and correcting them is called debugging. Use of the term bug to describe small engineering difculties dates back to at least 1889, when Thomas Edison had a bug with his phonograph. Three kinds of errors can occur in a program: syntax errors, runtime errors, and semantic errors. It is useful to distinguish between them in order to track them down more quickly.

1.4 Syntax errors


Python can only execute a program if the program is syntactically correct; otherwise, the process fails and returns an error message. Syntax refers to the structure of a program and the rules about that structure. For example, in English, a sentence must begin with a capital letter and end with a period. this sentence contains a syntax error. So does this one For most readers, a few syntax errors are not a signicant problem, which is why we can read the poetry of e. e. cummings without problems. Python is not so forgiving. If there is a single 1.2. What is a program? 3

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

syntax error anywhere in your program, Python will display an error message and quit, and you will not be able to run your program. During the rst few weeks of your programming career, you will probably spend a lot of time tracking down syntax errors. As you gain experience, though, you will make fewer errors and nd them faster.

1.5 Runtime errors


The second type of error is a runtime error, so called because the error does not appear until you run the program. These errors are also called exceptions because they usually indicate that something exceptional (and bad) has happened. Runtime errors are rare in the simple programs you will see in the rst few chapters, so it might be a while before you encounter one.

1.6 Semantic errors


The third type of error is the semantic error. If there is a semantic error in your program, it will run successfully, in the sense that the computer will not generate any error messages, but it will not do the right thing. It will do something else. Specically, it will do what you told it to do. The problem is that the program you wrote is not the program you wanted to write. The meaning of the program (its semantics) is wrong. Identifying semantic errors can be tricky because it requires you to work backward by looking at the output of the program and trying to gure out what it is doing.

1.7 Experimental debugging


One of the most important skills you will acquire is debugging. Although it can be frustrating, debugging is one of the most intellectually rich, challenging, and interesting parts of programming. In some ways, debugging is like detective work. You are confronted with clues, and you have to infer the processes and events that led to the results you see. Debugging is also like an experimental science. Once you have an idea what is going wrong, you modify your program and try again. If your hypothesis was correct, then you can predict the result of the modication, and you take a step closer to a working program. If your hypothesis was wrong, you have to come up with a new one. As Sherlock Holmes pointed out, When you have eliminated the impossible, whatever remains, however improbable, must be the truth. (A. Conan Doyle, The Sign of Four) For some people, programming and debugging are the same thing. That is, programming is the process of gradually debugging a program until it does what you want. The idea is that you should start with a program that does something and make small modications, debugging them as you go, so that you always have a working program. 4 Chapter 1. The way of the program

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

For example, Linux is an operating system kernel that contains millions of lines of code, but it started out as a simple program Linus Torvalds used to explore the Intel 80386 chip. According to Larry Greeneld, one of Linuss earlier projects was a program that would switch between displaying AAAA and BBBB. This later evolved to Linux (The Linux Users Guide Beta Version 1). Later chapters will make more suggestions about debugging and other programming practices.

1.8 Formal and natural languages


Natural languages are the languages that people speak, such as English, Spanish, and French. They were not designed by people (although people try to impose some order on them); they evolved naturally. Formal languages are languages that are designed by people for specic applications. For example, the notation that mathematicians use is a formal language that is particularly good at denoting relationships among numbers and symbols. Chemists use a formal language to represent the chemical structure of molecules. And most importantly: Programming languages are formal languages that have been designed to express computations. Formal languages tend to have strict rules about syntax. For example, 3+3=6 is a syntactically correct mathematical statement, but 3=+6$ is not. H2 O is a syntactically correct chemical name, but 2 Zz is not. Syntax rules come in two avors, pertaining to tokens and structure. Tokens are the basic elements of the language, such as words, numbers, and chemical elements. One of the problems with 3=+6$ is that $ is not a legal token in mathematics (at least as far as we know). Similarly, 2 Zz is not legal because there is no element with the abbreviation Zz. The second type of syntax rule pertains to the structure of a statement that is, the way the tokens are arranged. The statement 3=+6$ is structurally illegal because you cant place a plus sign immediately after an equal sign. Similarly, molecular formulas have to have subscripts after the element name, not before. When you read a sentence in English or a statement in a formal language, you have to gure out what the structure of the sentence is (although in a natural language you do this subconsciously). This process is called parsing. For example, when you hear the sentence, The other shoe fell, you understand that the other shoe is the subject and fell is the verb. Once you have parsed a sentence, you can gure out what it means, or the semantics of the sentence. Assuming that you know what a shoe is and what it means to fall, you will understand the general implication of this sentence. Although formal and natural languages have many features in common tokens, structure, syntax, and semantics there are many differences: ambiguity Natural languages are full of ambiguity, which people deal with by using contextual clues and other information. Formal languages are designed to be nearly or com-

1.8. Formal and natural languages

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

pletely unambiguous, which means that any statement has exactly one meaning, regardless of context. redundancy In order to make up for ambiguity and reduce misunderstandings, natural languages employ lots of redundancy. As a result, they are often verbose. Formal languages are less redundant and more concise. literalness Formal languages mean exactly what they say. On the other hand, natural languages are full of idiom and metaphor. If someone says, The other shoe fell, there is probably no shoe and nothing falling. Youll need to nd the original joke to understand the idiomatic meaning of the other shoe falling. Yahoo! Answers thinks it knows! People who grow up speaking a natural languageeveryoneoften have a hard time adjusting to formal languages. In some ways, the difference between formal and natural language is like the difference between poetry and prose, but more so: poetry Words are used for their sounds as well as for their meaning, and the whole poem together creates an effect or emotional response. Ambiguity is not only common but often deliberate. prose The literal meaning of words is more important, and the structure contributes more meaning. Prose is more amenable to analysis than poetry but still often ambiguous. program The meaning of a computer program is unambiguous and literal, and can be understood entirely by analysis of the tokens and structure. Here are some suggestions for reading programs (and other formal languages). First, remember that formal languages are much more dense than natural languages, so it takes longer to read them. Also, the structure is very important, so it is usually not a good idea to read from top to bottom, left to right. Instead, learn to parse the program in your head, identifying the tokens and interpreting the structure. Finally, the details matter. Little things like spelling errors and bad punctuation, which you can get away with in natural languages, can make a big difference in a formal language.

1.9 The rst program


Traditionally, the rst program written in a new language is called Hello, World! because all it does is display the words, Hello, World! In Python, the script looks like this: (For scripts, well show line numbers to the left of the Python statements.)
1

print("Hello, World!")

This is an example of using the print function, which doesnt actually print anything on paper. It displays a value on the screen. In this case, the result shown is
Hello, World!

The quotation marks in the program mark the beginning and end of the value; they dont appear in the result. Some people judge the quality of a programming language by the simplicity of the Hello, World! program. By this standard, Python does about as well as possible. 6 Chapter 1. The way of the program

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1.10 Comments
As programs get bigger and more complicated, they get more difcult to read. Formal languages are dense, and it is often difcult to look at a piece of code and gure out what it is doing, or why. For this reason, it is a good idea to add notes to your programs to explain in natural language what the program is doing. A comment in a computer program is text that is intended only for the human reader it is completely ignored by the interpreter. In Python, the # token starts a comment. The rest of the line is ignored. Here is a new version of Hello, World!.
1 2 3 4 5 6 7

#--------------------------------------------------# This demo program shows off how elegant Python is! # Written by Joe Soap, December 2010. # Anyone may freely copy or modify this program. #--------------------------------------------------print("Hello, World!") # Isnt this easy!

Youll also notice that weve left a blank line in the program. Blank lines are also ignored by the interpreter, but comments and blank lines can make your programs much easier for humans to parse. Use them liberally!

1.11 Glossary
algorithm A set of specic steps for solving a category of problems. bug An error in a program. comment Information in a program that is meant for other programmers (or anyone reading the source code) and has no effect on the execution of the program. debugging The process of nding and removing any of the three kinds of programming errors. exception Another name for a runtime error. formal language Any one of the languages that people have designed for specic purposes, such as representing mathematical ideas or computer programs; all programming languages are formal languages. high-level language A programming language like Python that is designed to be easy for humans to read and write. immediate mode A style of using Python where we type expressions at the command prompt, and the results are shown immediately. Contrast with script, and see the entry under Python shell. interpreter The engine that executes your Python scripts or expressions. 1.10. Comments 7

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

low-level language A programming language that is designed to be easy for a computer to execute; also called machine language or assembly language. natural language Any one of the languages that people speak that evolved naturally. object code The output of the compiler after it translates the program. parse To examine a program and analyze the syntactic structure. portability A property of a program that can run on more than one kind of computer. print function A function used in a program or script that causes the Python interpreter to display a value on its output device. problem solving The process of formulating a problem, nding a solution, and expressing the solution. program a sequence of instructions that species to a computer actions and computations to be performed. Python shell An interactive user interface to the Python interpreter. The user of a Python shell types commands at the prompt (>>>), and presses the return key to send these commands immediately to the interpreter for processing. The word shell comes from Unix. In the PyScripter used in this RLE version of the book, the Interpreter Window is where wed do the immediate mode interaction. runtime error An error that does not occur until the program has started to execute but that prevents the program from continuing. script A program stored in a le (usually one that will be interpreted). semantic error An error in a program that makes it do something other than what the programmer intended. semantics The meaning of a program. source code A program in a high-level language before being compiled. syntax The structure of a program. syntax error An error in a program that makes it impossible to parse and therefore impossible to interpret. token One of the basic elements of the syntactic structure of a program, analogous to a word in a natural language.

1.12 Exercises
1. Write an English sentence with understandable semantics but incorrect syntax. Write another English sentence which has correct syntax but has semantic errors. 2. Using the Python interpreter, type 1 + 2 and then hit return. Python evaluates this expression, displays the result, and then shows another prompt. * is the multiplication operator, and ** is the exponentiation operator. Experiment by entering different expressions and recording what is displayed by the Python interpreter. 8 Chapter 1. The way of the program

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

3. Type 1 2 and then hit return. Python tries to evaluate the expression, but it cant because the expression is not syntactically legal. Instead, it shows the error message:
File "<interactive input>", line 1 1 2 ^ SyntaxError: invalid syntax

In many cases, Python indicates where the syntax error occurred, but it is not always right, and it doesnt give you much information about what is wrong. So, for the most part, the burden is on you to learn the syntax rules. In this case, Python is complaining because there is no operator between the numbers. See if you can nd a few more examples of things that will produce error messages when you enter them at the Python prompt. Write down what you enter at the prompt and the last line of the error message that Python reports back to you. 4. Type print(hello). Python executes this, which has the effect of printing the letters h-e-l-l-o. Notice that the quotation marks that you used to enclose the string are not part of the output. Now type "hello" and describe your result. Make notes of when you see the quotation marks and when you dont. 5. Type cheese without the quotation marks. The output will look something like this:
Traceback (most recent call last): File "<interactive input>", line 1, in ? NameError: name cheese is not defined

This is a run-time error; specically, it is a NameError, and even more specically, it is an error because the name cheese is not dened. If you dont know what that means yet, you will soon. 6. Type 6 + 4 * 9 at the Python prompt and hit enter. Record what happens. Now create a python script with the following contents:
1

6 + 4 * 9

What happens when you run this script? Now change the script contents to:
1

print(6 + 4 * 9)

and run it again. What happened this time? Whenever an expression is typed at the Python prompt, it is evaluated and the result is automatically shown on the line below. (Like on your calculator, if you type this expression youll get the result 42.) A script is different, however. Evaluations of expressions are not automatically displayed, so it is necessary to use the print function to make the answer show up. It is hardly ever necessary to use the print function in immediate mode at the command prompt. 1.12. Exercises 9

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

10

Chapter 1. The way of the program

CHAPTER

TWO

VARIABLES, EXPRESSIONS AND STATEMENTS


2.1 Values and data types
A value is one of the fundamental things like a letter or a number that a program manipulates. The values we have seen so far are 4 (the result when we added 2 + 2), and "Hello, World!". These values are classied into different classes, or data types: 4 is an integer, and "Hello, World!" is a string, so-called because it contains a string of letters. You (and the interpreter) can identify strings because they are enclosed in quotation marks. If you are not sure what class a value falls into, Python has a function called type which can tell you.
>>> type("Hello, World!") <class str> >>> type(17) <class int>

Not surprisingly, strings belong to the class str and integers belong to the class int. Less obviously, numbers with a decimal point belong to a class called oat, because these numbers are represented in a format called oating-point. At this stage, you can treat the words class and type interchangeably. Well come back to a deeper understanding of what a class is in later chapters.
>>> type(3.2) <class float>

What about values like "17" and "3.2"? They look like numbers, but they are in quotation marks like strings.
>>> type("17") <class str> >>> type("3.2") <class str>

Theyre strings! 11

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Strings in Python can be enclosed in either single quotes () or double quotes ("), or three of each ( or """)
>>> type(This is a string.) <class str> >>> type("And so is this.") <class str> >>> type("""and this.""") <class str> >>> type(and even this...) <class str>

Double quoted strings can contain single quotes inside them, as in "Bruces beard", and single quoted strings can have double quotes inside them, as in The knights who say "Ni!". Strings enclosed with three occurrences of either quote symbol are called triple quoted strings. They can contain either single or double quotes:
>>> print("Oh no", she exclaimed, "Bens bike is broken!") "Oh no", she exclaimed, "Bens bike is broken!" >>>

Triple quoted strings can even span multiple lines:


>>> message = """This message will ... span several ... lines.""" >>> print(message) This message will span several lines. >>>

Python doesnt care whether you use single or double quotes or the three-of-a-kind quotes to surround your strings: once it has parsed the text of your program or command, the way it stores the value is identical in all cases, and the surrounding quotes are not part of the value. But when the interpreter wants to display a string, it has to decide which quotes to use to make it look like a string.
>>> This is a string. This is a string. >>> """And so is this.""" And so is this.

So the Python language designers usually chose to surround their strings by single quotes. What do think would happen if the string already contained single quotes? When you type a large integer, you might be tempted to use commas between groups of three digits, as in 42,000. This is not a legal integer in Python, but it does mean something else, which is legal:

12

Chapter 2. Variables, expressions and statements

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

>>> 42000 42000 >>> 42,000 (42, 0)

Well, thats not what we expected at all! Because of the comma, Python chose to treat this as a pair of values. Well come back to learn about pairs later. But, for the moment, remember not to put commas or spaces in your integers, no matter how big they are. Also revisit what we said in the previous chapter: formal languages are strict, the notation is concise, and even the smallest change might mean something quite different from what you intended.

2.2 Variables
One of the most powerful features of a programming language is the ability to manipulate variables. A variable is a name that refers to a value. The assignment statement creates new variables and gives them values:
>>> message = "Whats up, Doc?" >>> n = 17 >>> pi = 3.14159

This example makes three assignments. The rst assigns the string value "Whats up, Doc?" to a new variable named message. The second gives the integer 17 to n, and the third assigns the oating-point number 3.14159 to a variable called pi. The assignment token, =, should not be confused with equals, which uses the token ==. The assignment statement links a name, on the left hand side of the operator, with a value, on the right hand side. This is why you will get an error if you enter:
>>> 17 = n File "<interactive input>", line 1 SyntaxError: cant assign to literal

Tip: When reading or writing code, say to yourself n is assigned 17 or n gets the value 17. Dont say n equals 17. A common way to represent variables on paper is to write the name with an arrow pointing to the variables value. This kind of gure is called a state snapshot because it shows what state each of the variables is in at a particular instant in time. (Think of it as the variables state of mind). This diagram shows the result of executing the assignment statements:

If you ask the interpreter to evaluate a variable, it will produce the value that is currently linked to the variable: 2.2. Variables 13

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

>>> message Whats up, Doc? >>> n 17 >>> pi 3.14159

In each case the result is the value of the variable. Variables also have types; again, we can ask the interpreter what they are.
>>> type(message) <class str> >>> type(n) <class int> >>> type(pi) <class float>

The type of a variable is the type of the value it currently refers to. We use variables in a program to remember things, like the current score at the football game. But variables are variable. This means they can change over time, just like the scoreboard at a football game. You can assign a value to a variable, and later assign a different value to the same variable. (This is different from maths. In maths, if you give x the value 3, it cannot change to link to a different value half-way through your calculations!)
>>> day = "Thursday" >>> day Thursday >>> day = "Friday" >>> day Friday >>> day = 21 >>> day 21

Youll notice we changed the value of day three times, and on the third assignment we even gave it a value that was of a different type. A great deal of programming is about having the computer remember things, e.g. The number of missed calls on your phone, and then arranging to update or change the variable when you miss another call.

2.3 Variable names and keywords


Variable names can be arbitrarily long. They can contain both letters and digits, but they have to begin with a letter or an underscore. Although it is legal to use uppercase letters, by convention we dont. If you do, remember that case matters. Bruce and bruce are different variables. The underscore character ( _) can appear in a name. It is often used in names with multiple words, such as my_name or price_of_tea_in_china. 14 Chapter 2. Variables, expressions and statements

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

There are some situations in which names beginning with an underscore have special meaning, so a safe rule for beginners is to start all names with a letter. If you give a variable an illegal name, you get a syntax error:
>>> 76trombones = "big parade" SyntaxError: invalid syntax >>> more$ = 1000000 SyntaxError: invalid syntax >>> class = "Computer Science 101" SyntaxError: invalid syntax

76trombones is illegal because it does not begin with a letter. more$ is illegal because it contains an illegal character, the dollar sign. But whats wrong with class? It turns out that class is one of the Python keywords. Keywords dene the languages syntax rules and structure, and they cannot be used as variable names. Python has thirty-something keywords (and every now and again improvements to Python introduce or eliminate one or two): and def nally in pass yield as del for is raise True assert elif from lambda return False break else global nonlocal try None class except if not while continue exec import or with

You might want to keep this list handy. If the interpreter complains about one of your variable names and you dont know why, see if it is on this list. Programmers generally choose names for their variables that are meaningful to the human readers of the program they help the programmer document, or remember, what the variable is used for. Caution: Beginners sometimes confuse meaningful to the human readers with meaningful to the computer. So theyll wrongly think that because theyve called some variable average or pi, it will somehow magically calculate an average, or magically know that the variable pi should have a value like 3.14159. No! The computer doesnt understand what you intend the variable to mean. So youll nd some instructors who deliberately dont choose meaningful names when they teach beginners not because we dont think it is a good habit, but because were trying to reinforce the message that you the programmer must write the program code to calculate the average, and you must write an assignment statement to give the variable pi the value you want it to have.

2.4 Statements
A statement is an instruction that the Python interpreter can execute. We have only seen the assignment statement so far. Some other kinds of statements that well see shortly are while 2.4. Statements 15

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

statements, for statements, if statements, and import statements. (There are other kinds too!) When you type a statement on the command line, Python executes it. Statements dont produce any result.

2.5 Evaluating expressions


An expression is a combination of values, variables, operators, and calls to functions. If you type an expression at the Python prompt, the interpreter evaluates it and displays the result:
>>> 1 + 1 2 >>> len("hello") 5

In this example len is a built-in Python function that returns the number of characters in a string. Weve previously seen the print and the type functions, so this is our third example of a function! The evaluation of an expression produces a value, which is why expressions can appear on the right hand side of assignment statements. A value all by itself is a simple expression, and so is a variable.
>>> 17 17 >>> y = 3.14 >>> x = len("hello") >>> x 5 >>> y 3.14

2.6 Operators and operands


Operators are special tokens that represent computations like addition, multiplication and division. The values the operator uses are called operands. The following are all legal Python expressions whose meaning is more or less clear:
20+32 hour-1 hour*60+minute minute/60 5**2 (5+9)*(15-7)

The tokens +, -, and *, and the use of parenthesis for grouping, mean in Python what they mean in mathematics. The asterisk (*) is the token for multiplication, and ** is the token for exponentiation.
>>> 2 ** 3 8

16

Chapter 2. Variables, expressions and statements

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

>>> 3 ** 2 9

When a variable name appears in the place of an operand, it is replaced with its value before the operation is performed. Addition, subtraction, multiplication, and exponentiation all do what you expect. Example: so let us convert 645 minutes into hours:
>>> minutes = 645 >>> hours = minutes/60 >>> hours 10.75

Oops! In Python 3, the division operator / always yields a oating point result. What we might have wanted to know was how many whole hours there are, and how many minutes remain. Python gives us two different avours of the division operator. The second, called integer division uses the token //. It always truncates its result down to the next smallest integer (to the left on the number line).
>>> 7 / 4 1.75 >>> 7 // 4 1 >>> minutes = 645 >>> hours = minutes//60 >>> hours 10

Take care that you choose the correct falvour of the division operator. If youre working with expressions where you need oating point values, use the division operator that does the division accurately.

2.7 Type converter functions


Here well look at three more Python functions, int, oat and str, which will (attempt to) convert their arguments into types int, oat and str respectively. We call these type converter functions. The int function can take a oating point number or a string, and turn it into an int. For oating point numbers, it discards the decimal portion of the number - a process we call truncation towards zero on the number line. Let us see this in action:
>>> 3 >>> 3 >>> 3 >>> int(3.14) int(3.9999) int(3.0) int(-3.999) # Note that the result is closer to zero # This doesnt round to the closest int!

2.7. Type converter functions

17

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

-3 >>> int(minutes/60) 10 >>> int("2345") 2345 >>> int(17) 17 >>> int("23 bottles")

# parse a string to produce an int # even works if arg is already an int

This last case doesnt look like a number what do we expect?


Traceback (most recent call last): File "<interactive input>", line 1, in <module> ValueError: invalid literal for int() with base 10: 23 bottles

The type converter oat can turn an integer, a oat, or a syntactically legal string into a oat:
>>> float(17) 17.0 >>> float("123.45") 123.45

The type converter str turns its argument into a string:


>>> str(17) 17 >>> str(123.45) 123.45

2.8 Order of operations


When more than one operator appears in an expression, the order of evaluation depends on the rules of precedence. Python follows the same precedence rules for its mathematical operators that mathematics does. The acronym PEMDAS is a useful way to remember the order of operations: 1. Parentheses have the highest precedence and can be used to force an expression to evaluate in the order you want. Since expressions in parentheses are evaluated rst, 2 * (3-1) is 4, and (1+1)**(5-2) is 8. You can also use parentheses to make an expression easier to read, as in (minute * 100) / 60, even though it doesnt change the result. 2. Exponentiation has the next highest precedence, so 2**1+1 is 3 and not 4, and 3*1**3 is 3 and not 27. 3. Multiplication and both Division operators have the same precedence, which is higher than Addition and Subtraction, which also have the same precedence. So 2*3-1 yields 5 rather than 4, and 5-2*2 is 1, not 6. 4. Operators with the same precedence are evaluated from left-to-right. In algebra we say they are left-associative. So in the expression 6-3+2, the subtraction happens rst, 18 Chapter 2. Variables, expressions and statements

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

yielding 3. We then add 2 to get the result 5. If the operations had been evaluated from right to left, the result would have been 6-(3+2), which is 1. (The acronym PEDMAS could mislead you to thinking that division has higher precedence than multiplication, and addition is done ahead of subtraction - dont be misled. Subtraction and addition are at the same precedence, and the left-to-right rule applies.) Due to some historical quirk, an exception to the left-to-right left-associative rule is the exponentiation operator **, so a useful hint is to always use parentheses to force exactly the order you want when exponentiation is involved:
>>> 2 ** 3 ** 2 512 >>> (2 ** 3) ** 2 64 # the right-most ** operator gets done first! # Use parentheses to force the order you want!

The immediate mode command prompt of Python is great for exploring and experimenting with expressions like this.

2.9 Operations on strings


In general, you cannot perform mathematical operations on strings, even if the strings look like numbers. The following are illegal (assuming that message has type string):
>>> >>> >>> >>> message-1 "Hello"/123 message*"Hello" "15"+2 # # # # Error Error Error Error

Interestingly, the + operator does work with strings, but for strings, the + operator represents concatenation, not addition. Concatenation means joining the two operands by linking them end-to-end. For example:
1 2 3

fruit = "banana" baked_good = " nut bread" print(fruit + baked_good)

The output of this program is banana nut bread. The space before the word nut is part of the string, and is necessary to produce the space between the concatenated strings. The * operator also works on strings; it performs repetition. For example, Fun*3 is FunFunFun. One of the operands has to be a string; the other has to be an integer. On one hand, this interpretation of + and * makes sense by analogy with addition and multiplication. Just as 4*3 is equivalent to 4+4+4, we expect "Fun"*3 to be the same as "Fun"+"Fun"+"Fun", and it is. On the other hand, there is a signicant way in which string concatenation and repetition are different from integer addition and multiplication. Can you think of a property that addition and multiplication have that string concatenation and repetition do not?

2.9. Operations on strings

19

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

2.10 Input
There is a built-in function in Python for getting input from the user:
1

n = input("Please enter your name: ")

A sample run of this script in PyScripter would pop up a dialog window like this:

The user of the program can enter the name and click OK, and when this happens the text that has been entered is returned from the input function, and in this case assigned to the variable n. Even if you asked the user to enter their age, you would get back a string like "17". It would be your job, as the programmer, to convert that string into a int or a oat, using the int or oat converter functions we saw earlier.

2.11 Composition
So far, we have looked at the elements of a program variables, expressions, statements, and function calls in isolation, without talking about how to combine them. One of the most useful features of programming languages is their ability to take small building blocks and compose them into larger chunks. For example, we know how to get the user to enter some input, we know how to convert the string we get into a oat, we know how to write a complex expression, and we know how to print values. Lets put these together in a small four-step program that asks the user to input a value for the radius of a circle, and then computes the area of the circle from the formula

Firstly, well do the four steps one at a time:


1 2 3 4

response = input("What is your radius? ") r = float(response) area = 3.14159 * r**2 print("The area is ", area)

Now lets compose the rst two lines into a single line of code, and compose the second two lines into another line of code.

20

Chapter 2. Variables, expressions and statements

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2

r = float( input("What is your radius? ") ) print("The area is ", 3.14159 * r**2)

If we really wanted to be tricky, we could write it all in one statement:


1

print("The area is ", 3.14159*float(input("What is your radius?"))**2)

Such compact code may not be most understandable for humans, but it does illustrate how we can compose bigger chunks from our building blocks. If youre ever in doubt about whether to compose code or fragment it into smaller steps, try to make it as simple as you can for the human to follow. My choice would be the rst case above, with four separate steps.

2.12 The modulus operator


The modulus operator works on integers (and integer expressions) and gives the remainder when the rst number is divided by the second. In Python, the modulus operator is a percent sign (%). The syntax is the same as for other operators. It has the same precedence as the multiplication operator.
>>> >>> 2 >>> >>> 1 q = 7 // 3 print(q) r = 7 % 3 print(r) # This is integer division operator

So 7 divided by 3 is 2 with a remainder of 1. The modulus operator turns out to be surprisingly useful. For example, you can check whether one number is divisible by anotherif x % y is zero, then x is divisible by y. Also, you can extract the right-most digit or digits from a number. For example, x % 10 yields the right-most digit of x (in base 10). Similarly x % 100 yields the last two digits. It is also extremely useful for doing conversions, say from seconds, to hours, minutes and seconds. So lets write a program to ask the user to enter some seconds, and well convert them into hours, minutes, and remaining seconds.
1 2 3 4 5 6 7 8

total_secs = int(input("How many seconds, in total?")) hours = total_secs // 3600 secs_still_remaining = total_secs % 3600 minutes = secs_still_remaining // 60 secs_finally_remaining = secs_still_remaining % 60 print("Hrs=", hours, " mins=", minutes, "secs=", secs_finally_remaining)

2.12. The modulus operator

21

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

2.13 Glossary
assignment statement A statement that assigns a value to a name (variable). To the left of the assignment operator, =, is a name. To the right of the assignment token is an expression which is evaluated by the Python interpreter and then assigned to the name. The difference between the left and right hand sides of the assignment statement is often confusing to new programmers. In the following assignment:
n = n + 1

n plays a very different role on each side of the =. On the right it is a value and makes up part of the expression which will be evaluated by the Python interpreter before assigning it to the name on the left. assignment token = is Pythons assignment token, which should not be confused with the mathematical comparison operator using the same symbol. composition The ability to combine simple expressions and statements into compound statements and expressions in order to represent complex computations concisely. concatenate To join two strings end-to-end. data type A set of values. The type of a value determines how it can be used in expressions. So far, the types you have seen are integers (int), oating-point numbers (float), and strings (str). evaluate To simplify an expression by performing the operations in order to yield a single value. expression A combination of variables, operators, and values that represents a single result value. oat A Python data type which stores oating-point numbers. Floating-point numbers are stored internally in two parts: a base and an exponent. When printed in the standard format, they look like decimal numbers. Beware of rounding errors when you use floats, and remember that they are only approximate values. int A Python data type that holds positive and negative whole numbers. integer division An operation that divides one integer by another and yields an integer. Integer division yields only the whole number of times that the numerator is divisible by the denominator and discards any remainder. keyword A reserved word that is used by the compiler to parse program; you cannot use keywords like if, def, and while as variable names. modulus operator An operator, denoted with a percent sign ( %), that works on integers and yields the remainder when one number is divided by another. operand One of the values on which an operator operates. operator A special symbol that represents a simple computation like addition, multiplication, or string concatenation.

22

Chapter 2. Variables, expressions and statements

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

rules of precedence The set of rules governing the order in which expressions involving multiple operators and operands are evaluated. state snapshot A graphical representation of a set of variables and the values to which they refer, taken at a particular instant during the programs execution. statement An instruction that the Python interpreter can execute. So far we have only seen the assignment statement, but we will soon meet the import statement and the for statement. str A Python data type that holds a string of characters. value A number or string (or other things to be named later) that can be stored in a variable or computed in an expression. variable A name that refers to a value. variable name A name given to a variable. Variable names in Python consist of a sequence of letters (a..z, A..Z, and _) and digits (0..9) that begins with a letter. In best programming practice, variable names should be chosen so that they describe their use in the program, making the program self documenting.

2.14 Exercises
1. Take the sentence: All work and no play makes Jack a dull boy. Store each word in a separate variable, then print out the sentence on one line using print. 2. Add parenthesis to the expression 6 * 1 - 2 to change its value from 4 to -6. 3. Place a comment before a line of code that previously worked, and record what happens when you rerun the program. 4. Start the Python interpreter and enter bruce + 4 at the prompt. This will give you an error:
NameError: name bruce is not defined

Assign a value to bruce so that bruce + 4 evaluates to 10. 5. The formula for computing the nal amount if one is earning compound interest is given on Wikipedia as

2.14. Exercises

23

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Write a Python program that assigns the principal amount of R10000 to variable p, assign to n the value 12, and assign to r the interest rate of 8%. Then have the program prompt the user for the number of months t that the money will be compounded for. Calculate and print the nal amount after t months. 6. Evaluate the following numerical expressions in your head, then use the Python interpreter to check your results: (a) >>> 5 % 2 (b) >>> 9 % 5 (c) >>> 15 % 12 (d) >>> 12 % 15 (e) >>> 6 % 6 (f) >>> 0 % 7 (g) >>> 7 % 0 What happened with the last example? Why? If you were able to correctly anticipate the computers response in all but the last one, it is time to move on. If not, take time now to make up examples of your own. Explore the modulus operator until you are condent you understand how it works. 7. You look at the clock and it is exactly 2pm. You set an alarm to go off in 51 hours. At what time does the alarm go off? (Hint: you could count on your ngers, but this is not what were after. If you are tempted to count on your ngers, change the 51 to 5100.) 8. Write a Python program to solve the general version of the above problem. Ask the user for the time now (in hours), and ask for the number of hours to wait. Your program should output what the time will be on the clock when the alarm goes off.

24

Chapter 2. Variables, expressions and statements

CHAPTER

THREE

HELLO, LITTLE TURTLES!


There are many modules in Python that provide very powerful features that we can use in our own programs. Some of these can send email, or fetch web pages. The one well look at in this chapter allows us to create turtles and get them to draw shapes and patterns. The turtles are fun, but the real purpose of the chapter is to teach ourselves a little more Python, and to develop our theme of computational thinking, or thinking like a computer scientist. Most of the Python covered here will be explored in more depth later.

3.1 Our rst turtle program


Lets write a couple of lines of Python program to create a new turtle and start drawing a rectangle. (Well call the variable that refers to our rst turtle alex, but we can choose another name if we follow the naming rules from the previous chapter).
1 2 3 4 5 6 7 8 9

import turtle wn = turtle.Screen() alex = turtle.Turtle() alex.forward(50) alex.left(90) alex.forward(30) wn.mainloop()

# allows us to use turtles # creates a playground for turtles # create a turtle, assign to alex # tell alex to move forward by 50 units # tell alex to turn by 90 degrees # complete the second side of a rectangle # Wait for user to close window

When we run this program, a new window pops up:

25

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Here are a couple of things youll need to understand about this program. The rst line tells Python to load a module named turtle. That module brings us two new types that we can use: the Turtle type, and the Screen type. The dot notation turtle.Turtle means The Turtle type that is dened within the turtle module. (Remember that Python is case sensitive, so the module name, with a lowercase t, is different from the type Turtle.) We then create and open what it calls a screen (we would perfer to call it a window), which we assign to variable wn. Every window contains a canvas, which is the area inside the window on which we can draw. In line 3 we create a turtle. The variable alex is made to refer to this turtle. These rst three lines set us up, ready for doing some useful things. In lines 5-7, we instruct the object alex to move, and to turn. We do this by invoking, or activating, alexs methods these are the instructions that all turtles know how to respond to. The last line plays a part too: the wn variable refers to the window shown above. When we invoke its mainloop method, it enters a state where it waits for events (like keypresses, or mouse movement and clicks). The program will terminate when the user closes the window. An object can have various methods things it can do and it can also have attributes (sometimes called properties). For example, each turtle has a color attribute. The method invocation alex.color(red) will make alex red, and drawing will be red too. (Note the word color is spelled the American way!) The color of the turtle, the width of its pen, the position of the turtle within the window, which way it is facing, and so on are all part of its current state. Similarly, the window object has a background color, and some text in the title bar, and a size and position on the screen. These are all part of the state of the window object. Quite a number of methods exist that allow us to modify the turtle and the window objects. Well just show a couple. In this program weve only commented those lines that are different from the previous example (and weve used a different variable name for this turtle):
1 2 3 4

import turtle wn = turtle.Screen() wn.bgcolor("lightgreen") wn.title("Hello, Tess!")

# set the window background color # set the window title

26

Chapter 3. Hello, little turtles!

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

5 6 7 8 9 10 11 12 13 14

tess = turtle.Turtle() tess.color("blue") tess.pensize(3) tess.forward(50) tess.left(120) tess.forward(50) wn.mainloop()

# tell tess to change her color # tell tess to set her pen width

When we run this program, this new window pops up, and will remain on the screen until we close it.

Extend this program ... 1. Modify this program so that before it creates the window, it prompts the user to enter the desired background color. It should store the users responses in a variable, and modify the color of the window according to the users wishes. (Hint: you can nd a list of permitted color names at http://www.tcl.tk/man/tcl8.4/TkCmd/colors.htm. It includes some quite unusual ones, like peach puff and HotPink.) 2. Do similar changes to allow the user, at runtime, to set tess color. 3. Do the same for the width of tess pen. Hint: your dialog with the user will return a string, but tess pensize method expects its argument to be an int. So youll need to convert the string to an int before you pass it to pensize.

3.2 Instances a herd of turtles


Just like we can have many different integers in a program, we can have many turtles. Each of them is called an instance. Each instance has its own attributes and methods so alex

3.2. Instances a herd of turtles

27

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

might draw with a thin black pen and be at some position, while tess might be going in her own direction with a fat pink pen.
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

import turtle wn = turtle.Screen() wn.bgcolor("lightgreen") wn.title("Tess & Alex") tess = turtle.Turtle() tess.color("hotpink") tess.pensize(5) alex = turtle.Turtle() tess.forward(80) tess.left(120) tess.forward(80) tess.left(120) tess.forward(80) tess.left(120) tess.right(180) tess.forward(80) alex.forward(50) alex.left(90) alex.forward(50) alex.left(90) alex.forward(50) alex.left(90) alex.forward(50) alex.left(90) wn.mainloop()

# Set up the window and its attributes

# create tess and set some attributes

# create alex # Make tess draw equilateral triangle

# complete the triangle # turn tess around # and move her away from the origin # make alex draw a square

Here is what happens when alex completes his rectangle, and tess completes her triangle:

28

Chapter 3. Hello, little turtles!

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Here are some How to think like a computer scientist observations: There are 360 degrees in a full circle. If we add up all the turns that a turtle makes, no matter what steps occurred between the turns, we can easily gure out if they add up to some multiple of 360. This should convince us that alex is facing in exactly the same direction as he was when he was rst created. (Geometry conventions have 0 degrees facing East, and that is the case here too!) We could have left out the last turn for alex, but that would not have been as satisfying. If were asked to draw a closed shape like a square or a rectangle, it is a good idea to complete all the turns and to leave the turtle back where it started, facing the same direction as it started in. This makes reasoning about the program and composing chunks of code into bigger programs easier for us humans! We did the same with tess: she drew her triangle, and turned through a full 360 degress. Then we turned her around and moved her aside. Even the blank line 18 is a hint about how the programmers mental chunking is working: in big terms, tess movements were chunked as draw the triangle (lines 12-17) and then move away from the origin (lines 19 and 20). One of the key uses for comments is to record our mental chunking, and big ideas. Theyre not always explicit in the code. And, uh-huh, two turtles may not be enough for a herd. But the important idea is that the turtle module gives us a kind of factory that lets us create as many turtles as we need. Each instance has its own state and behaviour.

3.3 The for loop


When we drew the square, it was quite tedious. We had to move then turn, move then turn, etc. etc. four times. If we were drawing a hexagon, or an octogon, or a polygon with 42 sides, it would have been worse. 3.3. The for loop 29

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

So a basic building block of all programs is to be able to repeat some code, over and over again. Pythons for loop solves this for us. Lets say we have some friends, and wed like to send them each an email inviting them to our party. We dont quite know how to send email yet, so for the moment well just print a message for each friend:
1 2 3 4

for f in ["Joe","Amy","Brad","Angelina","Zuki","Thandi","Paris"]: invite = "Hi " + f + ". Please come to my party on Saturday!" print(invite) # more code can follow here ...

When we run this, the output looks like this:


Hi Hi Hi Hi Hi Hi Hi Joe. Please come to my party on Saturday! Amy. Please come to my party on Saturday! Brad. Please come to my party on Saturday! Angelina. Please come to my party on Saturday! Zuki. Please come to my party on Saturday! Thandi. Please come to my party on Saturday! Paris. Please come to my party on Saturday!

The variable f in the for statement at line 1 is called the loop variable. We could have chosen any other variable name instead. Lines 2 and 3 are the loop body. The loop body is always indented. The indentation determines exactly what statements are in the body of the loop. On each iteration or pass of the loop, rst a check is done to see if there are still more items to be processed. If there are none left (this is called the terminating condition of the loop), the loop has nished. Program execution continues at the next statement after the loop body, (e.g. in this case the next statement below the comment in line 4). If there are items still to be processed, the loop variable is updated to refer to the next item in the list. This means, in this case, that the loop body is executed here 7 times, and each time f will refer to a different friend. At the end of each execution of the body of the loop, Python returns to the for statement, to see if there are more items to be handled, and to assign the next one to f.

3.4 Flow of Execution of the for loop


As a program executes, the interpreter always keeps track of which statement is about to be executed. We call this the control ow, of the ow of execution of the program. When humans execute programs, they often use their nger to point to each statement in turn. So we could think of control ow as Pythons moving nger. Control ow until now has been strictly top to bottom, one statement at a time. The for loop changes this. Flowchart of a for loop

30

Chapter 3. Hello, little turtles!

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Control ow is often easy to visualize and understand if we draw a owchart. This shows the exact steps and logic of how the for statement executes.

3.5 The loop simplies our turtle program


To draw a square wed like to do the same thing four times move the turtle, and turn. We previously used 8 lines to have alex draw the four sides of a square. This does exactly the same, but using just three lines:
1 2 3

for i in [0,1,2,3]: alex.forward(50) alex.left(90)

Some observations: While saving some lines of code might be convenient, it is not the big deal here. What is much more important is that weve found a repeating pattern of statements, and reorganized our program to repeat the pattern. Finding the chunks and somehow getting our programs arranged around those chunks is a vital skill in How to think like a computer scientist. The values [0,1,2,3] were provided to make the loop body execute 4 times. We could have used any four values, but these are the conventional ones to use. In fact, they are so popular that Python gives us special built-in range objects:
1 2

for i in range(4): # Executes the body with i = 0, then 1, then 2, then 3

3.5. The loop simplies our turtle program

31

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

3 4

for x in range(10): # sets x to each of ... [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]

Computer scientists like to count from 0! range can deliver a sequence of values to the loop variable in the for loop. They start at 0, and in these cases do not include the 4 or the 10. Our little trick earlier to make sure that alex did the nal turn to complete 360 degrees has paid off: if we had not done that, then we would not have been able to use a loop for the fourth side of the square. It would have become a special case, different from the other sides. When possible, wed much prefer to make our code t a general pattern, rather than have to create a special case. So to repeat something four times, a good Python programmer would do this:
1 2 3

for i in range(4): alex.forward(50) alex.left(90)

By now you should be able to see how to change our previous program so that tess can also use a for loop to draw her equilateral triangle. But now, what would happen if we made this change?
1 2 3 4

for c in ["yellow", "red", "purple", "blue"]: alex.color(c) alex.forward(50) alex.left(90)

A variable can also be assigned a value that is a list. So lists can also be used in more general situations, not only in the for loop. The code above could be rewritten like this:
1 2 3 4 5 6

# Assign a list to a variable clrs = ["yellow", "red", "purple", "blue"] for c in clrs: alex.color(c) alex.forward(50) alex.left(90)

3.6 A few more turtle methods and tricks


Turtle methods can use negative angles or distances. So tess.foward(-100) will move tess backwards, and tess.left(-30) turns her to the right. Additionally, because there are 360 degrees in a circle, turning 30 to the left will get tess facing in the same direction as turning 330 to the right! (The on-screen animation will differ, though you will be able to tell if tess is turning clockwise or counter-clockwise!) This suggests that we dont need both a left and a right turn method we could be minimalists, and just have one method. There is also a backward method. (If you are very nerdy, you might enjoy saying alex.backward(-100) to move alex forward!) 32 Chapter 3. Hello, little turtles!

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Part of thinking like a scientist is to understand more of the structure and rich relationships in our eld. So revising a few basic facts about geometry and number lines, and spotting the relationships between left, right, backward, forward, negative and positive distances or angles values is a good start if were going to play with turtles. A turtles pen can be picked up or put down. This allows us to move a turtle to a different place without drawing a line. The methods are
1 2 3

alex.penup() alex.forward(100) alex.pendown()

# this moves alex, but no line is drawn

Every turtle can have its own shape. The ones available out of the box are arrow, blank, circle, classic, square, triangle, turtle.
1

alex.shape("turtle")

We can speed up or slow down the turtles animation speed. (Animation controls how quickly the turtle turns and moves forward). Speed settings can be set between 1 (slowest) to 10 (fastest). But if we set the speed to 0, it has a special meaning turn off animation and go as fast as possible.
1

alex.speed(10)

A turtle can stamp its footprint onto the canvas, and this will remain after the turtle has moved somewhere else. Stamping works, even when the pen is up. Lets do an example that shows off some of these new features:
1 2 3 4 5 6 7 8 9 10 11

import turtle wn = turtle.Screen() wn.bgcolor("lightgreen") tess = turtle.Turtle() tess.shape("turtle") tess.color("blue") tess.penup() size = 20 for i in range(30): tess.stamp() # this is new

# leave an impression on the canvas

3.6. A few more turtle methods and tricks

33

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

12 13 14 15 16

size = size + 3 tess.forward(size) tess.right(24) wn.mainloop()

# increase the size on every iteration # move tess along # and turn her

Be careful now! How many times was the body of the loop executed? How many turtle images do we see on the screen? All except one of the shapes we see on the screen here are footprints created by stamp. But the program still only has one turtle instance can you gure out which one here is the real tess? (Hint: if youre not sure, write a new line of code after the for loop to change tess color, or to put her pen down and draw a line, or to change her shape, etc.)

3.7 Glossary
attribute Some state or value that belongs to a particular object. For example, tess has a color. canvas A surface within a window where drawing takes place. control ow See ow of execution in the next chapter. for loop A statement in Python for convenient repetition of statements in the body of the loop. loop body Any number of statements nested inside a loop. The nesting is indicated by the fact that the statements are indented under the for loop statement. loop variable A variable used as part of a for loop. It is assigned a different value on each iteration of the loop. instance An object of a certain type, or class. tess and alex are different instances of the class Turtle.

34

Chapter 3. Hello, little turtles!

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

method A function that is attached to an object. Invoking or activating the method causes the object to respond in some way, e.g. forward is the method when we say tess.forward(100). invoke An object has methods. We use the verb invoke to mean activate the method. Invoking a method is done by putting parentheses after the method name, with some possible arguments. So tess.forward() is an invocation of the forward method. module A le containing Python denitions and statements intended for use in other Python programs. The contents of a module are made available to the other program by using the import statement. object A thing to which a variable can refer. This could be a screen window, or one of the turtles we have created. range A built-in function in Python for generating sequences of integers. It is especially useful when we need to write a for loop that executes a xed number of times. terminating condition A condition that occurs which causes a loop to stop repeating its body. In the for loops we saw in this chapter, the terminating condition has been when there are no more elements to assign to the loop variable.

3.8 Exercises
1. Write a program that prints We like Pythons turtles! 1000 times. 2. Give three attributes of your cellphone object. Give three methods of your cellphone. 3. Write a program that uses a for loop to print One of the months of the year is January One of the months of the year is February ... etc 4. Suppose our turtle tess is at heading 0 facing east. We execute the statement tess.left(3645). What does tess do, and what is her nal heading? 5. Assume you have the assignment xs = [12, 10, 32, 3, 66, 17, 42, 99, 20] (a) Write a loop that prints each of the numbers on a new line. (b) Write a loop that prints each number and its square on a new line. (c) Write a loop that adds all the numbers from the list into a variable called total. You should set the total variable to have the value 0 before you start adding them up, and print the value in total after the loop has completed. (d) Print the product of all the numbers in the list. (product means all multiplied together) 6. Use for loops to make a turtle draw these regular polygons (regular means all sides the same lengths, all angles the same):

3.8. Exercises

35

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

An equilateral triangle A square A hexagon (six sides) An octagon (eight sides) 7. A drunk pirate makes a random turn and then takes 100 steps forward, makes another random turn, takes another 100 steps, turns another random amount, etc. A social science student records the angle of each turn before the next 100 steps are taken. Her experimental data is [160, -43, 270, -97, -43, 200, -940, 17, -86]. (Positive angles are counter-clockwise.) Use a turtle to draw the path taken by our drunk friend. 8. Enhance your program above to also tell us what the drunk pirates heading is after he has nished stumbling around. (Assume he begins at heading 0). 9. If you were going to draw a regular polygon with 18 sides, what angle would you need to turn the turtle at each corner? 10. At the interactive prompt, anticipate what each of the following lines will do, and then record what happens. Score yourself, giving yourself one point for each one you anticipate correctly:
>>> >>> >>> >>> >>> >>> >>> >>> >>> >>> >>> import turtle wn = turtle.Screen() tess = turtle.Turtle() tess.right(90) tess.left(3600) tess.right(-90) tess.speed(10) tess.left(3600) tess.speed(0) tess.left(3645) tess.forward(-100)

11. Write a program to draw a shape like this:

Hints: Try this on a piece of paper, moving and turning your cellphone as if it was a turtle. Watch how many complete rotations your cellphone makes before you complete the star. Since each full rotation is 360 degrees, you can gure out the total number of degrees that your phone was rotated through. If you divide that by 5, because there are ve points to the star, youll know how many degrees to turn the turtle at each point. You can hide a turtle behind its invisibility cloak if you dont want it 36 Chapter 3. Hello, little turtles!

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

shown. It will still draw its lines if its pen is down. The method is invoked as tess.hideturtle() . To make the turtle visible abain, use tess.showturtle() . 12. Write a program to draw a face of a clock that looks something like this:

13. Create a turtle, and assign it to a variable. When you ask for its type, what do you get? 14. What is the collective noun for turtles? (Hint: they dont come in herds.) 15. What the collective noun for pythons? Is a python a viper? Is a python venomous?

3.8. Exercises

37

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

38

Chapter 3. Hello, little turtles!

CHAPTER

FOUR

FUNCTIONS
4.1 Functions
In Python, a function is a named sequence of statements that belong together. Their primary purpose is to help us organize programs into chunks that match how we think about the problem. The syntax for a function denition is:
def NAME( PARAMETERS ): STATEMENTS

We can make up any names we want for the functions we create, except that we cant use a name that is a Python keyword, and the names must follow the rules for legal identiers. There can be any number of statements inside the function, but they have to be indented from the def. In the examples in this book, we will use the standard indentation of four spaces. Function denitions are the second of several compound statements we will see, all of which have the same pattern: 1. A header line which begins with a keyword and ends with a colon. 2. A body consisting of one or more Python statements, each indented the same amount 4 spaces is the Python standard from the header line. Weve already seen the for loop which follows this pattern. So looking again at the function denition, the keyword in the header is def, which is followed by the name of the function and some parameters enclosed in parentheses. The parameter list may be empty, or it may contain any number of parameters separated from one another by commas. In either case, the parentheses are required. The parameters species what information, if any, we have to provide in order to use the new function. Suppose were working with turtles, and a common operation we need is to draw squares. Draw a square* is an abstraction, or a mental chunk, of a number of smaller steps. So lets write a function to capture the pattern of this building block:
1 2 3 4

import turtle def draw_square(t, sz): """Make turtle t draw a square of sz."""

39

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

5 6 7 8 9 10 11 12 13 14 15 16 17

for i in range(4): t.forward(sz) t.left(90)

wn = turtle.Screen() # Set up the window and its attributes wn.bgcolor("lightgreen") wn.title("Alex meets a function") alex = turtle.Turtle() draw_square(alex, 50) wn.mainloop() # create alex # Call the function to draw the square

This function is named draw_square. It has two parameters one to tell the function which turtle to move around, and the other to tell it the size of the square we want drawn. Make sure you know where the body of the function ends it depends on the indentation, and the blank lines dont count for this purpose! docstrings If the rst thing after the function header is a triple-quoted string, it is called a docstring and gets special treatment in Python and in some of the programming tools. For example, when using PyScripter, when we type a function name it will pop up a tooltip showing the parameters of the function, and the text from the docstring. So the string we write as documentation at the start of a function is retrievable by python tools at runtime. This is different from comments in our code, which are completely eliminated when the program is parsed. By convention, Python programmers use docstrings for the key documentation of their functions. Dening a new function does not make the function run. To do that we need a function call. Weve already seen how to call some built-in functions like print, range and int. Function calls contain the name of the function being executed followed by a list of values, called arguments,

40

Chapter 4. Functions

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

which are assigned to the parameters in the function denition. So in the second last line of the program, we call the function, and pass alex as the turtle to be manipulated, and 50 as the size of the square we want. While the function is executing, then, the variable sz refers to the value 50, and the variable t refers to the same turtle instance that the variable alex refers to. Once weve dened a function, we can call it as often as we like, and its statements will be executed each time we call it. And we could use it to get any of our turtles to draw a square. In the next example, weve changed the draw_square function a little, and we get tess to draw 15 squares, with some variations.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23

import turtle def draw_multicolor_square(t, sz): """Make turtle t draw a multi-color square of sz.""" for i in [red,purple,hotpink,blue]: t.color(i) t.forward(sz) t.left(90) wn = turtle.Screen() wn.bgcolor("lightgreen") tess = turtle.Turtle() tess.pensize(3) # Set up the window and its attributes

# create tess and set some attributes

size = 20 # size of the smallest square for i in range(15): draw_multicolor_square(tess, size) size = size + 10 # increase the size for next time tess.forward(10) # move tess along a little tess.right(18) # and give her some extra turn wn.mainloop()

4.2 Functions can call other functions


Lets assume now we want a function to draw a rectangle. We need to be able to call the function with different arguments for width and height. And, unlike the case of the square, we cannot repeat the same thing 4 times, because the four sides are not equal. 4.2. Functions can call other functions 41

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

So we eventually come up with this rather nice code that can draw a rectangle.
1 2 3 4 5 6 7

def draw_rectangle(t, w, h): """Get turtle t to draw a rectangle of width w and height h.""" for i in range(2): t.forward(w) t.left(90) t.forward(h) t.left(90)

The parameter names are deliberately chosen as single letters to ensure theyre not misunderstood. In real programs, once weve had more experience, we will insist on better variable names than this. But the point is that the program doesnt understand that were drawing a rectangle, or that the parameters represent the width and the height. Concepts like rectangle, width, and height are the meaning we humans have, not concepts that the program or the computer understands. Thinking like a scientist involves looking for patterns and relationships. In the code above, weve done that to some extent. We did not just draw four sides. Instead, we spotted that we could draw the rectangle as two halves, and used a loop to repeat that pattern twice. But now we might spot that a square is a special kind of rectangle. We already have a function that draws a rectangle, so we can use that to draw our square.
1 2

def draw_square(tx, sz): draw_rectangle(tx, sz, sz)

# a new version of draw_square

There are some points worth noting here: Functions can call other functions. Rewriting draw_square like this captures the relationship that weve spotted between squares and rectangles. A caller of this function might say draw_square(tess, 50). The parameters of this function, tx and sz, are assigned the values of the tess object, and the int 50 respectively. In the body of the function they are just like any other variable. When the call is made to draw_rectangle, the values in variables tx and sz are fetched rst, then the call happens. So as we enter the top of function draw_rectangle, its variable t is assigned the tess object, and w and h in that function are both given the value 50. So far, it may not be clear why it is worth the trouble to create all of these new functions. Actually, there are a lot of reasons, but this example demonstrates two: 1. Creating a new function gives us an opportunity to name a group of statements. Functions can simplify a program by hiding a complex computation behind a single command. The function (including its name) can capture our mental chunking, or abstraction, of the problem. 2. Creating a new function can make a program smaller by eliminating repetitive code. As we might expect, we have to create a function before we can execute it. In other words, the function denition has to be executed before the rst time it is called. 42 Chapter 4. Functions

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

4.3 Flow of execution


In order to ensure that a function is dened before its rst use, we have to know the order in which statements are executed, which is called the ow of execution. Weve already talked about this a little in the previous chapter. Execution always begins at the rst statement of the program. Statements are executed one at a time, in order from top to bottom. Function denitions do not alter the ow of execution of the program, but remember that statements inside the function are not executed until the function is called. Although it is not common, we can dene one function inside another. In this case, the inner denition isnt executed until the outer function is called. Function calls are like a detour in the ow of execution. Instead of going to the next statement, the ow jumps to the rst line of the called function, executes all the statements there, and then comes back to pick up where it left off. That sounds simple enough, until we remember that one function can call another. While in the middle of one function, the program might have to execute the statements in another function. But while executing that new function, the program might have to execute yet another function! Fortunately, Python is adept at keeping track of where it is, so each time a function completes, the program picks up where it left off in the function that called it. When it gets to the end of the program, it terminates. Whats the moral of this sordid tale? When we read a program, dont read from top to bottom. Instead, follow the ow of execution. Watch the ow of execution in action In PyScripter, we can watch the ow of execution by single-stepping through any program. PyScripter will highlight each line of code just before it is about to be executed. PyScripter also lets us hover the mouse over any variable in the program, and it will pop up the current value of that variable. So this makes it easy to inspect the state snapshot of the program the current values that are assigned to the programs variables. This is a powerful mechanism for building a deep and thorough understanding of what is happening at each step of the way. Learn to use the single-stepping feature well, and be mentally proactive: as you work through the code, challenge yourself before each step: What changes will this line make to any variables in the program? and Where will ow of execution go next? Let us go back and see how this works with the program above that draws 15 multicolor squares. First, were going to add one line of magic below the import statement not strictly necessary, but it will make our lives much simpler, because it prevents stepping into the module containing the turtle code.
import turtle __import__(turtle).__traceable__ = False

4.3. Flow of execution

43

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Now were ready to begin. Put the mouse cursor on the line of the program where we create the turtle screen, and press the F4 key. This will run the Python program up to, but not including, the line where we have the cursor. Our program will break now, and provide a highlight on the next line to be executed, something like this:

At this point we can press the F7 key (step into) repeatedly to single step through the code. Observe as we execute lines 10, 11, 12, ... how the turtle window gets created, how its canvas color is changed, how the title gets changed, how the turtle is created on the canvas, and then how the ow of execution gets into the loop, and from there into the function, and into the functions loop, and then repeatedly through the body of that loop. While we do this, we can also hover your mouse over some of the variables in the program, and conrm that their values match our conceptual model of what is happening. After a few loops, when were about to execute line 20 and were starting to get bored, we can use the key F8 to step over the function we are calling. This executes all the statements in the function, but without having to step through each one. We always have the choice to either go for the detail, or to take the high-level view and execute the function as a single chunk. There are some other options, including one that allow us to resume execution without further stepping. Find them under the Run menu of PyScripter.

4.4 Functions that return values


Most functions require arguments, values that control how the function does its job. For example, if we want to nd the absolute value of a number, we have to indicate what the number is. Python has a built-in function for computing the absolute value: 44 Chapter 4. Functions

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

>>> abs(5) 5 >>> abs(-5) 5

In this example, the arguments to the abs function are 5 and -5. Some functions take more than one argument. For example the built-in function pow takes two arguments, the base and the exponent. Inside the function, the values that are passed get assigned to variables called parameters.
>>> pow(2, 3) 8 >>> pow(7, 4) 2401

Another built-in function that takes more than one argument is max.
>>> max(7, 11) 11 >>> max(4, 1, 17, 2, 12) 17 >>> max(3 * 11, 5**3, 512 - 9, 1024**0) 503

max can be sent any number of arguments, separated by commas, and will return the maximum value sent. The arguments can be either simple values or expressions. In the last example, 503 is returned, since it is larger than 33, 125, and 1. Furthermore, functions like range, int, abs all return values that can be used to build more complex expressions. So an important difference between these functions and one like draw_square is that draw_square was not executed because we wanted it to compute a value on the contrary, we wrote draw_square because we wanted it to execute a sequence of steps that caused the turtle to draw. Functions that return values are called fruitful functions in this course. In many other languages, a chunk that doesnt return a value is called a procedure, but we will stick here with the Python way of also calling it a function, or if we want to stress it, a non-fruitful function. (Would the term barren function be easier?) How do we write our own fruitful function? In the exercises at the end of chapter 2 we saw the standard formula for compound interest, which well now write as a fruitful function:

4.4. Functions that return values

45

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2 3 4 5 6 7 8 9 10 11 12 13

def final_amt(p, r, n, t): """ Apply the compound interest formula to p to produce the final amount. """ a = p * (1 + r/n) ** (n*t) return a # This is new, and makes the function fruitful. # now that we have the function above, let us call it. toInvest = float(input("How much do you want to invest?")) fnl = final_amt(toInvest, 0.08, 12, 5) print("At the end of the period youll have R", fnl)

The return statement is followed an expression which is evaluated. Its result is returned to the caller as the fruit of calling this function. We prompted the user for the principal amount. The type of toInvest is a string, but we need a number before we can work with it. Because it is money, and could have decimal places, weve used the float type converter function to parse the string and return a oat. Notice how we entered the arguments for 8% interest, compounded 12 times per year, for 5 years. When we run this, we get the output At the end of the period youll have R 14898.457083 This is a bit messy with all these decimal places, but remember that Python doesnt understand that were working with money: it just does the calculation to the best of its ability, without rounding. Later well see how to format the string that is printed in such a way that it does get nicely rounded to two decimal places before printing. The line toInvest = oat(input(How much do you want to invest?)) also shows yet another example of composition we can call a function like oat, and its arguments can be the results of other function calls (like input) that weve called along the way. Notice something else very important here. The name of the variable we pass as an argument toInvest has nothing to do with the name of the parameter p. It is as if p = toInvest is executed when nal_amt is called. It doesnt matter what the value was named in the caller, in nal_amt its name is p.

46

Chapter 4. Functions

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

These short variable names are getting quite tricky, so perhaps wed prefer one of these versions instead:
1 2 3 4 5 6 7 8 9

def final_amt_v2(principalAmount, nominalPercentageRate, numTimesPerYear, years): a = principalAmount * (1 + nominalPercentageRate / numTimesPerYear) ** (numTimesPerYear*years) return a def final_amt_v3(amt, rate, compounded, years): a = amt * (1 + rate/compounded) ** (componded*years) return a

They all do the same thing. Use your judgement to write code that can be best understood by other humans! Short variable names are more economical and sometimes make code easier to read: E = mc2 would not be nearly so memorable if Einstein had used longer variable names! If you do prefer short names, make sure you also have some comments to enlighten the reader about what the variables are used for.

4.5 Variables and parameters are local


When we create a local variable inside a function, it only exists inside the function, and we cannot use it outside. For example, consider again this function:
1 2 3

def final_amt(p, r, n, t): a = p * (1 + r/n) ** (n*t) return a

If we try to use a, outside the function, well get an error:


>>> a NameError: name a is not defined

The variable a is local to nal_amt, and is not visible outside the function. Additionally, a only exists while the function is being executed we call this its lifetime. When the execution of the function terminates, the local variables are destroyed. Parameters are also local, and act like local variables. For example, the lifetimes of p, r, n, t begin when nal_amt is called, and the lifetime ends when the function completes its execution. So it is not possible for a function to set some local variable to a value, complete its execution, and then when it is called again next time, recover the local variable. Each call of the function creates new local variables, and their lifetimes expire when the function returns to the caller.

4.6 Turtles Revisited


Now that we have fruitful functions, we can focus our attention on reorganizing our code so that it ts more nicely into our mental chunks. This process of rearrangement is called refactoring the code. 4.5. Variables and parameters are local 47

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Two things were always going to want to do when working with turtles is to create the window for the turtle, and to create one or more turtles. We could write some functions to make these tasks easier in future:
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

def make_window(colr, ttle): """ Set up the window with the given background color and title. Returns the new window. """ w = turtle.Screen() w.bgcolor(colr) w.title(ttle) return w

def make_turtle(colr, sz): """ Set up a turtle with the given color and pensize. Returns the new turtle. """ t = turtle.Turtle() t.color(colr) t.pensize(sz) return t

wn = tess alex dave

make_window("lightgreen", "Tess and Alex dancing") = make_turtle("hotpink", 5) = make_turtle("black", 1) = make_turtle("yellow", 2)

The trick about refactoring code is to anticipate which things we are likely to want to change each time we call the function: these should become the parameters, or changeable bits, of the functions we write.

4.7 Glossary
argument A value provided to a function when the function is called. This value is assigned to the corresponding parameter in the function. The argument can be the result of an expression which may involve operators, operands and calls to other fruitful functions. body The second part of a compound statement. The body consists of a sequence of statements all indented the same amount from the beginning of the header. The standard amount of indentation used within the Python community is 4 spaces. compound statement A statement that consists of two parts: 1. header - which begins with a keyword determining the statement type, and ends with a colon.

48

Chapter 4. Functions

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

2. body - containing one or more statements indented the same amount from the header. The syntax of a compound statement looks like this:
keyword expression: statement statement ...

docstring If the rst thing in a function body is a string (or, well see later, in other situations too) that is attached to the function as its __doc__ attribute, and can be used by tools like PyScripter. ow of execution The order in which statements are executed during a program run. frame A box in a stack diagram that represents a function call. It contains the local variables and parameters of the function. function A named sequence of statements that performs some useful operation. Functions may or may not take parameters and may or may not produce a result. function call A statement that executes a function. It consists of the name of the function followed by a list of arguments enclosed in parentheses. function composition Using the output from one function call as the input to another. function denition A statement that creates a new function, specifying its name, parameters, and the statements it executes. fruitful function A function that returns a value when it is called. header line The rst part of a compound statement. A header line begins with a keyword and ends with a colon (:) import statement A statement which permits functions and variables dened in another Python module to be brought into the environment of another script. To use the features of the turtle, we need to rst import the turtle module. lifetime Variables and objects have lifetimes they are created at some point during program execution, and will be destroyed at some time. local variable A variable dened inside a function. A local variable can only be used inside its function. Parameters of a function are also a special kind of local variable. parameter A name used inside a function to refer to the value which was passed to it as an argument. refactor A fancy word to describe reorganizing our program code, usually to make it more understandable. Typically, we have a program that is already working, then we go back to tidy it up. It often involves choosing better variable names, or spotting repeated patterns and moving that code into a function. stack diagram A graphical representation of a stack of functions, their variables, and the values to which they refer.

4.7. Glossary

49

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

traceback A list of the functions that are executing, printed when a runtime error occurs. A traceback is also commonly refered to as a stack trace, since it lists the functions in the order in which they are stored in the runtime stack.

4.8 Exercises
1. Write a non-fruitful function to draw a square. Use it in a program to draw the image shown below. Assume each side is 20 units. (Hint: notice that the turtle has already moved away from the ending point of the last square when the program ends.)

2. Write a program to draw this. Assume the innermost square is 20 units per side, and each successive square is 20 units bigger, per side, than the one inside it.

3. Write a non-fruitful function draw_poly(t, n, sz) which makes a turtle draw a regular polygon. When called with draw_poly(tess, 8, 50), it will draw a shape like this:

4. Draw this pretty pattern.

50

Chapter 4. Functions

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

5. The two spirals in this picture differ only by the turn angle. Draw both.

6. Write a non-fruitful function draw_equitriangle(t, sz) which calls draw_poly from the previous question to have its turtle draw a equilateral triangle. 7. Write a fruitful function sum_to(n) that returns the sum of all integer numbers up to and including n. So sum_to(10) would be 1+2+3...+10 which would return the value 55. 8. Write a function area_of_circle(r) which returns the area of a circle of radius r. 9. Write a non-fruitful function to draw a star, where the length of each side is 100 units. (Hint: You should turn the turtle by 144 degrees at each point.)

10. Extend your program above. Draw ve stars, but between each, pick up the pen, move 4.8. Exercises 51

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

forward by 350 units, turn right by 144, put the pen down, and draw the next star. Youll get something like this:

What would it look like if you didnt pick up the pen?

52

Chapter 4. Functions

CHAPTER

FIVE

CONDITIONALS
Programs get really interesting when we can test conditions and change the program behaviour depending on the outcome of the tests. Thats what this chapter is about.

5.1 Boolean values and expressions


A boolean value is either true or false. It is named after the British mathematician, George Boole, who rst formulated boolean algebra some rules for reasoning about and combining these values. This is the basis of all modern computer logic. In Python, the two boolean values are True and False (the capitalization must be exactly as shown), and the Python type is called bool.
>>> type(True) <class bool> >>> type(true) Traceback (most recent call last): File "<interactive input>", line 1, in <module> NameError: name true is not defined

A boolean expression is an expression that evaluates to produce a result which is a boolean value. For example, the operator == tests if two values are equal. It produces (or yields) a boolean value:
>>> 5 True >>> 5 False >>> j >>> j True == (3 + 2) == 6 = "hel" + "lo" == "hello" # is five equal 5 to the result of 3 + 2?

In the rst statement, the two operands evaluate to equal values, so the expression evaluates to True; in the second statement, 5 is not equal to 6, so we get False. The == operator is one of six common comparison operators which all produce a bool result; here are all six:

53

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

x x x x x x

== y != y > y < y >= y <= y

# # # # # #

produce True if ... x is equal to y x is not equal to y x is greater than y x is less than y x is greater than or equal to y x is less than or equal to y

Although these operations are probably familiar, the Python symbols are different from the mathematical symbols. A common error is to use a single equal sign (=) instead of a double equal sign (==). Remember that = is an assignment operator and == is a comparison operator. Also, there is no such thing as =< or =>. Like any other types weve seen so far, boolean values can be assigned to variables, printed, etc.
>>> oldEnoughToGetDrivingLicence = age >= 17 >>> print(oldEnoughToGetDrivingLicence) True >>> type(oldEnoughToGetDrivingLicence) <class bool>

5.2 Logical operators


There are three logical operators, and, or, and not, that allow us to build more complex boolean expressions from simpler boolean expressions. The semantics (meaning) of these operators is similar to their meaning in English. For example, x > 0 and x < 10 produces True only if x is greater than 0 and at the same time, x is less than 10. n % 2 == 0 or n % 3 == 0 is True if either of the conditions is True, that is, if the number n is divisible by 2 or it is divisible by 3. (What do you think happens if n is divisible by both 2 and by 3 at the same time? Will the expression yield True or False? Try it in your Python interpreter.) Finally, the not operator negates a boolean value, so not(x > y) is True if (x > y) is False, that is, if x is less than or equal to y.

5.3 Truth Tables


A truth table is a small table that allows us to list all the possible inputs, and to give the results for the logical operators. Because the and and or operators each have two operands, there are only four rows in a truth table that describes the semantics of and. a False False True True b False True False True a and b False False False True

54

Chapter 5. Conditionals

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

In a Truth Table, we sometimes use T and F as shorthand for the two boolean values: here is the truth table describing or: a F F T T b F T F T a or b F T T T

The third logical operator, not, only takes a single operand, so its truth table only has two rows: a F T not a T F

5.4 Simplifying Boolean Expressions


An algebra gives us some rules for simplifying and rearranging expressions. For example, we are all familiar in school algebra with rules like:
n * 0 == 0

Here are some boolean algebra rules for working with the and operator:
x and False == False False and x == False y and x == x and y x and True == x True and x == x x and x == x

Here are some corresponding rules for the or operator:


x or False == x False or x == x y or x == x or y x or True == True True or x == True x or x == x

Two not operators cancel each other:


not (not x) == x

5.5 Conditional execution


In order to write useful programs, we almost always need the ability to check conditions and change the behavior of the program accordingly. Conditional statements give us this ability. The simplest form is the if statement: 5.4. Simplifying Boolean Expressions 55

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2 3 4 5 6 7

if x % 2 == 0: print(x, " print("Did else: print(x, " print("Did

is even.") you know that 2 is the only even number that is prime?") is odd.") you know that multiplying two odd numbers " + "always gives an odd result?")

The boolean expression after the if statement is called the condition. If it is true, then all the indented statements get executed. If not, then all the statements indented under the else clause get executed. Flowchart of a if statement with an else

The syntax for an if statement looks like this:


1 2 3 4

if BOOLEAN EXPRESSION: STATEMENTS_1 else: STATEMENTS_2

# executed if condition evaluates to True # executed if condition evaluates to False

As with the function denition from the last chapter and other compound statements like for, the if statement consists of a header line and a body. The header line begins with the keyword if followed by a boolean expression and ends with a colon (:). The indented statements that follow are called a block. The rst unindented statement marks the end of the block. Each of the statements inside the rst block of statements are executed in order if the boolean expression evaluates to True. The entire rst block of statements is skipped if the boolean expression evaluates to False, and instead all the statements under the else clause are executed. 56 Chapter 5. Conditionals

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

There is no limit on the number of statements that can appear under the two clauses of an if statement, but there has to be at least one statement in each block. Occasionally, it is useful to have a section with no statements (usually as a place keeper, or scaffolding, for code we havent written yet). In that case, we can use the pass statement, which does nothing except act as a placeholder.
1 2 3 4

if True: pass else: pass

# This is always true # so this is always executed, but it does nothing

5.6 Omitting the else clause


Flowchart of an if with no else

Another form of the if statement is one in which the else clause is omitted entirely. In this case, when the condition evaluates to True, the statements are executed, otherwise the ow of execution continues to the statement after the if.
1 2 3 4 5 6

if x < 0: print("The negative number ", x, " is not valid here.") x = 42 print("Ive decided to use the number 42 instead.") print("The square root of ", x, "is", math.sqrt(x))

In this case, the print function that outputs the square root is the one after the if not because we left a blank line, but because of the way the code is indented. Note too that the function 5.6. Omitting the else clause 57

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

call math.sqrt(x) will give an error unless we have an import math statement, usually placed near the top of our script. Python terminology Python documentation sometimes uses the term suite of statements to mean what we have called a block here. They mean the same thing, and since most other languages and computer scientists use the word block, well stick with that. Notice too that else is not a statement. The if statement has two clauses, one of which is the (optional) else clause.

5.7 Chained conditionals


Sometimes there are more than two possibilities and we need more than two branches. One way to express a computation like that is a chained conditional:
1 2 3 4 5 6

if x < y: STATEMENTS_A elif x > y: STATEMENTS_B else: STATEMENTS_C

Flowchart of this chained conditional

elif is an abbreviation of else if. Again, exactly one branch will be executed. There is no limit of the number of elif statements but only a single (and optional) nal else statement is allowed and it must be the last branch in the statement:

58

Chapter 5. Conditionals

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2 3 4 5 6 7 8

if choice == a: function_a() elif choice == b: function_b() elif choice == c: function_c() else: print("Invalid choice.")

Each condition is checked in order. If the rst is false, the next is checked, and so on. If one of them is true, the corresponding branch executes, and the statement ends. Even if more than one condition is true, only the rst true branch executes.

5.8 Nested conditionals


One conditional can also be nested within another. (It is the same theme of composibility, again!) We could have written the previous example as follows: Flowchart of this nested conditional

1 2 3 4 5 6 7

if x < y: STATEMENTS_A else: if x > y: STATEMENTS_B else: STATEMENTS_C

5.8. Nested conditionals

59

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

The outer conditional contains two branches. The second branch contains another if statement, which has two branches of its own. Those two branches could contain conditional statements as well. Although the indentation of the statements makes the structure apparent, nested conditionals very quickly become difcult to read. In general, it is a good idea to avoid them when we can. Logical operators often provide a way to simplify nested conditional statements. For example, we can rewrite the following code using a single conditional:
1 2 3

if 0 < x: # assume x is an int here if x < 10: print("x is a positive single digit.")

The print function is called only if we make it past both the conditionals, so instead of the above which uses two if statements each with a simple condition, we could make a more complex condition using the and operator. Now we only need a single if statement:
1 2

if 0 < x and x < 10: print("x is a positive single digit.")

5.9 The return statement


The return statement, with or without a value, depending on whether the function is fruitful or not, allows us to terminate the execution of a function before (or when) we reach the end. One reason to use an early return is if we detect an error condition:
1 2 3 4 5 6 7

def print_square_root(x): if x <= 0: print("Positive numbers only, please.") return result = x**0.5 print("The square root of", x, "is", result)

The function print_square_root has a parameter named x. The rst thing it does is check whether x is less than or equal to 0, in which case it displays an error message and then uses return to exit the function. The ow of execution immediately returns to the caller, and the remaining lines of the function are not executed.

5.10 Logical opposites


Each of the six relational operators has a logical opposite: for example, suppose we can get a driving licence when our age is greater or equal to 17, we can not get the driving licence when we are less than 17. Notice that the opposite of >= is <.

60

Chapter 5. Conditionals

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition operator == != < <= > >= logical opposite != == >= > <= <

Understanding these logical opposites allows us to sometimes get rid of not operators. not operators are often quite difcult to read in computer code, and our intentions will usually be clearer if we can eliminate them. For example, if we wrote this Python:
1 2 3

if not (age >= 17): print("Hey, youre too young to get a driving licence!") return

it would probably be clearer to use the simplication laws, and to write instead:
1 2 3

if age < 17: print("Hey, youre too young to get a driving licence!") return

Two powerful simplication laws (called de Morgans laws) that are often helpful when dealing with complicated boolean expressions are:
not (x and y) not (x or y) == == (not x) or (not y) (not x) and (not y)

For example, suppose we can slay the dragon only if our magic lightsabre sword is charged to 90% or higher, and we have 100 or more energy units in our protective shield. We nd this fragment of Python code in the game:
1 2 3 4

if not ((swordCharge >= 0.90) and (shieldEnergy >= 100)): print("Your attack has no effect, the dragon fries you to a crisp!") else: print("The dragon crumples in a heap. You rescue the gorgeous princess!"

de Morgans laws together with the logical opposites would let us rework the condition in a (perhaps) easier to understand way like this:
1 2 3 4

if (swordCharge < 0.90) or (shieldEnergy < 100): print("Your attack has no effect, the dragon fries you to a crisp!") else: print("The dragon crumples in a heap. You rescue the gorgeous princess!"

We could also get rid of the not by swapping around the then and else parts of the conditional. So here is a third version, also equivalent:
1 2 3 4

if (swordCharge >= 0.90) and (shieldEnergy >= 100): print("The dragon crumples in a heap. You rescue the gorgeous princess!" else: print("Your attack has no effect, the dragon fries you to a crisp!")

5.10. Logical opposites

61

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

This version is probably the best of the three, because it very closely matches the initial English statement. Clarity of our code (for other humans), and making it easy to see that the code does what was expected should always be a high priority. As our programming skills grow well nd we have more than one way to solve any problem. So good programs are designed. We make choices that favour clarity, simplicity, and elegance. The job title software architect says a lot about what we do we are architects who engineer our products to balance beauty, functionality, simplicity and clarity in our creations. Tip: Once our program works, we should play around a bit trying to polish it up. Write good comments. Think about whether the code would be clearer with different variable names. Could we have done it more elegantly? Should we rather use a function? Can we simplify the conditionals? We think of our code as our creation, our work of art! We make it great.

5.11 Type conversion


Weve had a rst look at this in an earlier chapter. Seeing it again wont hurt! Many Python types come with a built-in function that attempts to convert values of another type into its own type. The int(ARGUMENT) function, for example, takes any value and converts it to an integer, if possible, or complains otherwise:
>>> int("32") 32 >>> int("Hello") ValueError: invalid literal for int() with base 10: Hello

int can also convert oating-point values to integers, but remember that it truncates the fractional part:
>>> -2 >>> 3 >>> 42 >>> 1 int(-2.3) int(3.99999) int("42") int(1.0)

The float(ARGUMENT) function converts integers and strings to oating-point numbers:


>>> float(32) 32.0 >>> float("3.14159") 3.14159 >>> float(1) 1.0

62

Chapter 5. Conditionals

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

It may seem odd that Python distinguishes the integer value 1 from the oating-point value 1.0. They may represent the same number, but they belong to different types. The reason is that they are represented differently inside the computer. The str(ARGUMENT) function converts any argument given to it to type string:
>>> str(32) 32 >>> str(3.14149) 3.14149 >>> str(True) True >>> str(true) Traceback (most recent call last): File "<interactive input>", line 1, in <module> NameError: name true is not defined

str(ARGUMENT) will work with any value and convert it into a string. As mentioned earlier, True is boolean value; true is just an ordinary variable name, and is not dened here, so we get an error.

5.12 A Turtle Bar Chart


The turtle has a lot more power than weve seen so far. If you want to see the full documentation, look at http://docs.python.org/library/turtle.html, or within PyScripter, use Help and search for the turtle module. Here are a couple of new tricks for our turtles: We can get a turtle to display text on the canvas at the turtles current position. The method is alex.write("Hello"). We can ll a shape (circle, semicircle, triangle, etc.) with a color. It is a two-step process. First we call the method alex.begin_fill(), then we draw the shape, then we call alex.end_fill(). Weve previously set the color of our turtle - we can now also set its ll color, which need not be the same as the turtle and the pen color. We use alex.color("blue","red") to set the turtle to draw in blue, and ll in red. Ok, so can we get tess to draw a bar chart? Let us start with some data to be charted, xs = [48, 117, 200, 240, 160, 260, 220] Corresponding to each data measurement, well draw a simple rectangle of that height, with a xed width.
1 2 3 4 5

def draw_bar(t, height): """ Get turtle t to draw one bar, of height. """ t.left(90) t.forward(height) # Draw up the left side t.right(90)

5.12. A Turtle Bar Chart

63

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

6 7 8 9 10 11 12 13 14

t.forward(40) t.right(90) t.forward(height) t.left(90) t.forward(10) ... for v in xs: draw_bar(tess, v)

# width of bar, along the top # And down again! # put the turtle facing the way we found it. # leave small gap after each bar

# assume xs and tess are ready

Ok, not fantasically impressive, but it is a nice start! The important thing here was the mental chunking, or how we broke the problem into smaller pieces. Our chunk is to draw one bar, and we wrote a function to do that. Then, for the whole chart, we repeatedly called our function. Next, at the top of each bar, well print the value of the data. Well do this in the body of draw_bar, by adding t.write( + str(height)) as the new third line of the body. Weve put a little space in front of the number, and turned the number into a string. Without this extra space we tend to cramp our text awkwardly against the bar to the left. The result looks a lot better now:

64

Chapter 5. Conditionals

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

And now well add two lines to ll each bar. Our nal program now looks like this:
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

def draw_bar(t, height): """ Get turtle t to draw one bar, of height. """ t.begin_fill() # added this line t.left(90) t.forward(height) t.write( + str(height)) t.right(90) t.forward(40) t.right(90) t.forward(height) t.left(90) t.end_fill() # added this line t.forward(10) wn = turtle.Screen() wn.bgcolor("lightgreen") tess = turtle.Turtle() tess.color("blue", "red") tess.pensize(3) # Set up the window and its attributes

# create tess and set some attributes

xs = [48,117,200,240,160,260,220] for a in xs: draw_bar(tess, a) wn.mainloop()

It produces the following, which is more satisfying:

5.12. A Turtle Bar Chart

65

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Mmm. Perhaps the bars should not be joined to each other at the bottom. Well need to pick up the pen while making the gap between the bars. Well leave that as an exercise for you!

5.13 Glossary
block A group of consecutive statements with the same indentation. body The block of statements in a compound statement that follows the header. boolean algebra Some rules for rearranging and reasoning about boolean expressions. boolean expression An expression that is either true or false. boolean value There are exactly two boolean values: True and False. Boolean values result when a boolean expression is evaluated by the Python interepreter. They have type bool. branch One of the possible paths of the ow of execution determined by conditional execution. chained conditional A conditional branch with more than two possible ows of execution. In Python chained conditionals are written with if ... elif ... else statements. comparison operator One of the six operators that compares two values: ==, !=, >, <, >=, and <=. condition The boolean expression in a conditional statement that determines which branch is executed. conditional statement A statement that controls the ow of execution depending on some condition. In Python the keywords if, elif, and else are used for conditional statements. logical operator One of the operators that combines boolean expressions: and, or, and not. 66 Chapter 5. Conditionals

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

nesting One program structure within another, such as a conditional statement inside a branch of another conditional statement. prompt A visual cue that tells the user to input data. truth table A concise table of boolean values that can describe the semantics of an operator. type conversion An explicit function call that takes a value of one type and computes a corresponding value of another type. wrapping code in a function The process of adding a function header and parameters to a sequence of program statements is often refered to as wrapping the code in a function. This process is very useful whenever the program statements in question are going to be used multiple times. It is even more useful when it allows the programmer to express their mental chunking, and how theyve broken a complex problem into pieces.

5.14 Exercises
1. Assume the days of the week are numbered 0,1,2,3,4,5,6 from Sunday to Saturday. Write a function which is given the day number, and it returns the day name (a string). 2. You go on a wonderful holiday (perhaps to jail, if you dont like happy exercises) leaving on day number 3 (a Wednesday). You return home after 137 sleeps. Write a general version of the program which asks for the starting day number, and the length of your stay, and it will tell you the name of day of the week you will return on. 3. Give the logical opposites of these conditions (a) a > b (b) a >= b (c) a >= 18 and day == 3 (d) a >= 18 and day != 3 4. What do these expressions evaluate to? (a) 3 == 3 (b) 3 != 3 (c) 3 >= 4 (d) not (3 < 4) 5. Complete this truth table:

5.14. Exercises

67

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition p F F F F T T T T q F F T T F F T T r F T F T F T F T (not (p and q)) or r ? ? ? ? ? ? ? ?

6. Write a function which is given an exam mark, and it returns a string the grade for that mark according to this scheme: Mark >= 75 [70-75) [60-70) [50-60) [45-50) [40-45) < 40 Grade First Upper Second Second Third F1 Supp F2 F3

The square and round brackets denote closed and open intervals. A closed interval includes the number, and open interval excludes it. So 39.99999 gets grade F3, but 40 gets grade F2. Assume
xs = [83, 75, 74.9, 70, 69.9, 65, 60, 59.9, 55, 50, 49.9, 45, 44.9, 40, 39.9, 2, 0]

Test your function by printing the mark and the grade for all the elements in this list. 7. Modify the turtle bar chart program so that the pen is up for the small gaps between each bar. 8. Modify the turtle bar chart program so that the bar for any value of 200 or more is lled with red, values between [100 and 200) are lled with yellow, and bars representing values less than 100 are lled with green. 9. In the turtle bar chart program, what do you expect to happen if one or more of the data values in the list is negative? Try it out. Change the program so that when it prints the text value for the negative bars, it puts the text below the bottom of the bar. 10. Write a function find_hypot which, given the length of two sides of a right-angled triangle, returns the length of the hypotenuse. (Hint: x ** 0.5 will return the square root.) 11. Write a function is_rightangled which, given the length of three sides of a triangle, will determine whether the triangle is right-angled. Assume that the third argument to the function is always the longest side. It will return True if the triangle is right-angled, or False otherwise. Hint: oating point arithmetic is not always exactly accurate, so it is not safe to test oating point numbers for equality. If a good programmer wants to know whether x is

68

Chapter 5. Conditionals

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

equal or close enough to y, they would probably code it up as


if abs(x-y) < 0.000001: ... # if x is approximately equal to y

12. Extend the above program so that the sides can be given to the function in any order.

5.14. Exercises

69

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

70

Chapter 5. Conditionals

CHAPTER

SIX

FRUITFUL FUNCTIONS
6.1 Return values
The built-in functions we have used, such as abs, pow, int, max, and range, have produced results. Calling each of these functions generates a value, which we usually assign to a variable or use as part of an expression.
1 2

biggest = max(3, 7, 2, 5) x = abs(3 - 11) + 10

We also wrote our own function to return the nal amount for a compound interest calculation. In this chapter, we are going to write more functions that return values, which we will call fruitful functions, for want of a better name. The rst example is area, which returns the area of a circle with the given radius:
1 2 3

def area(radius): b = 3.14159 * radius**2 return b

We have seen the return statement before, but in a fruitful function the return statement includes a return value. This statement means: evaluate the return expression, and then return it immediately as the result (the fruit) of this function. The expression provided can be arbitrarily complicated, so we could have written this function like this:
1 2

def area(radius): return 3.14159 * radius * radius

On the other hand, temporary variables like b above often make debugging easier. Sometimes it is useful to have multiple return statements, one in each branch of a conditional. We have already seen the built-in abs, now we see how to write our own:
1 2 3 4 5

def absolute_value(x): if x < 0: return -x else: return x

71

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Another way to write the above function is to leave out the else and just follow the if condition by the second return statement.
1 2 3 4

def absolute_value(x): if x < 0: return -x return x

Think about this version and convince yourself it works the same as the rst one. Code that appears after a return statement, or any other place the ow of execution can never reach, is called dead code, or unreachable code. In a fruitful function, it is a good idea to ensure that every possible path through the program hits a return statement. The following version of absolute_value fails to do this:
1 2 3 4 5

def bad_absolute_value(x): if x < 0: return -x elif x > 0: return x

This version is not correct because if x happens to be 0, neither condition is true, and the function ends without hitting a return statement. In this case, the return value is a special value called None:
>>> print(bad_absolute_value(0)) None

All Python functions return None whenever they do not return another value. It is also possible to use a return statement in the middle of a for loop, in which case control immediately returns from the function. Let us assume that we want a function which looks through a list of words. It should return the rst 2-letter word. If there is not one, it should return the empty string:
1 2 3 4 5

def find_first_2_letter_word(xs): for wd in xs: if len(wd) == 2: return wd return >>> find_first_2_letter_word([This, is, a, dead, parrot]) is >>> find_first_2_letter_word(["I", "like", "cheese"])

Single-step through this code and convince yourself that in the rst test case that weve provided, the function returns while processing the second element in the list: it does not have to traverse the whole list.

72

Chapter 6. Fruitful functions

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

6.2 Program development


At this point, you should be able to look at complete functions and tell what they do. Also, if you have been doing the exercises, you have written some small functions. As you write larger functions, you might start to have more difculty, especially with runtime and semantic errors. To deal with increasingly complex programs, we are going to suggest a technique called incremental development. The goal of incremental development is to avoid long debugging sessions by adding and testing only a small amount of code at a time. As an example, suppose you want to nd the distance between two points, given by the coordinates (x1 , y1 ) and (x2 , y2 ). By the Pythagorean theorem, the distance is:

The rst step is to consider what a distance function should look like in Python. In other words, what are the inputs (parameters) and what is the output (return value)? In this case, the two points are the inputs, which we can represent using four parameters. The return value is the distance, which is a oating-point value. Already we can write an outline of the function that captures our thinking so far:
1 2

def distance(x1, y1, x2, y2): return 0.0

Obviously, this version of the function doesnt compute distances; it always returns zero. But it is syntactically correct, and it will run, which means that we can test it before we make it more complicated. To test the new function, we call it with sample values:
>>> distance(1, 2, 4, 6) 0.0

We chose these values so that the horizontal distance equals 3 and the vertical distance equals 4; that way, the result is 5 (the hypotenuse of a 3-4-5 triangle). When testing a function, it is useful to know the right answer. At this point we have conrmed that the function is syntactically correct, and we can start adding lines of code. After each incremental change, we test the function again. If an error occurs at any point, we know where it must be in the last line we added. A logical rst step in the computation is to nd the differences x2 - x1 and y2 - y1 . We will store those values in temporary variables named dx and dy.
1 2 3 4

def distance(x1, y1, x2, y2): dx = x2 - x1 dy = y2 - y1 return 0.0

If we call the function with the arguments shown above, when the ow of execution gets to the return statement, dx should be 3 and dy should be 4. We can check that this is the case

6.2. Program development

73

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

in PyScripter by putting the cursor on the return statement, and running the program to break execution when it gets to the cursor (using the F4 key). Then we inspect the variables dx and dy by hovering the mouse above them, to conrm that the function is getting the right parameters and performing the rst computation correctly. If not, there are only a few lines to check. Next we compute the sum of squares of dx and dy:
1 2 3 4 5

def distance(x1, y1, x2, y2): dx = x2 - x1 dy = y2 - y1 dsquared = dx*dx + dy*dy return 0.0

Again, we could run the program at this stage and check the value of dsquared (which should be 25). Finally, using the fractional exponent 0.5 to nd the square root, we compute and return the result:
1 2 3 4 5 6

def distance(x1, y1, x2, y2): dx = x2 - x1 dy = y2 - y1 dsquared = dx*dx + dy*dy result = dsquared**0.5 return result

If that works correctly, you are done. Otherwise, you might want to inspect the value of result before the return statement. When you start out, you might add only a line or two of code at a time. As you gain more experience, you might nd yourself writing and debugging bigger conceptual chunks. Either way, stepping through your code one line at a time and veryifying that each step matches your expectations can save you a lot of debugging time. As you improve your programming skills you should nd yourself managing bigger and bigger chunks: this is very similar to the way we learnt to read letters, syllables, words, phrases, sentences, paragraphs, etc., or the way we learn to chunk music from indvidual notes to chords, bars, phrases, and so on. The key aspects of the process are: 1. Start with a working skeleton program and make small incremental changes. At any point, if there is an error, you will know exactly where it is. 2. Use temporary variables to hold intermediate values so that you can easily inspect and check them. 3. Once the program is working, relax, sit back, and play around with your options. (There is interesting research that links playfulness to better understanding, better learning, more enjoyment, and a more positive mindset about what you can achieve so spend some time ddling around!) You might want to consolidate multiple statements into one bigger compound expression, or rename the variables youve used, or see if you can make the function shorter. A good guideline is to aim for making code as easy as possible for others to read.

74

Chapter 6. Fruitful functions

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Here is another version of the function. It makes use of a square root function that is in the math module (well learn about modules shortly). Which do you prefer? Which looks closer to the Pythagorean formula we started out with?
1 2 3 4

import math def distance(x1, y1, x2, y2): return math.sqrt( (x2-x1)**2 + (y2-y1)**2 ) >>> distance(1, 2, 4, 6) 5.0

6.3 Debugging with print


Another powerful technique for debugging (an alternative to single-stepping and inspection of program variables), is to insert extra print functions in carefully selected places in your code. Then, by inspecting the output of the program, you can check whether the algorithm is doing what you expect it to. Be clear about the following, however: You must have a clear solution to the problem, and must know what should happen before you can debug a program. Work on solving the problem on a piece of paper (perhaps using a owchart to record the steps you take) before you concern yourself with writing code. Writing a program doesnt solve the problem it simply automates the manual steps you would take. So rst make sure you have a pen-and-paper manual solution that works. Programming then is about making those manual steps happen automatically. Do not write chatterbox functions. A chatterbox is a fruitful function that, in addition to its primary task, also asks the user for input, or prints output, when it would be more useful if it simply shut up and did its work quietly. For example, weve seen built-in functions like range, max and abs. None of these would be useful building blocks for other programs if they prompted the user for input, or printed their results while they performed their tasks. So a good tip is to avoid calling print and input functions inside fruitful functions, unless the primary purpose of your function is to perform input and output. The one exception to this rule might be to temporarily sprinkle some calls to print into your code to help debug and understand what is happening when the code runs, but these will then be removed once you get things working.

6.4 Composition
As you should expect by now, you can call one function from within another. This ability is called composition. As an example, well write a function that takes two points, the center of the circle and a point on the perimeter, and computes the area of the circle.

6.3. Debugging with print

75

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Assume that the center point is stored in the variables xc and yc, and the perimeter point is in xp and yp. The rst step is to nd the radius of the circle, which is the distance between the two points. Fortunately, weve just written a function, distance, that does just that, so now all we have to do is use it:
1

radius = distance(xc, yc, xp, yp)

The second step is to nd the area of a circle with that radius and return it. Again we will use one of our earlier functions:
1 2

result = area(radius) return result

Wrapping that up in a function, we get:


1 2 3 4

def area2(xc, yc, xp, yp): radius = distance(xc, yc, xp, yp) result = area(radius) return result

We called this function area2 to distinguish it from the area function dened earlier. There can only be one function with a given name within a module. The temporary variables radius and result are useful for development, debugging, and single-stepping through the code to inspect what is happening, but once the program is working, we can make it more concise by composing the function calls:
1 2

def area2(xc, yc, xp, yp): return area(distance(xc, yc, xp, yp))

6.5 Boolean functions


Functions can return boolean values, which is often convenient for hiding complicated tests inside functions. For example:
1 2 3 4 5

def is_divisible(x, y): if x % y == 0: return True else: return False

The name of this function is is_divisible. It is common to give boolean functions names that sound like yes/no questions. is_divisible returns either True or False to indicate whether the x is or is not divisible by y. We can make the function more concise by taking advantage of the fact that the condition of the if statement is itself a boolean expression. We can return it directly, avoiding the if statement altogether:
1 2

def is_divisible(x, y): return x % y == 0

76

Chapter 6. Fruitful functions

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

This session shows the new function in action:


>>> is_divisible(6, 4) False >>> is_divisible(6, 3) True

Boolean functions are often used in conditional statements:


1 2 3 4

if is_divisible(x, y): ... # do something ... else: ... # do something else ...

It might be tempting to write something like:


1

if is_divisible(x, y) == True:

but the extra comparison is unnecessary.

6.6 Programming with style


Readability is very important to programmers, since in practice programs are read and modied far more often then they are written. All the code examples in this book will be consistent with the Python Enhancement Proposal 8 (PEP 8), a style guide developed by the Python community. Well have more to say about style as our programs become more complex, but a few pointers will be helpful already: use 4 spaces for indentation imports should go at the top of the le separate function denitions with two blank lines keep function denitions together keep top level statements, including function calls, together at the bottom of the program

6.7 Unit testing


It is a common best practice in software development these days to include automatic unit testing of source code. Unit testing provides a way to automatically verify that individual pieces of code, such as functions, are working properly. This makes it possible to change the implementation of a function at a later time and quickly test that it still does what it was intended to do. Unit testing also forces the programmer to think about the different cases that the function needs to handle. You also only have to type the tests once into the script, rather than having to keep entering the same test data over and over as you develop your code. 6.6. Programming with style 77

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Extra code in your program which is there because it makes debugging or testing easier is called scaffolding. A collection of tests for some code is called its test suite. There are a few different preferred ways to do unit testing in Python but at this stage were going to ignore what the Python community usually does, and were going to start with two functions that well write ourselves. Well use these for writing our unit tests. Lets start with the absolute_value function that we wrote earlier in this chapter. Recall that we wrote a few different versions, the last of which was incorrect, and had a bug. Would tests have caught this bug? First we plan our tests. Wed like to know if the function returns the correct value when its argument is negative, or when its argument is positive, or when its argument is zero. When planning your tests, youll always want to think carefully about the edge cases here, an argument of 0 to absolute_value is on the edge of where the function behaviour changes, and as we saw at the beginning of the chapter, it is an easy spot for the programmer to make a mistake! So it is a good case to include in our test suite. Were going to write a helper function for checking the results of one test. It takes two arguments the actual value that was returned from the computation, and the value we expected to get. It compares these, and will either print a message telling us that the test passed, or it will print a message to inform us that the test failed. The rst two lines of the body (after the functions docstring) can be copied to your own code as they are here: they import a module called sys, and extract the callers line number from the stack frame. This allows us to print the line number of the test, which will help when we want to x any tests that fail.
1 2 3 4 5 6 7 8 9 10 11 12

def test(actual, expected): """ Compare the actual to the expected value, and print a suitable message. """ import sys linenum = sys._getframe(1).f_lineno # get the callers line number. if (expected == actual): msg = "Test on line {0} passed.".format(linenum) else: msg = ("Test on line {0} failed. Expected {1}, but got {2}." . format(linenum, expected, actual)) print(msg)

There is also some slightly tricky string formatting using the format method which we will gloss over for the moment, and cover in detail in a future chapter. But with this function written, we can proceed to construct our test suite:
1 2 3 4 5 6 7 8

def test_suite(): """ Run the suite of tests for code in this module (this file). """ test(absolute_value(17), 17) test(absolute_value(-17), 17) test(absolute_value(0), 0) test(absolute_value(3.14), 3.14) test(absolute_value(-3.14), 3.14)

78

Chapter 6. Fruitful functions

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

9 10

test_suite()

# and here is the call to run the tests

Here youll see that weve constructed ve tests in our test suite. We could run this against the rst or second versions (the correct versions) of absolute_value, and wed get output similar to the following:
Test Test Test Test Test on on on on on line line line line line 24 25 26 27 28 passed. passed. passed. passed. passed.

But lets say you change the function to an incorrect version like this:
1 2 3 4 5 6

def absolute_value(n): # Buggy version """ Compute the absolute value of n """ if n < 0: return 1 elif n > 0: return n

Can you nd at least two mistakes in this code? Running our test suite we get:
Test Test Test Test Test on on on on on line line line line line 24 25 26 27 28 passed. failed. Expected 17, but got 1. failed. Expected 0, but got None. passed. failed. Expected 3.14, but got 1.

These are three examples of failing tests.

6.8 Glossary
boolean function A function that returns a boolean value. The only possible values of the bool type are False and True. chatterbox function A function which interacts with the user (using input or print) when it should not. Silent functions that just convert their input arguments into their output results are usually the most useful ones. composition (of functions) Calling one function from within the body of another, or using the return value of one function as an argument to the call of another. dead code Part of a program that can never be executed, often because it appears after a return statement. fruitful function A function that yields a return value instead of None. incremental development A program development plan intended to simplify debugging by adding and testing only a small amount of code at a time.

6.8. Glossary

79

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

None A special Python value. One use in Python is that it is returned by functions that do not execute a return statement with a return argument. return value The value provided as the result of a function call. scaffolding Code that is used during program development to assist with development and debugging. The unit test code that we added in this chapter are examples of scaffolding. temporary variable A variable used to store an intermediate value in a complex calculation. test suite A collection of tests for some code you have written. unit testing An automatic procedure used to validate that individual units of code are working properly.

6.9 Exercises
All of the exercises below should be added to a single le. In that le, you should also add the test and test_suite scaffolding functions shown above, and then, as you work through the exercises, add the new tests to your test suite. (If you open the online version of the textbook, you can easily cut and paste the tests and the fragments of code into your Python editor.) After completing each exercise, conrm that all the tests pass. 1. The four compass points can be abbreviated by single-letter strings as N, E, S, and W. Write a function turn_clockwise that takes one of these four compass points as its parameter, and returns the next compass point in the clockwise direction. Here are some tests that should pass:
test(turn_clockwise("N"), "E") test(turn_clockwise("W"), "N")

You might ask What if the argument to the function is some other value? For all other cases, the function should return the value None:
test(turn_clockwise(42), None) test(turn_clockwise("rubbish"), None)

2. Write a function day_name that converts an integer number 0 to 6 into the name of a day. Assume day 0 is Sunday. Once again, return None if the arguments to the function are not valid. Here are some tests that should pass:
test(day_name(3), "Wednesday") test(day_name(6), "Saturday") test(day_name(42), None)

3. Write the inverse function day_num which is given a day name, and returns its number:
test(day_num("Friday"), 5) test(day_num("Sunday"), 0) test(day_num(day_name(3)), 3) test(day_name(day_num("Thursday")), "Thursday")

80

Chapter 6. Fruitful functions

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Once again, if this function is given an invalid argument, it should return None:
test(day_num("Halloween"), None);

4. Write a function that helps answer questions like Today is Wednesday. I leave on holiday in 19 days time. What day will that be? So the function must take a day name and a delta argument the number of days to add and should return the resulting day name:
test(day_add("Monday", 4), "Friday") test(day_add("Tuesday", 0), "Tuesday") test(day_add("Tuesday", 14), "Tuesday") test(day_add("Sunday", 100), "Tuesday")

Hint: use the rst two functions written above to help you write this one. 5. Can your day_add function already work with negative deltas? For example, -1 would be yesterday, or -7 would be a week ago:
test(day_add("Sunday", -1), "Saturday") test(day_add("Sunday", -7), "Sunday") test(day_add("Tuesday", -100), "Sunday")

If your function already works, explain why. If it does not work, make it work. Hint: Play with some cases of using the modulus function % (introduced at the beginning of the previous chapter). Specically, explore what happens to x % 7 when x is negative. 6. Write a function days_in_month which takes the name of a month, and returns the number of days in the month. Ignore leap years:
test(days_in_month("February"), 28) test(days_in_month("December"), 31)

If the function is given invalid arguments, it should return None. 7. Write a function to_secs that converts hours, minutes and seconds to a total number of seconds. Here are some tests that should pass:
test(to_secs(2, test(to_secs(2, test(to_secs(0, test(to_secs(0, test(to_secs(0, 30, 10), 9010) 0, 0), 7200) 2, 0), 120) 0, 42), 42) -10, 10), -590)

8. Extend to_secs so that it can cope with real values as inputs. It should always return an integer number of seconds (truncated towards zero):
test(to_secs(2.5, 0, 10.71), 9010) test(to_secs(2.433,0,0), 8758)

9. Write three functions that are the inverses of to_secs:

6.9. Exercises

81

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

(a) hours_in returns the whole integer number of hours represented by a total number of seconds. (b) minutes_in returns the whole integer number of left over minutes in a total number of seconds, once the hours have been taken out. (c) seconds_in returns the left over seconds represented by a total number of seconds. You may assume that the total number of seconds passed to these functions is an integer. Here are some test cases:
test(hours_in(9010), 2) test(minutes_in(9010), 30) test(seconds_in(9010), 10)

10. Which of these tests fail? Explain why.


test(3 % 4, 0) test(3 % 4, 3) test(3 / 4, 0) test(3 // 4, 0) test(3+4 * 2, 14) test(4-2+2, 0) test(len("hello, world!"), 13)

11. Write a compare function that returns 1 if a > b, 0 if a == b, and -1 if a < b


test(compare(5, 4), 1) test(compare(7, 7), 0) test(compare(2, 3), -1) test(compare(42, 1), 1)

12. Write a function called hypotenuse that returns the length of the hypotenuse of a right triangle given the lengths of the two legs as parameters:
test(hypotenuse(3, 4), 5.0) test(hypotenuse(12, 5), 13.0) test(hypotenuse(24, 7), 25.0) test(hypotenuse(9, 12), 15.0)

13. Write a function slope(x1, y1, x2, y2) that returns the slope of the line through the points (x1, y1) and (x2, y2). Be sure your implementation of slope can pass the following tests:
test(slope(5, test(slope(1, test(slope(1, test(slope(2, 3, 2, 2, 4, 4, 3, 3, 1, 2), 2), 3), 2), 1.0) 0.0) 0.5) 2.0)

Then use a call to slope in a new function named intercept(x1, y1, x2, y2) that returns the y-intercept of the line through the points (x1, y1) and (x2, y2)

82

Chapter 6. Fruitful functions

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

test(intercept(1, 6, 3, 12), 3.0) test(intercept(6, 1, 1, 6), 7.0) test(intercept(4, 6, 12, 8), 5.0)

14. Write a function called is_even(n) that takes an integer as an argument and returns True if the argument is an even number and False if it is odd. Add your own tests to the test suite. 15. Now write the function is_odd(n) that returns True when n is odd and False otherwise. Include unit tests for this function too. Finally, modify it so that it uses a call to is_even to determine if its argument is an odd integer, and ensure that its test still pass. 16. Write a function is_factor(f, n) that passes these tests:
test(is_factor(3, 12), True) test(is_factor(5, 12), False) test(is_factor(7, 14), True) test(is_factor(7, 15), False) test(is_factor(1, 15), True) test(is_factor(15, 15), True) test(is_factor(25, 15), False)

An important role of unit tests is that they can also act as unambiguous specications of what is expected. These test cases answer the question Do we treat 1 and 15 as factors of 15? 17. Write is_multiple to satisfy these unit tests:
test(is_multiple(12, test(is_multiple(12, test(is_multiple(12, test(is_multiple(12, test(is_multiple(12, 3), 4), 5), 6), 7), True) True) False) True) False)

Can you nd a way to use is_factor in your denition of is_multiple? 18. Write the function f2c(t) designed to return the integer value of the nearest degree Celsius for given tempurature in Fahrenheit. (hint: you may want to make use of the built-in function, round. Try printing round.__doc__ in a Python shell or looking up help for the round function, and experimenting with it until you are comfortable with how it works.)
test(f2c(212), 100) test(f2c(32), 0) test(f2c(-40), -40) test(f2c(36), 2) test(f2c(37), 3) test(f2c(38), 3) test(f2c(39), 4) # boiling point of water # freezing point of water # Wow, what an interesting case!

19. Now do the opposite: write the function c2f which converts Celcius to Fahrenheit: 6.9. Exercises 83

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

test(c2f(0), 32) test(c2f(100), 212) test(c2f(-40), -40) test(c2f(12), 54) test(c2f(18), 64) test(c2f(-48), -54)

84

Chapter 6. Fruitful functions

CHAPTER

SEVEN

ITERATION
Computers are often used to automate repetitive tasks. Repeating identical or similar tasks without making errors is something that computers do well and people do poorly. Repeated execution of a set of statements is called iteration. Because iteration is so common, Python provides several language features to make it easier. Weve already seen the for statement in chapter 3. This the the form of iteration youll likely be using most often. But in this chapter weve going to look at the while statement another way to have your program do iteration, useful in slightly different circumstances. Before we do that, lets just review a few ideas...

7.1 Reassignment
As we have mentioned previously, it is legal to make more than one assignment to the same variable. A new assignment makes an existing variable refer to a new value (and stop referring to the old value).
1 2 3 4

bruce = 5 print(bruce) bruce = 7 print(bruce)

The output of this program is:


5 7

because the rst time bruce is printed, its value is 5, and the second time, its value is 7. Here is what reassignment looks like in a state snapshot:

With reassignment it is especially important to distinguish between an assignment statement and a boolean expression that tests for equality. Because Python uses the equal token (=)

85

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

for assignment, it is tempting to interpret a statement like a = b as a boolean test. Unlike mathematics, it is not! Remember that the Python token for the equality operator is ==. Note too that an equality test is symmetric, but assignment is not. For example, if a == 7 then 7 == a. But in Python, the statement a = 7 is legal and 7 = a is not. Furthermore, in mathematics, a statement of equality is always true. If a == b now, then a will always equal b. In Python, an assignment statement can make two variables equal, but because of the possibility of reassignment, they dont have to stay that way:
1 2 3

a = 5 b = a a = 3

# after executing this line, a and b are now equal # after executing this line, a and b are no longer equal

The third line changes the value of a but does not change the value of b, so they are no longer equal. (In some programming languages, a different symbol is used for assignment, such as <- or :=, to avoid confusion. Python chose to use the tokens = for assignment, and == for equality. This is a popular choice also found in languages like C, C++, Java, and C#.)

7.2 Updating variables


When an assignment statement is executed, the right-hand-side expression (i.e. the expression that comes after the assignment token) is evaluated rst. Then the result of that evaluation is written into the variable on the left hand side, thereby changing it. One of the most common forms of reassignment is an update, where the new value of the variable depends on its old value.
1 2

n = 5 n = 3*n + 1

Line 2 means get the current value of n, multiply it by three and add one, and put the answer back into n as its new value. So after executing the two lines above, n will have the value 16. If you try to get the value of a variable that doesnt exist yet, youll get an error:
>>> w = x + 1 Traceback (most recent call last): File "<interactive input>", line 1, in NameError: name x is not defined

Before you can update a variable, you have to initialize it, usually with a simple assignment:
>>> x = 0 >>> x = x + 1

This second statement updating a variable by adding 1 to it is very common. It is called an increment of the variable; subtracting 1 is called a decrement. Sometimes programmers also talk about bumping a variable, which means the same as incrementing it by 1.

86

Chapter 7. Iteration

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

7.3 The for loop revisited


Recall that the for loop processes each item in a list. Each item in turn is (re-)assigned to the loop variable, and the body of the loop is executed. We saw this example in an earlier chapter:
1 2 3

for f in ["Joe", "Amy", "Brad", "Angelina", "Zuki", "Thandi", "Paris"]: invitation = "Hi " + f + ". Please come to my party on Saturday!" print(invitation)

Running through all the items in a list is called traversing the list, or traversal. Let us write a function now to sum up all the elements in a list of numbers. Do this by hand rst, and try to isolate exactly what steps you take. Youll nd you need to keep some running total of the sum so far, either on a piece of paper, or in your head. Remembering things from one step to the next is precisely why we have variables in a program: so well need some variable to remember the running total. It should be initialized with a value of zero, and then we need to traverse the items in the list. For each item, well want to update the running total by adding the next number to it.
1 2 3 4 5 6 7 8 9 10 11 12 13

def mysum(xs): """ Sum all the numbers in the list xs, and return the total. """ running_total = 0 for x in xs: running_total = running_total + x return running_total #add tests like these to your test suite ... test(mysum([1, 2, 3, 4]), 10) test(mysum([1.25, 2.5, 1.75]), 5.5) test(mysum([1, -2, 3]), 2) test(mysum([ ]), 0) test(mysum(range(11)), 55) # 11 is not in the list.

7.4 The while statement


Here is a fragment of code that demonstrates the use of the while statement:
1 2 3 4 5 6 7 8 9 10 11 12

def sum_to(n): """ Return the sum of 1+2+3 ... n """ ss = 0 v = 1 while v <= n: ss = ss + v v = v + 1 return ss # for your test suite test(sum_to(4), 10) test(sum_to(1000), 500500)

7.3. The for loop revisited

87

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

You can almost read the while statement as if it were English. It means, While v is less than or equal to n, continue executing the body of the loop. Within the body, each time, increment v. When v passes n, return your accumulated sum. More formally, here is precise ow of execution for a while statement: 1. Evaluate the condition, yielding False or True. 2. If the condition is false, exit the while statement and continue execution at the next statement. 3. If the condition is true, execute each of the statements in the body and then go back to step 1. The body consists of all of the statements below the header with the same indentation. This type of ow is called a loop because the third step loops back around to the top. Notice that if the condition is false the rst time through the loop, the statements inside the loop are never executed. The body of the loop should change the value of one or more variables so that eventually the condition becomes false and the loop terminates. Otherwise the loop will repeat forever, which is called an innite loop. An endless source of amusement for computer scientists is the observation that the directions on shampoo, lather, rinse, repeat, are an innite loop. In the case here, we can prove that the loop terminates because we know that the value of n is nite, and we can see that the value of v increments each time through the loop, so eventually it will have to exceed n. In other cases, it is not so easy to tell. What you will notice here is that the while loop is more work for you the programmer than the equivalent for loop. When using a while loop one has to control the loop variable yourself: give it an initial value, test for completion, and then make sure you change something in the body so that the loop terminates. By comparison, here is an alternative function that uses for instead:
1 2 3 4 5 6

def sum_to(n): """ Return the sum of 1+2+3 ... n """ ss = 0 for v in range(n+1): ss = ss + v return ss

Notice the slightly tricky call to the range function we had to add one onto n, because range generates its list up to but not including the value you give it. It would be easy to make a programming mistake and overlook this, but because weve made the investment of writing some unit tests, our test suite would have caught our error. So why have two kinds of loop if for looks easier? This next example shows a case where we need the extra power that we get from the while loop.

88

Chapter 7. Iteration

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

7.5 The 3n + 1 sequence


Lets look at a simple sequence that has fascinated and foxed mathematicians for many years. They still cannot answer even quite simple questions about this. The computational rule for creating the sequence is to start from some given n, and to generate the next term of the sequence from n, either by halving n, (whenever n is even), or else by multiplying it by three and adding 1. The sequence terminates when n reaches 1. This Python function captures that algorithm:
1 2 3 4 5 6 7 8 9 10 11

def seq3np1(n): """ Print the 3n+1 sequence from n, terminating when it reaches 1. """ while n != 1: print(n, end=, ) if n % 2 == 0: # n is even n = n // 2 else: # n is odd n = n * 3 + 1 print(n, end=.\n)

Notice rst that the print function has an extra argument end=, . This tells the print function to follow the printed string with whatever the programmer chooses (in this case, a comma followed by a space), instead of ending the line. So each time something is printed in the loop, it is printed on the same line, with the output separated by commas. The call to print(n, end=.\n) when the loop terminates will then print the nal value of n followed by a period and a newline character. (Youll cover the \n (newline character) in the next chapter). The condition for this loop is n != 1, so the loop will continue running until n is 1, which will make the condition false. Each time through the loop, the program outputs the value of n and then checks whether it is even or odd. If it is even, the value of n is divided by 2 using integer division. If it is odd, the value is replaced by n * 3 + 1. Here are some examples:
>>> seq3np1(3) 3, 10, 5, 16, 8, 4, 2, 1. >>> seq3np1(19) 19, 58, 29, 88, 44, 22, 11, 34, 17, 52, 26, 13, 40, 20, 10, 5, 16, 8, 4, 2, 1. >>> seq3np1(21) 21, 64, 32, 16, 8, 4, 2, 1. >>> seq3np1(16) 16, 8, 4, 2, 1. >>>

Since n sometimes increases and sometimes decreases, there is no obvious proof that n will ever reach 1, or that the program terminates. For some particular values of n, we can prove termination. For example, if the starting value is a power of two, then the value of n will be 7.5. The 3n + 1 sequence 89

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

even each time through the loop until it reaches 1. The previous example ends with such a sequence, starting with 16. You might like to have some fun and see if you can nd a small starting number that needs more than a hundred steps before it terminates. Particular values aside, the interesting question was rst posed by a German mathematician called Lothar Collatz: the Collatz conjecture (also known as the 3n + 1 conjecture, is that this sequence terminates for all positive values of n. So far, no one has been able to prove it or disprove it! Think carefully about what would be needed for a proof or disproof of the conjecture All positive integers will eventually converge to 1. With fast computers we have been able to test every integer up to very large values, and so far, they all eventually end up at 1. But this doesnt mean that there might not be some as-yet untested number which does not reduce to 1. Youll notice that if you dont stop when you reach one the sequence gets into its own loop: 1, 4, 2, 1, 4, 2, 1, 4 ... So another possibility is that there might be other cycles that we just havent found. Wikipedia has an informative article about the Collatz conjecture. The sequence also goes under a number of other names (Hailstone sequence, Wonderous numbers, etc.), and youll nd out just how many integers have already been tested by computer, and found to always converge! Choosing between for and while Use a for loop if you know, before you start looping, the maximum number of times that youll need to execute the body. For example, if youre traversing a list of elements, you know that the maximum number of loop iterations you can possibly need is all the elements in the list. Or if you need to print the 12 times table, we know right away how many times the loop will need to run. So any problem like iterate this weather model for 1000 cycles, or search this list of words, nd all prime numbers up to 10000 suggest that a for loop is best. By contrast, if you are required to repeat some computation until some condition is met, and you cannot calculate in advance when this will happen, as we did in this 3n + 1 problem, youll need a while loop. We call the rst case denite iteration we have some denite bounds for what is needed. The latter case is called indenite iteration were not sure how many iterations well need we cannot even establish an upper bound!

7.6 Tracing a program


To write effective computer programs a programmer needs to develop the ability to trace the execution of a computer program. Tracing involves becoming the computer and following the ow of execution through a sample program run, recording the state of all variables and any output the program generates after each instruction is executed. 90 Chapter 7. Iteration

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

To understand this process, lets trace the call to seq3np1(3) from the previous section. At the start of the trace, we have a local variable, n (the parameter), with an initial value of 3. Since 3 is not equal to 1, the while loop body is executed. 3 is printed and 3 % 2 == 0 is evaluated. Since it evaluates to False, the else branch is executed and 3 * 3 + 1 is evaluated and assigned to n. To keep track of all this as you hand trace a program, make a column heading on a piece of paper for each variable created as the program runs and another one for output. Our trace so far would look something like this:
n -3 10 output printed so far --------------------3,

Since 10 != 1 evaluates to True, the loop body is again executed, and 10 is printed. 10 % 2 == 0 is true, so the if branch is executed and n becomes 5. By the end of the trace we have:
n -3 10 5 16 8 4 2 1 output printed so far --------------------3, 3, 10, 3, 10, 5, 3, 10, 5, 16, 3, 10, 5, 16, 8, 3, 10, 5, 16, 8, 4, 3, 10, 5, 16, 8, 4, 2, 3, 10, 5, 16, 8, 4, 2, 1.

Tracing can be a bit tedious and error prone (thats why we get computers to do this stuff in the rst place!), but it is an essential skill for a programmer to have. From this trace we can learn a lot about the way our code works. We can observe that as soon as n becomes a power of 2, for example, the program will require log2 (n) executions of the loop body to complete. We can also see that the nal 1 will not be printed as output within the body of the loop, which is why we put the special print function at the end. Tracing a program is, of course, related to single-stepping through your code and being able to inspect the variables. Using the computer to single-step for you is less error prone and more convenient. Also, as your programs get more complex, they might execute many millions of steps before they get to the code that youre really interested in, so manual tracing becomes impossible. Being able to set a breakpoint where you need one is far more powerful. So we strongly encourage you to invest time in learning using to use your programming environment (PyScripter, in these notes) to full effect. There are also some great visualization tools becoming available to help you trace and understand small fragments of Python code. The one we recommend is at http://netserv.ict.ru.ac.za/python3_viz Weve cautioned against chatterbox functions, but used them here. As we learn a bit more Python, well be able to show you how to generate a list of values to hold the sequence, rather than having the function print them. Doing this would remove the need to have all these pesky 7.6. Tracing a program 91

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

print functions in the middle of our logic, and will make the function more useful.

7.7 Counting digits


The following function counts the number of decimal digits in a positive integer:
1 2 3 4 5 6

def num_digits(n): count = 0 while n != 0: count = count + 1 n = n // 10 return count

A call to print(num_digits(710)) will display 3. Trace the execution of this function call (perhaps using the single step function in PyScripter, or the Python visualizer, or on some paper) to convince yourself that it works. This function demonstrates an important pattern of computation called a counter. The variable count is initialized to 0 and then incremented each time the loop body is executed. When the loop exits, count contains the result the total number of times the loop body was executed, which is the same as the number of digits. If we wanted to only count digits that are either 0 or 5, adding a conditional before incrementing the counter will do the trick:
1 2 3 4 5 6 7 8

def num_zero_and_five_digits(n): count = 0 while n > 0: digit = n % 10 if digit == 0 or digit == 5: count = count + 1 n = n // 10 return count

Conrm that test(num_zero_and_five_digits(1055030250), 7) passes. Notice, however, that test(num_digits(0), 1) fails. Explain why. Do you think this is a bug in the code, or a bug in the specications, or our expectations, or the tests?

7.8 Abbreviated assignment


Incrementing a variable is so common that Python provides an abbreviated syntax for it:
>>> >>> >>> 1 >>> count = 0 count += 1 count count += 1

92

Chapter 7. Iteration

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

>>> count 2

count += 1 is an abreviation for count = count + 1 . We pronouce the operator as plus-equals. The increment value does not have to be 1:
>>> n = 2 >>> n += 5 >>> n 7

There are similar abbreviations for -=, *=, /=, //= and %=:
>>> >>> >>> 10 >>> >>> 6 >>> >>> 3 >>> >>> 1 n = 2 n *= 5 n n -= 4 n n //= 2 n n %= 2 n

7.9 Help and meta-notation


Python comes with extensive documentation for all its built-in functions, and its libraries. Different systems have different ways of accessing this help. In PyScripter, click on the Help menu item, and select Python Manuals. Then search for help on the built-in function range. Youll get something like this...

7.9. Help and meta-notation

93

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Notice the square brackets in the description of the arguments. These are examples of metanotation notation that describes Python syntax, but is not part of it. The square brackets in this documentation mean that the argument is optional the programmer can omit it. So what this rst line of help tells us is that range must always have a stop argument, but it may have an optional start argument (which must be followed by a comma if it is present), and it can also have an optional step argument, preceded by a comma if it is present. The examples from help show that range can have either 1, 2 or 3 arguments. The list can start at any starting value, and go up or down in increments other than 1. The documentation here also says that the arguments must be integers. Other meta-notation youll frequently encounter is the use of bold and italics. The bold means that these are tokens keywords or symbols typed into your Python code exactly as they are, whereas the italic terms stand for something of this type. So the syntax description for variable in list : means you can substitute any legal variable and any legal list when you write your Python code. This (simplied) description of the print function, shows another example of meta-notation in which the ellipses (...) mean that you can have as many objects as you like (even zero), separated by commas: print( [object, ... ] ) Meta-notation gives us a concise and powerful way to describe the pattern of some syntax or feature.

94

Chapter 7. Iteration

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

7.10 Tables
One of the things loops are good for is generating tabular data. Before computers were readily available, people had to calculate logarithms, sines and cosines, and other mathematical functions by hand. To make that easier, mathematics books contained long tables listing the values of these functions. Creating the tables was slow and boring, and they tended to be full of errors. When computers appeared on the scene, one of the initial reactions was, This is great! We can use the computers to generate the tables, so there will be no errors. That turned out to be true (mostly) but shortsighted. Soon thereafter, computers and calculators were so pervasive that the tables became obsolete. Well, almost. For some operations, computers use tables of values to get an approximate answer and then perform computations to improve the approximation. In some cases, there have been errors in the underlying tables, most famously in the table the Intel Pentium processor chip used to perform oating-point division. Although a log table is not as useful as it once was, it still makes a good example of iteration. The following program outputs a sequence of values in the left column and 2 raised to the power of that value in the right column:
1 2

for x in range(13): # generate numbers 0 to 12 print(x, \t, 2**x)

The string \t represents a tab character. The backslash character in \t indicates the beginning of an escape sequence. Escape sequences are used to represent invisible characters like tabs and newlines. The sequence \n represents a newline. An escape sequence can appear anywhere in a string; in this example, the tab escape sequence is the only thing in the string. How do you think you represent a backslash in a string? As characters and strings are displayed on the screen, an invisible marker called the cursor keeps track of where the next character will go. After a print function, the cursor normally goes to the beginning of the next line. The tab character shifts the cursor to the right until it reaches one of the tab stops. Tabs are useful for making columns of text line up, as in the output of the previous program:
0 1 2 3 4 5 6 7 8 9 10 11 12 1 2 4 8 16 32 64 128 256 512 1024 2048 4096

7.10. Tables

95

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Because of the tab characters between the columns, the position of the second column does not depend on the number of digits in the rst column.

7.11 Two-dimensional tables


A two-dimensional table is a table where you read the value at the intersection of a row and a column. A multiplication table is a good example. Lets say you want to print a multiplication table for the values from 1 to 6. A good way to start is to write a loop that prints the multiples of 2, all on one line:
1 2 3

for i in range(1, 7): print(2 * i, end= print()

Here weve used the range function, but made it start its sequence at 1. As the loop executes, the value of i changes from 1 to 6. When all the elements of the range have been assigned to i, the loop terminates. Each time through the loop, it displays the value of 2 * i, followed by three spaces. Again, the extra end= argument in the print function suppresses the newline, and uses three spaces instead. After the loop completes, the second call to print nishes the current line, and starts a new line. The output of the program is:
2 4 6 8 10 12

So far, so good. The next step is to encapsulate and generalize.

7.12 Encapsulation and generalization


Encapsulation is the process of wrapping a piece of code in a function, allowing you to take advantage of all the things functions are good for. You have already seen some examples of encapsulation, including is_divisible in a previous chapter. Generalization means taking something specic, such as printing the multiples of 2, and making it more general, such as printing the multiples of any integer. This function encapsulates the previous loop and generalizes it to print multiples of n:
1 2 3 4

def print_multiples(n): for i in range(1, 7): print(n * i, end= print()

To encapsulate, all we had to do was add the rst line, which declares the name of the function and the parameter list. To generalize, all we had to do was replace the value 2 with the parameter n.

96

Chapter 7. Iteration

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

If we call this function with the argument 2, we get the same output as before. With the argument 3, the output is:
3 6 9 12 15 18

With the argument 4, the output is:


4 8 12 16 20 24

By now you can probably guess how to print a multiplication table by calling print_multiples repeatedly with different arguments. In fact, we can use another loop:
1 2

for i in range(1, 7): print_multiples(i)

Notice how similar this loop is to the one inside print_multiples. All we did was replace the print function with a function call. The output of this program is a multiplication table:
1 2 3 4 5 6 2 4 6 8 10 12 3 6 9 12 15 18 4 8 12 16 20 24 5 10 15 20 25 30 6 12 18 24 30 36

7.13 More encapsulation


To demonstrate encapsulation again, lets take the code from the last section and wrap it up in a function:
1 2 3

def print_mult_table(): for i in range(1, 7): print_multiples(i)

This process is a common development plan. We develop code by writing lines of code outside any function, or typing them in to the interpreter. When we get the code working, we extract it and wrap it up in a function. This development plan is particularly useful if you dont know how to divide the program into functions when you start writing. This approach lets you design as you go along.

7.14 Local variables


You might be wondering how we can use the same variable, i, in both print_multiples and print_mult_table. Doesnt it cause problems when one of the functions changes the value of the variable?

7.13. More encapsulation

97

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

The answer is no, because the i in print_multiples and the i in print_mult_table are not the same variable. Variables created inside a function denition are local; you cant access a local variable from outside its home function. That means you are free to have multiple variables with the same name as long as they are not in the same function. Python examines all the statements in a function - if any of them assign a value to a variable, that is the clue that Python uses to make the variable a local variable. The stack diagram for this program shows that the two variables named i are not the same variable. They can refer to different values, and changing one does not affect the other.

The value of i in print_mult_table goes from 1 to 6. In the diagram it happens to be 3. The next time through the loop it will be 4. Each time through the loop, print_mult_table calls print_multiples with the current value of i as an argument. That value gets assigned to the parameter n. Inside print_multiples, the value of i goes from 1 to 6. In the diagram, it happens to be 2. Changing this variable has no effect on the value of i in print_mult_table. It is common and perfectly legal to have different local variables with the same name. In particular, names like i and j are used frequently as loop variables. If you avoid using them in one function just because you used them somewhere else, you will probably make the program harder to read. The visualizer at http://netserv.ict.ru.ac.za/python3_viz/ shows very clearly how the two variables i are distinct variables, and how they have independent values. (The visualizer has a limit of showing 100 steps, though not quite enough to run this particular example all the way to the end.)

7.15 The break statement


The break statement is used to immediately leave the body of its loop. The next statement to be executed is the rst one after the body:
1 2 3 4 5

for i in [12, 16, 17, 24, 29]: if i % 2 == 1: # if the number is odd break # immediately exit the loop print(i) print("done")

This prints:

98

Chapter 7. Iteration

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

12 16 done

The pre-test loop - standard loop behaviour for and while loops do their tests at the start, before executing any part of the body. Theyre called pre-test loops, because the test happens before (pre) the body. break and return are our tools for adapting this standard behaviour.

7.16 Other avours of loops


Sometimes wed like to have the middle-test loop with the exit test in the middle of the body, rather than at the beginning or at the end. Or a post-test loop that puts its exit test as the last thing in the body. Other languages have different syntax amd keywords for these different avours, but Python just uses a combination of while and if condition: break to get the job done. A typical example is a problem where the user has to input numbers to be summed. To indicate that there are no more inputs, the user enters a special value, often the value -1, or the empty string. This needs a middle-exit loop pattern: input the next number, then test whether to exit, or else process the number: The middle-test loop owchart

7.16. Other avours of loops

99

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2 3 4 5 6 7

total = 0 while True: response = input("Enter the next number. (Leave blank to end)") if response == "": break total += int(response) print("The total of the numbers you entered is ", total)

Convince yourself that this ts the middle-exit loop owchart: line 3 does some useful work, lines 4 and 5 can exit the loop, and if they dont line 6 does more useful work before the next iteration starts. The while bool-expr: uses the boolean expression to determine whether to iterate again. True is a trivial boolean expression, so while True: means always do the loop body again. This is a language idiom a convention that most programmers will recognize immediately. Since the expression on line 2 will never terminate the loop, (it is a dummy test) the programmer must arrange to break (or return) out of the loop body elsewhere, in some other way (i.e. in lines 4 and 5 in this sample). A clever compiler or interpreter will understand that line 2 is a fake test that must always succeed, so it wont even generate a test, and our owchart never even put the diamond-shape dummy test box at the top of the loop! Similarly, by just moving the if condition: break to the end of the loop body we create a pattern for a post-test loop. Post-test loops are used when you want to be sure that the loop body always executes at least once (because the rst test only happens at the end of the execution of the rst loop body). This is useful, for example, if we want to play an interactive game against the user we always want to play at least one game:
1 2 3 4 5 6

while True: play_the_game_once() response = input("Play again? (yes or no)") if response != "yes": break print("Goodbye!")

Hint: think about where you want the exit test to happen 100 Chapter 7. Iteration

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Once youve recognized that you need a loop to repeat something, think about its terminating condition when will I want to stop iterating? Then gure out whether you need to do the test before starting the rst (and every other) iteration, or at the end of the rst (and every other) iteration, or perhaps in the middle of each iteration. Iteractive programs that require input from the user or read from les often need to exit their loops in the middle or at the end of an iteration, when it becomes clear that there is no more data to process, or the user doesnt want to play our game anymore.

7.17 An example
The following program implements a simple guessing game:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18

import random # We cover random numbers in chapter 10 rng = random.Random() # so you can peek ahead. number = rng.randrange(1, 1000) # Get random number between [1 and 1000). guesses = 0 msg = "" while True: guess = int(input(msg + "\nGuess my number between 1 and 1000: ")) guesses += 1 if guess > number: msg += str(guess) + " is too high.\n" elif guess < number: msg += str(guess) + " is too low.\n" else: break input("\n\nGreat, you got it in {0} guesses!\n\n" .format(guesses))

This program makes use of the mathematical law of trichotomy (given real numbers a and b, exactly one of these three must be true: a > b, a < b, or a == b). At line 18 there is a call to the input function, but we dont do anything with the result, not even assign it to a variable. This is legal in Python. Here it has the effect of popping up the input dialog window and waiting for the user to respond before the program terminates. Programmers often use the trick of doing some extra input at the end of a script, just to keep the window open. Also notice the use of the msg variable, initially an empty string, on lines 6, 12 and 14. Each time through the loop we extend the message being displayed: this allows us to display the programs feedback right at the same place as were asking for the next guess.

7.17. An example

101

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

7.18 The continue statement


This is a control ow statement that causes the program to immediately skip the processing of the rest of the body of the loop, for the current iteration. But the loop still carries on running for its remaining iterations:
1 2 3 4 5

for i in [12, 16, 17, 24, 29, 30]: if i % 2 == 1: # if the number is odd continue # dont process it print(i) print("done")

This prints:
12 16 24 30 done

7.19 More generalization


As another example of generalization, imagine you wanted a program that would print a multiplication table of any size, not just the six-by-six table. You could add a parameter to print_mult_table:
1 2 3

def print_mult_table(high): for i in range(1, high+1): print_multiples(i)

We replaced the value 1 with the expression high+1. If we call print_mult_table with the argument 7, it displays:
1 2 2 4 3 6 4 8 5 10 6 12

102

Chapter 7. Iteration

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

3 4 5 6 7

6 8 10 12 14

9 12 15 18 21

12 16 20 24 28

15 20 25 30 35

18 24 30 36 42

This is ne, except that we probably want the table to be square with the same number of rows and columns. To do that, we add another parameter to print_multiples to specify how many columns the table should have. Just to be annoying, we call this parameter high, demonstrating that different functions can have parameters with the same name (just like local variables). Heres the whole program:
1 2 3 4 5 6 7 8

def print_multiples(n, high): for i in range(1, high+1): print(n * i, end= ) print() def print_mult_table(high): for i in range(1, high+1): print_multiples(i, high)

Notice that when we added a new parameter, we had to change the rst line of the function (the function heading), and we also had to change the place where the function is called in print_mult_table. Now, when we call print_mult_table(7):
1 2 3 4 5 6 7 2 4 6 8 10 12 14 3 6 9 12 15 18 21 4 8 12 16 20 24 28 5 10 15 20 25 30 35 6 12 18 24 30 36 42 7 14 21 28 35 42 49

When you generalize a function appropriately, you often get a program with capabilities you didnt plan. For example, you might notice that, because ab = ba, all the entries in the table appear twice. You could save ink by printing only half the table. To do that, you only have to change one line of print_mult_table. Change
1

print_multiples(i, high+1)

to
1

print_multiples(i, i+1)

and you get:


1 2 3 4 4 6 8

9 12

16

7.19. More generalization

103

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

5 6 7

10 12 14

15 18 21

20 24 28

25 30 35

36 42

49

7.20 Functions
A few times now, we have mentioned all the things functions are good for. By now, you might be wondering what exactly those things are. Here are some of them: 1. Capturing your mental chunking. Breaking your complex tasks into sub-tasks, and giving the sub-tasks a meaningful name is a powerful mental technique. Look back at the example that illustrated the post-test loop: we assumed that we had a function called play_the_game_once. This chunking allowed us to put aside the details of the particular game is it a card game, or noughts and crosses, or a role playing game and simply focus on one isolated part of our program logic letting the player choose whether they want to play again. 2. Dividing a long program into functions allows you to separate parts of the program, debug them in isolation, and then compose them into a whole. 3. Functions facilitate the use of iteration. 4. Well-designed functions are often useful for many programs. Once you write and debug one, you can reuse it.

7.21 Paired Data


Weve already seen lists of names and lists of numbers in Python. Were going to peek ahead in the textbook a little, and show a more advanced way of representing our data. Making a pair of things in Python is as simple as putting them into parentheses, like this:
1

year_born = ("Paris Hilton", 1981)

We can put many pairs into a list of pairs:


1 2

celebs = [("Brad Pitt", 1963), ("Jack Nicholson", 1937), ("Justin Bieber", 1994)]

Here is a quick sample of things we can do with structured data like this. First, print all the celebs:
1 2

print(celebs) print(len(celebs)) [(Brad Pitt, 1963), (Jack Nicholson, 1937), (Justin Bieber, 1994)] 3

Notice that the celebs list has just 3 elements, each of them pairs.

104

Chapter 7. Iteration

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Now print the names of those celebs born before 1980:


1 2 3

for (nm, yr) in celebs: if yr < 1980: print(nm) Brad Pitt Jack Nicholson

There is a new feature in the for loop: instead of using a single loop control variable, weve put a pair of variable names, nm and yr there instead. The loop is executed three times - once for each pair, and on each iteration both the variables are assigned values from the celeb that is being handled.

7.22 Nested Loops for Nested Data


Now well come up with an even more adventurous list of structured data. In this case, we have a list of students. Each student has a name which is paired up with another list of subjects that they are enrolled for:
1 2 3 4 5 6

students = [ ("John", ["CompSci", "Physics"]), ("Vusi", ["Maths", "CompSci", "Stats"]), ("Jess", ["CompSci", "Accounting", "Economics", "Management"]), ("Sarah", ["InfSys", "Accounting", "Economics", "CommLaw"]), ("Zuki", ["Sociology", "Economics", "Law", "Stats", "Music"])]

Here weve assigned a list of ve elements to the variable students. Lets print out each student name, and the number of subjects they are enrolled for:
1 2 3

# print all students with a count of their courses. for (name, subjects) in students: print(name, "takes", len(subjects), "courses")

Python agreeably responds with the following output:


John takes 2 courses Vusi takes 3 courses Jess takes 4 courses Sarah takes 4 courses Zuki takes 5 courses

Now wed like to ask how many students are taking CompSci. This needs a counter, and for each student we need a second loop that tests each of the subjects in turn:
1 2 3 4 5 6

# Count how many students are taking CompSci counter = 0 for (name, subjects) in students: for s in subjects: # a nested loop! if s == "CompSci" : counter += 1

7.22. Nested Loops for Nested Data

105

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

7 8

print("The number of students taking CompSci is", counter) The number of students taking CompSci is 3

You should set up a list of your own data that interests you perhaps a list of your CDs, each containing a list of song titles on the CD, or a list of movie titles, each with a list of movie stars who acted in the movie. You could then ask questions like Which movies starred Angelina Jolie?

7.23 Newtons method for nding square roots


Loops are often used in programs that compute numerical results by starting with an approximate answer and iteratively improving it. For example, before we had calculators or computers, people needed to calculate square roots manually. Newton used a particularly good method (there is some evidence that this method was known many years before). Suppose that you want to know the square root of n. If you start with almost any approximation, you can compute a better approximation (closer to the actual answer) with the following formula:
1

better = (approx + n/approx)/2

Repeat this calculation a few times using your calculator. Can you see why each iteration brings your estimate a little closer? One of the amazing properties of this particular algorithm is how quickly it converges to an accurate answer a great advantage for doing it manually. By using a loop and repeating this formula until the better approximation gets close enough to the previous one, we can write a function for computing the square root. (In fact, this is how your calculator nds square roots it may have a slightly different formula and method, but it is also based on repeatedly improving its guesses.) This is an example of an indenite iteration problem: we cannot predict in advance how many times well want to improve our guess we just want to keep getting closer and closer. Our stopping condition for the loop will be when our old guess and our improved guess are close enough to each other. Ideally, wed like the old and new guess to be exactly equal to each other when we stop. But exact equality is a tricky notion in computer arithmetic when real numbers are involved. Because real numbers are not represented absolutely accurately (after all, a number like pi or the square root of two has an innite number of decimal places because it is irrational), we need to formulate the stopping test for the loop by asking is a close enough to b? This stopping condition can be coded like this:
1 2

if abs(a-b) < 0.001: break

# make this smaller for better accuracy

Notice that we take the absolute value of the difference between a and b! This problem is also a good example of when a middle-exit loop is appropriate:

106

Chapter 7. Iteration

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2 3 4 5 6 7 8 9 10 11 12 13

def sqrt(n): approx = n/2.0 # start with some or other guess at the answer while True: better = (approx + n/approx)/2.0 if abs(approx - better) < 0.001: break; approx = better return better # test cases print(sqrt(25.0)) print(sqrt(49.0)) print(sqrt(81.0))

The output is:


5.00000000002 7.0 9.0

See if you can improve the approximations by changing the stopping condition. Also, step through the algorithm (perhaps by hand, using your calculator) to see how many iterations were needed before it achieved this level of accuracy for sqrt(5).

7.24 Algorithms
Newtons method is an example of an algorithm: it is a mechanical process for solving a category of problems (in this case, computing square roots). Some kinds of knowledge are not algorithmic. For example, learning dates from history or your multiplication tables involves memorization of specic solutions. But the techniques you learned for addition with carrying, subtraction with borrowing, and long division are all algorithms. Or if you are an avid Sudoku puzzle solver, you might have some specic set of steps that you always follow. One of the characteristics of algorithms is that they do not require any intelligence to carry out. They are mechanical processes in which each step follows from the last according to a simple set of rules. And theyre designed to solve a general class or category of problems, not just a single problem. Understanding that hard problems can be solved by step-by-step algorithmic processes (and having technology to execute these algorithms for us) is one of the major breakthroughs that has had enormous benets. So while the execution of the algorithm may be boring and may require no intelligence, algorithmic or computational thinking i.e. using algorithms and automation as the basis for approaching problems is rapidly transforming our society. Some claim that this shift towards algorithmic thinking and processes is going to have even more impact on our society than the invention of the printing press. And the process of designing algorithms is interesting, intellectually challenging, and a central part of what we call programming.

7.24. Algorithms

107

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Some of the things that people do naturally, without difculty or conscious thought, are the hardest to express algorithmically. Understanding natural language is a good example. We all do it, but so far no one has been able to explain how we do it, at least not in the form of a step-by-step mechanical algorithm.

7.25 Glossary
algorithm A step-by-step process for solving a category of problems. body The statements inside a loop. breakpoint A place in your program code where program execution will pause (or break), allowing you to inspect the state of the programs variables, or single-step through individual statements, executing them one at a time. bump Programmer slang. Synonym for increment. counter A variable used to count something, usually initialized to zero and incremented in the body of a loop. cursor An invisible marker that keeps track of where the next character will be printed. decrement Decrease by 1. denite iteration A loop where we have an upper bound on the number of times the body will be executed. Denite iteration is usually best coded as a for loop. development plan A process for developing a program. In this chapter, we demonstrated a style of development based on developing code to do simple, specic things and then encapsulating and generalizing. encapsulate To divide a large complex program into components (like functions) and isolate the components from each other (by using local variables, for example). escape sequence An escape character, \, followed by one or more printable characters used to designate a nonprintable character. generalize To replace something unnecessarily specic (like a constant value) with something appropriately general (like a variable or parameter). Generalization makes code more versatile, more likely to be reused, and sometimes even easier to write. increment Both as a noun and as a verb, increment means to increase by 1. innite loop A loop in which the terminating condition is never satised. indenite iteration A loop where we just need to keep going until some condition is met. A while statement is used for this case. initialization (of a variable) To initialize a variable is to give it an initial value. Since in Python variables dont exist until they are assigned values, they are initialized when they are created. In other programming languages this is not the case, and variables can be created without being initialized, in which case they have either default or garbage values.

108

Chapter 7. Iteration

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

iteration Repeated execution of a set of programming statements. loop A statement or group of statements that execute repeatedly until a terminating condition is satised. loop variable A variable used as part of the terminating condition of a loop. meta-notation Extra symbols or notation that helps describe other notation. Here we introduced square brackets, ellipses, italics, and bold as meta-notation to help describe optional, repeatable, substitutable and xed parts of the Python syntax. middle-test loop A loop that executes some of the body, then tests for the exit condition, and then may execute some more of the body. We dont have a special Python construct for this case, but can use while and break together. nested loop A loop inside the body of another loop. newline A special character that causes the cursor to move to the beginning of the next line. post-test loop A loop that executes the body, then tests for the exit condition. We dont have a special Python construct for this, but can use while and break together. pre-test loop A loop that tests before deciding whether the execute its body. for and while are both pre-test loops. reassignment Making more than one assignment to the same variable during the execution of a program. single-step A mode of interpreter execution where you are able to execute your program one step at a time, and inspect the consequences of that step. Useful for debugging and building your internal mental model of what is going on. tab A special character that causes the cursor to move to the next tab stop on the current line. trichotomy Given any real numbers a and b, exactly one of the following relations holds: a < b, a > b, or a == b. Thus when you can establish that two of the relations are false, you can assume the remaining one is true. trace To follow the ow of execution of a program by hand, recording the change of state of the variables and any output produced.

7.26 Exercises
This chapter showed us how to sum a list of items, and how to count items. The counting example also had an if statement that let us only count some selected items. In the previous chapter we also showed a function find_first_2_letter_word that allowed us an early exit from inside a loop by using return when some condition occurred. We now also have break to exit a loop (but not the enclosing function, and continue to abandon the current iteration of the loop without ending the loop. Composition of list traversal, summing, counting, testing conditions and early exit is a rich collection of building blocks that can be combined in powerful ways to create many functions that are all slightly different.

7.26. Exercises

109

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

The rst six questions are typical functions you should be able to write using only these building blocks. 1. Write a function to count how many odd numbers are in a list. 2. Sum up all the even numbers in a list. 3. Sum up all the negative numbers in a list. 4. Count how many words in a list have length 5. 5. Sum all the elements in a list up to but not including the rst even number. (Write your unit tests. What if there is no even number?) 6. Count how many words occur in a list up to and including the rst occurrence of the word sam. (Write your unit tests for this case too. What if sam does not occur?) 7. Add a print function to Newtons sqrt function that prints out better each time it is calculated. Call your modied function with 25 as an argument and record the results. 8. Trace the execution of the last version of print_mult_table and gure out how it works. 9. Write a function print_triangular_numbers(n) that prints out the rst n triangular numbers. A call to print_triangular_numbers(5) would produce the following output:
1 2 3 4 5 1 3 6 10 15

(hint: use a web search to nd out what a triangular number is.) 10. Write a function, is_prime, which takes a single integer argument and returns True when the argument is a prime number and False otherwise. Add tests for cases like this:
test(is_prime(11), True) test(is_prime(35), False) test(is_prime(19911129), True)

The last case could represent your birth date. Were you born on a prime day? In a class of 100 students, how many do you think would have prime birth dates? 11. Revisit the drunk pirate problem from the exercises in chapter 3. This time, the drunk pirate makes a turn, and then takes some steps forward, and repeats this. Our social science student now records pairs of data: the angle of each turn, and the number of steps taken after the turn. Her experimental data is [(160, 20), (-43, 10), (270, 8), (-43, 12)]. Use a turtle to draw the path taken by our drunk friend. 12. Many interesting shapes can be drawn by the turtle by giving a list of pairs like we did above, where the rst item of the pair is the angle to turn, and the second item is the distance to move forward. Set up a list of pairs so that the turtle draws a house with a

110

Chapter 7. Iteration

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

cross through the centre, as show here. This should be done without going over any of the lines / edges more than once, and without lifting your pen.

13. Not all shapes like the one above can be drawn without lifting your pen, or going over an edge more than once. Which of these can be drawn?

Now read Wikipedias article(http://en.wikipedia.org/wiki/Eulerian_path) about Eulerian paths. Learn how to tell immediately by inspection whether it is possible to nd a solution or not. If the path is possible, youll also know where to put your pen to start drawing, and where you should end up! 14. What will num_digits(0) return? Modify it to return 1 for this case. Why does a call to num_digits(-24) result in an innite loop? (hint: -1//10 evaluates to -1) Modify num_digits so that it works correctly with any integer value. Add these tests:
test(num_digits(0), 1) test(num_digits(-12345), 5)

15. Write a function num_even_digits(n) that counts the number of even digits in n. These tests should pass:
test(num_even_digits(123456), 3) test(num_even_digits(2468), 4) test(num_even_digits(1357), 0) test(num_even_digits(0), 1)

16. Write a function sum_of_squares(xs) that computes the sum of the squares of the numbers in the list xs. For example, sum_of_squares([2, 3, 4]) should return 4+9+16 which is 29:
test(sum_of_squares([2, 3, 4]), 29) test(sum_of_squares([ ]), 0) test(sum_of_squares([2, -3, 4]), 29)

17. You and your friend are in a team to write a two-player game, human against computer, such as Tic-Tac-Toe / Noughts and Crosses. Your friend will write the logic to play one round of the game, while you will write the logic to allow many rounds of play, keep score, decide who plays, rst, etc. The two of you negotiate on how the two parts of 7.26. Exercises 111

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

the program will t together, and you come up with this simple scaffolding (which your friend will improve later):
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17

# your friend will complete this function def play_once(human_plays_first): """ Must play one round of the game. If the parameter is True, the human gets to play first, else the computer gets to play first. When the round ends, the return value of the function is one of -1 (human wins), 0 (game drawn), 1 (computer wins). """ # This is all dummy scaffolding code right at the moment... import random # see ch 10 for details rng = random.Random() # pick a random result between -1 and 1. result = rng.randrange(-1,2) print("Human plays first={0}, winner={1} " .format(human_plays_first, result)) return result

(a) Write the main program which repeatedly calls this function to play the game, and after each round it announces the outcome as I win!, You win!, or Game drawn!. It then asks the player Do you want to play again? and either plays again, or says Goodbye, and terminates. (b) Keep score of how many wins each player has had, and how many draws there have been. After each round of play, also announce the scores. (c) Add logic so that the players take turns to play rst. (d) Compute the percentage of wins for the human, out of all games played. Also announce this at the end of each round. (e) Draw a owchart of your logic.

112

Chapter 7. Iteration

CHAPTER

EIGHT

STRINGS
8.1 A compound data type
So far we have seen built-in types like int, float, bool, str and weve seen lists and pairs. Strings, lists, and pairs are qualitatively different from the others because they are made up of smaller pieces. In the case of strings, theyre made up of smaller strings each containing one character. Types that comprise smaller pieces are called compound data types. Depending on what we are doing, we may want to treat a compound data type as a single thing, or we may want to access its parts. This ambiguity is useful.

8.2 Working with strings as single things


We previously saw that each turtle instance has its own attributes and a number of methods that can be applied to the instance. For example, we could set the turtles color, and we wrote tess.turn(90). Just like a turtle, a strings is also an object. So each string instance has its own attributes and methods. For example:
>>> ss = "Hello, World!" >>> tt = ss.upper() >>> tt HELLO, WORLD!

upper is a method that can be invoked on any string object to create a new string, in which all the characters are in uppercase. (The original string ss remains unchanged.) There are also methods like lower, capitalize, and swapcase that do other interesting stuff. To learn what methods are available, you can consult the Help documentation, look for the string module, and read the documentation. Or, if youre a bit lazier, simply type the following into a PyScripter script:

113

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2

ss = "Hello, World!" tt = ss.

When you type the period to select one of the methods of ss, PyScripter will pop up a selection window showing all the methods (there are around 70 of them thank goodness well only use a few of those!) that could be used on your string.

When you type the name of the method, some further help about its parameter and return type, and its docstring, will be displayed. This is a good example of a tool PyScripter using the meta-information the docstrings provided by the module programmers.

114

Chapter 8. Strings

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

8.3 Working with the parts of a string


The indexing operator (Python uses square brackets to enclose the index) selects a single character substring from a string:
>>> fruit = "banana" >>> m = fruit[1] >>> print(m)

The expression fruit[1] selects character number 1 from fruit, and creates a new string containing just this one character. The variable m refers to the result. When we display m, we could get a surprise:
a

Computer scientists always start counting from zero! The letter at subscript position zero of "banana" is b. So at position [1] we have the letter a. If you want the zero-eth letter of a string, you just put 0, or any expression with the value 0, in the brackets:
>>> m = fruit[0] >>> print(m) b

The expression in brackets is called an index. An index species a member of an ordered collection, in this case the collection of characters in the string. The index indicates which one you want, hence the name. It can be any integer expression. Note that indexing returns a string Python has no special type for a single character. It is just a string of length 1. Weve also seen lists previously. The same indexing notation works to extract elements from a list:
>>> prime_nums = [2, 3, 5, 7, 11, 13, 17, 19, 23, 29, 31] >>> prime_nums[4] 11 >>> friends = ["Joe", "Amy", "Brad", "Angelina", "Zuki", "Thandi", "Paris"] >>> friends[3] Angelina

8.4 Length
The len function, when applied to a string, returns the number of characters in a string:
>>> fruit = "banana" >>> len(fruit) 6

To get the last letter of a string, you might be tempted to try something like this: 8.3. Working with the parts of a string 115

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2

sz = len(fruit) last = fruit[sz]

# ERROR!

That wont work. It causes the runtime error IndexError: string index out of range. The reason is that there is no letter at index position 6 in "banana". Since we started counting at zero, the six indexes are numbered 0 to 5. To get the last character, we have to subtract 1 from length:
1 2

sz = len(fruit) last = fruit[sz-1]

Alternatively, we can use negative indices, which count backward from the end of the string. The expression fruit[-1] yields the last letter, fruit[-2] yields the second to last, and so on. As you might have guessed, indexing with a negative index also works like this for lists. We wont use negative indexes in the rest of these notes not many computer languages use this idiom, and youll probably be better off avoiding it. But there is plenty of Python code out on the Internet that will use this trick, so it is best to know that it exists.

8.5 Traversal and the for loop


A lot of computations involve processing a string one character at a time. Often they start at the beginning, select each character in turn, do something to it, and continue until the end. This pattern of processing is called a traversal. One way to encode a traversal is with a while statement:
1 2 3 4 5

ix = 0 while ix < len(fruit): letter = fruit[ix] print(letter) ix += 1

This loop traverses the string and displays each letter on a line by itself. The loop condition is ix < len(fruit), so when ix is equal to the length of the string, the condition is false, and the body of the loop is not executed. The last character accessed is the one with the index len(fruit)-1, which is the last character in the string. But weve previously seen how the for loop can easily iterate over lists: for also works for strings:
1 2

for c in fruit: print(c)

Each time through the loop, the next character in the string is assigned to the variable c. The loop continues until no characters are left. The following example shows how to use concatenation and a for loop to generate an abecedarian series. Abecedarian refers to a series or list in which the elements appear in alphabetical order. For example, in Robert McCloskeys book Make Way for Ducklings, the names of 116 Chapter 8. Strings

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

the ducklings are Jack, Kack, Lack, Mack, Nack, Ouack, Pack, and Quack. This loop outputs these names in order:
1 2 3 4 5

prefixes = "JKLMNOPQ" suffix = "ack" for p in prefixes: print(p + suffix)

The output of this program is:


Jack Kack Lack Mack Nack Oack Pack Qack

Of course, thats not quite right because Ouack and Quack are misspelled. Youll x this as an exercise below.

8.6 Slices
A substring of a string is obtained by taking a slice. Similarly, we can slice a list to refer to some sublist of the items in the list:
>>> s = "Peter, Paul, and Mary" >>> print(s[0:5]) Peter >>> print(s[7:11]) Paul >>> print(s[17:21]) Mary >>> friends = ["Joe", "Amy", "Brad", "Angelina", "Zuki", "Thandi", "Paris"] >>> print(friends[2:4]) [Brad, Angelina]

The operator [n:m] returns the part of the string from the nth character to the mth character, including the rst but excluding the last. This behavior makes sense if you imagine the indices pointing between the characters, as in the following diagram:

Now if you imagine this as a piece of paper, the slice operator [n:m] cuts the paper at the n and m positions.

8.6. Slices

117

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Two tricks are added to this: if you omit the rst index (before the colon), the slice starts at the beginning of the string (or list). If you omit the second index, the slice extends to the end of the string (or list). Thus:
>>> fruit = "banana" >>> fruit[:3] ban >>> fruit[3:] ana

What do you think s[:] means? What about friends[4:]?

8.7 String comparison


The comparison operators work on strings. To see if two strings are equal:
1 2

if word == "banana": print("Yes, we have no bananas!")

Other comparison operations are useful for putting words in lexicographical order:
1 2 3 4 5 6

if word < "banana": print("Your word, " + word + ", comes before banana.") elif word > "banana": print("Your word, " + word + ", comes after banana.") else: print("Yes, we have no bananas!")

This is similar to the alphabetical order you would use with a dictionary, except that all the uppercase letters come before all the lowercase letters. As a result:
Your word, Zebra, comes before banana.

A common way to address this problem is to convert strings to a standard format, such as all lowercase, before performing the comparison. A more difcult problem is making the program realize that zebras are not fruit.

8.8 Strings are immutable


It is tempting to use the [] operator on the left side of an assignment, with the intention of changing a character in a string. For example:
1 2 3

greeting = "Hello, world!" greeting[0] = J print(greeting)

# ERROR!

Instead of producing the output Jello, world!, this code produces the runtime error TypeError: str object does not support item assignment.

118

Chapter 8. Strings

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Strings are immutable, which means you cant change an existing string. The best you can do is create a new string that is a variation on the original:
1 2 3

greeting = "Hello, world!" newGreeting = J + greeting[1:] print(newGreeting)

The solution here is to concatenate a new rst letter onto a slice of greeting. This operation has no effect on the original string.

8.9 The in and not in operators


The in operator tests if one string is a substring of another:
>>> p in apple True >>> i in apple False >>> ap in apple True >>> pa in apple False

Note that a string is a substring of itself, and the empty string is a substring of any other string. (Also note that computer scientists like to think about these edge cases quite carefully!)
>>> a in a True >>> apple in apple True >>> in a True >>> in apple True

The not in operator returns the logical opposite results of in:


>>> x not in apple True

Combining the in operator with string concatenation using +, we can write a function that removes all the vowels from a string:
1 2 3 4 5 6 7 8

def remove_vowels(s): vowels = "aeiouAEIOU" s_without_vowels = "" for x in s: if x not in vowels: s_without_vowels += x return s_without_vowels

8.9. The in and not in operators

119

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

9 10

test(remove_vowels("compsci"), "cmpsc") test(remove_vowels("aAbEefIijOopUus"), "bfjps")

8.10 A find function


What does the following function do?
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16

def find(strng, ch): """ Find and return the index of ch in strng. Return -1 if ch does not occur in strng. """ ix = 0 while ix < len(strng): if strng[ix] == ch: return ix ix += 1 return -1 test(find("Compsci", test(find("Compsci", test(find("Compsci", test(find("Compsci", "p"), "C"), "i"), "x"), 3) 0) 6) -1)

In a sense, find is the opposite of the indexing operator. Instead of taking an index and extracting the corresponding character, it takes a character and nds the index where that character appears. If the character is not found, the function returns -1. This is another example where we see a return statement inside a loop. If strng[ix] == ch, the function returns immediately, breaking out of the loop prematurely. If the character doesnt appear in the string, then the program exits the loop normally and returns -1. This pattern of computation is sometimes called a eureka traversal or short-circuit evaluation, because as soon as we nd what we are looking for, we can cry Eureka!, take the short-circuit, and stop looking.

8.11 Looping and counting


The following program counts the number of times the letter a appears in a string, and is another example of the counter pattern introduced in Counting digits:
1 2 3 4 5

def count_a(text): count = 0 for c in text: if c == a: count += 1

120

Chapter 8. Strings

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

6 7 8

return(count) test(count_a("banana"), 3)

8.12 Optional parameters


To nd the locations of the second or third occurence of a character in a string, we can modify the find function, adding a third parameter for the starting postion in the search string:
1 2 3 4 5 6 7 8 9

def find2(strng, ch, start): ix = start while ix < len(strng): if strng[ix] == ch: return ix ix += 1 return -1 test(find2(banana, a, 2), 3)

The call find2(banana, a, 2) now returns 3, the index of the rst occurrence of a in banana after index 2. What does find2(banana, n, 3) return? If you said, 4, there is a good chance you understand how find2 works. Better still, we can combine find and find2 using an optional parameter:
1 2 3 4 5 6 7

def find(strng, ch, start=0): ix = start while ix < len(strng): if strng[ix] == ch: return ix ix += 1 return -1

When a function has an optional parameter, the caller may provide a matching argument. If the third argument is provided to find, it gets assigned to start. But if the caller leaves the argument out, then start is given a default value indicated by the assignment start=0 in the function denition. So the call find(banana, a, 2) to this version of find behaves just like find2, while in the call find(banana, a), start will be set to the default value of 0. Adding another optional parameter to find makes it search from a starting position, up to but not including the end position:
1 2 3 4 5 6

def find(strng, ch, start=0, end=None): ix = start if end == None: end = len(strng) while ix < end: if strng[ix] == ch:

8.12. Optional parameters

121

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

7 8 9

return ix ix += 1 return -1

The optional value for end is interesting: we give it a default value None if the caller does not supply any argument. In the body of the function we test what end is, and if the caller did not supply any argument, we reassign end to be the length of the string. If the caller has supplied an argument for end, however, the callers value will be used in the loop. The semantics of start and end in this function are precisely the same as they are in the range function. Here are some test cases that should pass:
1 2 3 4 5 6

ss = "Python strings have some interesting methods." test(find(ss, s), 7) test(find(ss, s, 7), 7) test(find(ss, s, 8), 13) test(find(ss, s, 8, 13), -1) test(find(ss, .), len(ss)-1)

8.13 The built-in find method


Now that weve done all this work to write a powerful find function, we can let on that strings already have their own built-innd method. It can do everything that our one can do, and more!
1 2 3 4 5

test(ss.find(s), 7) test(ss.find(s, 7), 7) test(ss.find(s, 8), 13) test(ss.find(s, 8, 13), -1) test(ss.find(.), len(ss)-1)

The built-in find method is more general than our version. It can nd substrings, not just single characters:
>>> "banana".find("nan") 2 >>> "banana".find("na", 3) 4

Usually wed prefer to use the methods that Python provides rather than reinvent our own equivalents. But many of the built-in functions and methods make good teaching exercises, and the underlying techniques you learn are your building blocks to becoming a procient programmer.

122

Chapter 8. Strings

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

8.14 The split method


One of the most useful methods on strings is the split method: it splits a single multi-word string into a list of individual words, removing all the whitespace between them. (Whitespace means any tabs, newlines, or spaces.) This allows us to read input as a single string, and split it into words.
>>> ss = Well I never did said Alice >>> wds = ss.split() >>> wds [Well, I, never, did, said, Alice]

8.15 Cleaning up your strings


Well often work with strings that contain punctuation, or tab and newline characters, especially, as well see in a future chapter, when we read our text from les or from the Internet. But if were writing a program, say, to count word frequencies or check the spelling of each word, wed prefer to strip off these unwanted characters. Well show just one example of how to strip punctuation from a string. Remember that strings are immutable, so we cannot change the string with the punctuation we need to traverse the original string and create a new string, omitting any punctuation:
1 2 3 4 5 6 7 8

punctuation = !"#$%&\()*+,-./:;<=>?@[\\]^_{|}~ def remove_punctuation(s): s_without_punct = "" for letter in s: if letter not in punctuation: s_without_punct += letter return s_without_punct

Setting up that rst assignment is messy and error-prone. Fortunately, the Python string module already does it for us. So we will make a slight improvement to this program well import the string module and use its denition:
1 2 3 4 5 6 7 8 9 10 11 12 13

import string def remove_punctuation(s): s_without_punct = "" for letter in s: if letter not in string.punctuation: s_without_punct += letter return s_without_punct test(remove_punctuation("Well, I never did!", said Alice.), "Well I never did said Alice") test(remove_punctuation("Are you very, very, sure?"), "Are you very very sure")

8.14. The split method

123

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Composing together this function and the split method from the previous section makes a useful combination well clean out the punctuation, and split will clean out the newlines and tabs while turning the string into a list of words:
1 2 3 4 5 6 7 8 9 10 11 12

my_story = Pythons are constrictors, which means that they will squeeze the life out of their prey. They coil themselves around their prey and with each breath the creature takes the snake will squeeze a little tighter until they stop breathing completely. Once the heart stops the prey is swallowed whole. The entire animal is digested in the snakes stomach except for fur or feathers. What do you think happens to the fur, feathers, beaks, and eggshells? The extra stuff gets passed out as --you guessed it --- snake POOP! wds = remove_punctuation(my_story).split() print(wds)

The output:
[Pythons, are, constrictors, ... , it, snake, POOP]

There are other useful functions in the string module, other ways to classify characters, and other methods on string instances, but this book isnt intended to be a reference manual. On the other hand, the Python Library Reference is. Along with a wealth of other documentation, it is available from the Python website, http://www.python.org.

8.16 The format method for strings


The easiest and most powerful way to format a string in Python 3 is to use the format method. To see how this works, lets start with a few examples:
1 2 3 4 5 6 7 8 9 10 11 12

s1 = "His name is {0}!".format("Arthur") print(s1) name = "Alice" age = 10 s2 = "I am {0} and I am {1} years old.".format(name, age) print(s2) n1 = 4 n2 = 5 s3 = "2**10 = {0} and {1} * {2} = {3:f}".format(2**10, n1, n2, n1 * n2) print(s3)

Running the script produces:


His name is Arthur! I am Alice and I am 10 years old. 2**10 = 1024 and 4 * 5 = 20.000000

124

Chapter 8. Strings

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

The key idea is that one provides a formatter string which contains placeholder elds, ... {0} ... {1} ... {2} ... etc. The format method of a string uses the numbers as indexes into its arguments, and substitutes the appropriate argument into each placeholder eld. Each of the placeholders can also contain a format specication it is always introduced by the : symbol. This can control things like whether the eld is aligned left <, centered ^, or right > the width allocated to the eld within the result string (a number like 10) the type of conversion (well initially only force conversion to oat, f, as we did in line 11 of the code above, or perhaps well ask integer numbers to be converted to hexadecimal using x) if the type conversion is a oat, you can also specify how many decimal places are wanted (typically, .2f is useful for working with currencies to two decimal places.) Lets do a few simple and common examples that should be enough for most needs. If you need to do anything more esoteric, use help and read all the gory details.
1 2 3 4 5 6 7 8 9 10

n1 = "Paris" n2 = "Whitney" n3 = "Hilton" print("Ppi to three decimal places is {0:.3f}".format(3.1415926)) print("123456789 123456789 123456789 123456789 123456789 123456789") print("|||{0:<15}|||{1:^15}|||{2:>15}|||Born in {3}|||" .format(n1,n2,n3,1981)) print("The decimal value {0} converts to hex value {0:x}" .format(123456))

This script produces the output:


The value of pi to three decimal places is 3.142 123456789 123456789 123456789 123456789 123456789 123456789 |||Paris ||| Whitney ||| Hilton|||Born in 1981||| The decimal value 123456 converts to hex value 1e240

You can have multiple placeholders indexing the same argument, or perhaps even have extra arguments that are not referenced at all:
1 2 3 4 5 6 7 8 9

letter = Dear {0} {2}. {0}, I have an interesting money-making proposition for you! If you deposit $10 million into my bank account, I can double your money ... print(letter.format("Paris", "Whitney", "Hilton")) print(letter.format("Bill", "Henry", "Gates"))

This produces the following:

8.16. The format method for strings

125

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Dear Paris Hilton. Paris, I have an interesting money-making proposition for you! If you deposit $10 million into my bank account, I can double your money ... Dear Bill Gates. Bill, I have an interesting money-making proposition for you! If you deposit $10 million into my bank account I can, double your money ...

As you might expect, youll get an index error if your placeholders refer to arguments that you do not provide:
>>> "hello {3}".format("Dave") Traceback (most recent call last): File "<interactive input>", line 1, in <module> IndexError: tuple index out of range

The following example illustrates the real utility of string formatting. First, well try to print a table without using string formatting:
1 2 3 4

print("i\ti**2\ti**3\ti**5\ti**10\ti**20") for i in range(1, 11): print(i, \t, i**2, \t, i**3, \t, i**5, \t, i**10, \t, i**20)

This program prints out a table of various powers of the numbers from 1 to 10. (This assumes that the tab width is 8. You might see something even worse than this if you tab width is set to 4.) In its current form it relies on the tab character ( \t) to align the columns of values, but this breaks down when the values in the table get larger than the tab width:
i 1 2 3 4 5 6 7 8 9 10 i**2 1 4 9 16 25 36 49 64 81 100 i**3 1 8 27 64 125 216 343 512 729 1000 i**5 1 32 243 1024 3125 7776 16807 32768 59049 100000 i**10 i**20 1 1 1024 1048576 59049 3486784401 1048576 1099511627776 9765625 95367431640625 60466176 3656158440062976 282475249 79792266297612001 1073741824 1152921504606846976 3486784401 12157665459056928801 10000000000 100000000000000000000

One possible solution would be to change the tab width, but the rst column already has more space than it needs. The best solution would be to set the width of each column independently. As you may have guessed by now, string formatting provides a much nicer solution. We can also right-justify each eld:
1 2 3 4

layout = "{0:>4}{1:>6}{2:>6}{3:>8}{4:>13}{5:>24}" print(layout.format(i, i**2, i**3, i**5, i**10, i**20)) for i in range(1, 11):

126

Chapter 8. Strings

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

print(layout.format(i, i**2, i**3, i**5, i**10, i**20))

Running this version produces the following (much more satisfying) output:
i 1 2 3 4 5 6 7 8 9 10 i**2 1 4 9 16 25 36 49 64 81 100 i**3 1 8 27 64 125 216 343 512 729 1000 i**5 1 32 243 1024 3125 7776 16807 32768 59049 100000 i**10 1 1024 59049 1048576 9765625 60466176 282475249 1073741824 3486784401 10000000000 i**20 1 1048576 3486784401 1099511627776 95367431640625 3656158440062976 79792266297612001 1152921504606846976 12157665459056928801 100000000000000000000

8.17 Summary
This chapter introduced a lot of new ideas. The following summary may prove helpful in remembering what you learned. indexing ([]) Access a single character in a string using its position (starting from 0). Example: This[2] evaluates to i. length function (len) Returns the number of characters in a string. len(happy) evaluates to 5. Example:

for loop traversal (for) Traversing a string means accessing each character in the string, one at a time. For example, the following for loop:
for ix in Example: ...

executes the body of the loop 7 times with different values of ix each time. slicing ([:]) A slice is a substring of a string. Example: bananas and cream[3:6] evaluates to ana (so does bananas and cream[1:4]). string comparison (>, <, >=, <=, ==, !=) The six common comparision operators work with strings, evaluating according to lexicographical order. Examples: apple < banana evaluates to True. Zeta < Appricot evaluates to False. Zebra <= aardvark evaluates to True because all upper case letters precede lower case letters. in and not in operator (in, not in) The in operator tests whether one string is contained inside another string. Examples: heck in "Ill be checking for you." evaluates to True. cheese in "Ill be checking for you." evaluates to False.

8.17. Summary

127

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

8.18 Glossary
compound data type A data type in which the values are made up of components, or elements, that are themselves values. default value The value given to an optional parameter if no argument for it is provided in the function call. docstring A string constant on the rst line of a function or module denition (and as we will see later, in class and method denitions as well). Docstrings provide a convinient way to associate documentation with code. Docstrings are also used by programming tools to provide interactive help. dot notation Use of the dot operator, ., to access methods and attributes of an object. immutable A compound data type whose elements can not be assigned new values. index A variable or value used to select a member of an ordered collection, such as a character from a string, or an element from a list. optional parameter A parameter written in a function header with an assignment to a default value which it will receive if no corresponding argument is given for it in the function call. short-circuit evaluation A style of programming that shortcuts extra work as soon as the outcome is know with certainty. In this chapter our find function returned as soon as it found what it was looking for: it didnt traverse all the rest of the items in the string. slice A part of a string (substring) specied by a range of indices. More generally, a subsequence of any sequence type in Python can be created using the slice operator (sequence[start:stop]). traverse To iterate through the elements of a collection, performing a similar operation on each. whitespace Any of the characters that move the cursor without printing visible characters. The constant string.whitespace contains all the white-space characters.

8.19 Exercises
We suggest you create a single le containing the test scaffolding from our previous chapters, and put all functions that require tests into that le. 1. What is the result of each of the following:
>>> >>> >>> >>> >>> >>> >>> Python[1] "Strings are sequences of characters."[5] len("wonderful") Mystery[:4] p in Pineapple apple in Pineapple pear not in Pineapple

128

Chapter 8. Strings

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

>>> apple > pineapple >>> pineapple < Peach

2. Modify:
1 2 3 4 5

prefixes = "JKLMNOPQ" suffix = "ack" for letter in prefixes: print(letter + suffix)

so that Ouack and Quack are spelled correctly. 3. Encapsulate


1 2 3 4 5 6

fruit = "banana" count = 0 for char in fruit: if char == a: count += 1 print(count)

in a function named count_letters, and generalize it so that it accepts the string and the letter as arguments. Make the function return the number of characters, rather than print the answer. The caller should do the printing. 4. Now rewrite the count_letters function so that instead of traversing the string, it repeatedly calls the find method, with the optional third parameter to locate new occurrences of the letter being counted. 5. Assign to a variable in your program a triple-quoted string that contains your favourite paragraph of text - perhaps a poem, a speech, instructions to bake a cake, some inspirational verses, etc. Write a function which removes all punctuation from the string, breaks the string into a list of words, and counts the number of words in your text that contain the letter e. Your program should print an analysis of the text like this:
Your text contains 243 words, of which 109 (44.8%) contain an e.

6. Print out a neatly formatted multiplication table, up to 12 x 12. 7. Write a function that reverses its string argument, and satises these tests:
1 2 3 4

test(reverse(happy), yppah) test(reverse(Python), nohtyP) test(reverse(), ) test(reverse(a), a)

8. Write a function that mirrors its argument:


1 2

test(mirror(good), gooddoog) test(mirror(Python), PythonnohtyP)

8.19. Exercises

129

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

3 4

test(mirror(), ) test(mirror(a), aa)

9. Write a function that removes all occurrences of a given letter from a string:
1 2 3 4 5 6

test(remove_letter(a, test(remove_letter(a, test(remove_letter(z, test(remove_letter(i, test(remove_letter(b, test(remove_letter(b,

apple), pple) banana), bnn) banana), banana) Mississippi), Msssspp) ), ) c), c)

10. Write a function that recognizes palindromes. (Hint: use your reverse function to make this easy!):
1 2 3 4 5 6 7

test(is_palindrome(abba), True) test(is_palindrome(abab), False) test(is_palindrome(tenet), True) test(is_palindrome(banana), False) test(is_palindrome(straw warts), True) test(is_palindrome(a), True) #test(is_palindrome(), ??) # Is an empty string a palindrome?

11. Write a function that counts how many times a substring occurs in a string:
1 2 3 4 5 6

test(count(is, Mississippi), 2) test(count(an, banana), 2) test(count(ana, banana), 2) test(count(nana, banana), 1) test(count(nanan, banana), 0) test(count(aaa, aaaaaa), 4)

12. Write a function that removes the rst occurrence of a string from another string:
1 2 3 4

test(remove(an, banana), bana) test(remove(cyc, bicycle), bile) test(remove(iss, Mississippi), Missippi) test(remove(egg, bicycle), bicycle)

13. Write a function that removes all occurrences of a string from another string:
1 2 3 4

test(remove_all(an, banana), ba) test(remove_all(cyc, bicycle), bile) test(remove_all(iss, Mississippi), Mippi) test(remove_all(eggs, bicycle), bicycle)

130

Chapter 8. Strings

CHAPTER

NINE

TUPLES
9.1 Tuples are used for grouping data
Weve seen earlier that we can group together pairs of values by putting them into parentheses. Recall this example:
>>> year_born = ("Paris Hilton", 1981)

This is an example of a data structure a mechanism for grouping and organizing data to make it easier to use. The pair is an example of a tuple. Generalizing this, a tuple can be used to group any number of items into a single compound value. Syntactically, a tuple is a comma-separated sequence of values. Although it is not necessary, it is conventional to enclose tuples in parentheses:
>>> julia = ("Julia", "Roberts", 1967, "Duplicity", 2009, "Actress", "Atlanta, Georgia")

Tuples are useful for representing what other languages often call records some related information that belongs together, like your student record. There is no description of what each of these elds means, but we can guess. A tuple lets us chunk together related information and use it as a single thing. Tuples support the same sequence operations as strings. The index operator selects an element from a tuple.
>>> julia[2] 1967

But if we try to use item assignment to modify one of the elements of the tuple, we get an error:
>>> julia[0] = X TypeError: tuple object does not support item assignment

So like strings, tuples are immutable. Once Python has created a tuple in memory, it cannot be changed. Of course, even if we cant modify the elements of a tuple, we can always make the julia variable reference a new tuple holding different information. To construct the new tuple, it is convenient that we can slice parts of the old tuple and join up the bits to make the new tuple. 131

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

So if julia has a new recent lm, we could change her variable to reference a new tuple that used some information from the old one:

>>> julia = julia[:3] + ("Eat Pray Love", 2010) + julia[5:] >>> julia (Julia, Roberts, 1967, Eat Pray Love, 2010, Actress, Atlanta, Georg

To create a tuple with a single element (but youre probably not likely to do that too often), we have to include the nal comma, because without the nal comma, Python treats the (5) below as an integer in parentheses:
>>> tup = (5,) >>> type(tup) <class tuple> >>> x = (5) >>> type(x) <class int>

9.2 Tuple assignment


Python has a very powerful tuple assignment feature that allows a tuple of variables on the left of an assignment to be assigned values from a tuple on the right of the assignment. (We already saw this used for pairs, but it generalizes.)
(name, surname, b_year, movie, m_year, profession, b_place) = julia

This does the equivalent of seven assignment statements, all on one easy line. One requirement is that the number of variables on the left must match the number of elements in the tuple. Once in a while, it is useful to swap the values of two variables. With conventional assignment statements, we have to use a temporary variable. For example, to swap a and b:
1 2 3

temp = a a = b b = temp

Tuple assignment solves this problem neatly:


1

(a, b) = (b, a)

The left side is a tuple of variables; the right side is a tuple of values. Each value is assigned to its respective variable. All the expressions on the right side are evaluated before any of the assignments. This feature makes tuple assignment quite versatile. Naturally, the number of variables on the left and the number of values on the right have to be the same:
>>> (a, b, c, d) = (1, 2, 3) ValueError: need more than 3 values to unpack

132

Chapter 9. Tuples

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

9.3 Tuples as return values


Functions can always only return a single value, but by making that value a tuple, we can effectively group together as many values as we like, and return them together. This is very useful we often want to know some batsmans highest and lowest score, or we want to nd the mean and the standard deviation, or we want to know the year, the month, and the day, or if were doing some some ecological modelling we may want to know the number of rabbits and the number of wolves on an island at a given time. For example, we could write a function that returns both the area and the circumference of a circle of radius r:
1 2 3 4 5

def f(r): """ Return (circumference, area) of a circle of radius r """ c = 2 * math.pi * r a = math.pi * r * r return (c, a)

9.4 Composability of Data Structures


We saw in an earlier chapter that we could make a list of pairs, and we had an example where one of the items in the tuple was itself a list:
students = [ ("John", ["CompSci", "Physics"]), ("Vusi", ["Maths", "CompSci", "Stats"]), ("Jess", ["CompSci", "Accounting", "Economics", "Management"]), ("Sarah", ["InfSys", "Accounting", "Economics", "CommLaw"]), ("Zuki", ["Sociology", "Economics", "Law", "Stats", "Music"])]

Tuples items can themselves be other tuples. For example, we could improve the information about our movie stars to hold the full date of birth rather than just the year, and we could have a list of some of her movies and dates that they were made, and so on:
julia_more_info = ( ("Julia", "Roberts"), (8, "October", 1967), "Actress", ("Atlanta", "Georgia"), [ ("Duplicity", 2009), ("Notting Hill", 1999), ("Pretty Woman", 1990), ("Erin Brockovich", 2000), ("Eat Pray Love", 2010), ("Mona Lisa Smile", 2003), ("Oceans Twelve", 2004) ])

Notice in this case the tuple has just ve elements - but each of those in turn can be another tuple, a list, a string, or any other kind of Python value.

9.3. Tuples as return values

133

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

9.5 Glossary
data structure An organization of data for the purpose of making it easier to use. immutable data type A data type which cannot be modied. Assignments to elements or slices (sub-parts) of immutable types cause a runtime error. mutable data type A data type which can be modied. All mutable types are compound types. Lists and dictionaries (coming soon) are mutable data types; strings and tuples are not. tuple An immutable data type that contains related elements. Tuples are used to group together related data, such as a persons name, their age, and their gender. tuple assignment An assignment to all of the elements in a tuple using a single assignment statement. Tuple assignment occurs simultaneously rather than in sequence, making it useful for swapping values.

9.6 Exercises
1. Weve said nothing in this chapter about whether you can pass tuples as arguments to a function. Construct a small Python example to test whether this is possible, and write up your ndings. 2. Is a pair a generalization of a tuple, or is a tuple a generalization of a pair? 3. Is a pair a kind of tuple, or is a tuple a kind of pair?

134

Chapter 9. Tuples

CHAPTER

TEN

EVENT-DRIVEN PROGRAMMING
Most programs and devices like your cellphone respond to events things that happen. For example, you might move your mouse, and the computer responds. Or you click a button, and the program does something interesting. In this chapter well touch very briey on how event-driven programming works.

10.1 Keypress events


Heres a program with some new features. Copy it into your workspace, run it. When the turtle window opens, press the arrow keys and make tess move about!
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

import turtle turtle.setup(400,500) wn = turtle.Screen() wn.title("Handling keypresses!") wn.bgcolor("lightgreen") tess = turtle.Turtle() # # # # # determine the window size get a reference to the window Change the window title Set the background color Create our favorite turtle

# The next four functions are our "event handlers". def h1(): tess.forward(30) def h2(): tess.left(45) def h3(): tess.right(45) def h4(): wn.bye()

# Close down the turtle window

# These lines "wire up" keypresses to the handlers weve defined. wn.onkey(h1, "Up") wn.onkey(h2, "Left") wn.onkey(h3, "Right") wn.onkey(h4, "q")

135

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

27 28 29 30 31 32

# Now we need to tell the window to start listening for events, # If any of the keys that were monitoring is pressed, its # handler will be called. wn.listen() wn.mainloop()

Here are some points to note: You need the call to the windows listen method at line 31, otherwise it wont notice your keypresses. We named our handler functions h1, h2 and so on, but you can choose better names. The handlers can be arbitrarily complex functions that call other functions, etc. Pressing the q key on the keyboard calls function h4 (because we bound the q key to h4 on line 26). While executing h4, the windows bye method (line 24) closes the turtle window, which causes the windows mainloop call (line 31) to end its execution. Since we did not write any more statements after line 32, this means that our program has completed everything, so it too will terminate. You can refer to keys on the keyboard by their character code (as we did in line 26), or by their symbolic names. Some of the symbolic names you can try are Cancel (the Break key), BackSpace, Tab, Return(the Enter key), Shift_L (any Shift key), Control_L (any Control key), Alt_L (any Alt key), Pause, Caps_Lock, Escape, Prior (Page Up), Next (Page Down), End, Home, Left, Up, Right, Down, Print, Insert, Delete, F1, F2, F3, F4, F5, F6, F7, F8, F9, F10, F11, F12, Num_Lock, and Scroll_Lock.

10.2 Mouse events


A mouse event is a bit different from a keypress event because its handler needs two parameters to receive x,y coordinate information telling us where the mouse was when the event occurred.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17

import turtle turtle.setup(400,500) wn = turtle.Screen() wn.title("How to handle mouse clicks on the window!") wn.bgcolor("lightgreen") tess = turtle.Turtle() tess.color("purple") tess.pensize(3) tess.shape("circle") def h1(x, y): tess.goto(x, y) wn.onclick(h1) wn.mainloop() # Wire up a click on the window.

136

Chapter 10. Event-Driven Programming

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

There is a new turtle method used at line 14 this allows us to move the turtle to an absolute coordinate position. (Most of the examples that weve seen so far move the turtle relative to where it currently is). So what this program does is move the turtle (and draw a line) to wherever the mouse is clicked. Try it out! If we add this line before line 14, well learn a useful debugging trick too:
wn.title("Got click at coords {0}, {1}".format(x, y))

Because we can easily change the text in the windows title bar, it is a useful place to display occasional debugging or status information. (Of course, this is not the real purpose of the window title!) But there is more! Not only can the window receive mouse events: individual turtles can also have their own handlers for mouse clicks. The turtle that receives the click event will be the one under the mouse. So well create two turtles. Each will bind a handler to its onclick event. And the two handlers can do different things for their turtles.
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

import turtle turtle.setup(400,500) # determine the window size wn = turtle.Screen() # get a reference to the window wn.title("Handling mouse clicks!") # Change the window title wn.bgcolor("lightgreen") # Set the background color tess = turtle.Turtle() # Create two turtles tess.color("purple") alex = turtle.Turtle() # and move them apart alex.color("blue") alex.forward(100) def handler_for_tess(x, y): wn.title("Tess clicked at {0}, {1}".format(x, y)) tess.left(42) tess.forward(30) def handler_for_alex(x, y): wn.title("Alex clicked at {0}, {1}".format(x, y)) alex.right(84) alex.forward(50) tess.onclick(handler_for_tess) alex.onclick(handler_for_alex) wn.mainloop()

Run this, click on the turtles, see what happens!

10.2. Mouse events

137

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

10.3 Automatic events from a timer


Alarm clocks, kitchen timers, and thermonuclear bombs in James Bond movies are set to create an automatic event after a certain interval. The turtle module in Python has a timer that can cause an event when its time is up.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17

import turtle turtle.setup(400,500) wn = turtle.Screen() wn.title("Using a timer") wn.bgcolor("lightgreen") tess = turtle.Turtle() tess.color("purple") tess.pensize(3) def h1( ): tess.forward(100) tess.left(56) wn.ontimer(h1, 2000) wn.mainloop()

On line 16 the timer is started and set to explode in 2000 milliseconds (2 seconds). When the event does occur, the handler is called, and tess springs into action. Unfortunately, when one sets a timer, it only goes off once. So a common idiom, or style, is to restart the timer inside the handler. In this way the timer will keep on giving new events. Try this program:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17

import turtle turtle.setup(400,500) wn = turtle.Screen() wn.title("Using a timer to get events!") wn.bgcolor("lightgreen") tess = turtle.Turtle() tess.color("purple") def h1( ): tess.forward(100) tess.left(56) wn.ontimer(h1, 60) h1() wn.mainloop()

138

Chapter 10. Event-Driven Programming

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

10.4 An example: state machines


A state machine is a system that can be in one of a few different states. We draw a state diagram to represent the machine, where each state is drawn as a circle or an ellipse. Certain events occur which cause the system to leave one state and transition into a different state. These state transitions are usually drawn as an arrow on the diagram. This idea is not new: when you rst turn on your cellphone, it goes into a state which we could call Awaiting PIN. When the correct PIN is entered, it transitions into a different state say Ready. Then you could lock the phone, and it would enter a Locked state, and so on. A simple state machine that we encounter often is a trafc light. Here is a state diagram which shows that the machine continually cycles through three different states, which weve numbered 0, 1 and 2.

Were going to build a program that uses a turtle to simulate the trafc lights. There are three lessons here. The rst shows off some different ways to use our turtles. The second demonstrates how we would program a state machine in Python, by using a variable to keep track of the current state, and a number of different if statements to inspect the current state, and take the actions as we change to a different state. The third lesson is to use events from the keyboard to trigger the state changes. Copy and run this program. Make sure you understand what each line does, consulting the documentation as you need to.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17

import turtle

# Tess becomes a traffic light.

turtle.setup(400,500) wn = turtle.Screen() wn.title("Tess becomes a traffic light!") wn.bgcolor("lightgreen") tess = turtle.Turtle()

def draw_housing(): Draw a nice housing to hold the traffic lights tess.pensize(3) tess.color(black, darkgrey) tess.begin_fill() tess.forward(80) tess.left(90) tess.forward(200)

10.4. An example: state machines

139

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

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

tess.circle(40, 180) tess.forward(200) tess.left(90) tess.end_fill()

draw_housing() tess.penup() # Position Tess onto the place where the green light should be tess.forward(40) tess.left(90) tess.forward(50) # Turn tess into a big green circle tess.shape(circle) tess.shapesize(3) tess.fillcolor(green) # # # # A traffic light is a kind of state machine with three states, Green, Orange, Red. We number these states 0, 1, 2 When the machine changes state, we change tess position and her fillcolor.

# This variable holds the current state of the machine state_num = 0

def advance_state_machine(): global state_num if state_num == 0: # transition from state 0 to state 1 tess.forward(70) tess.fillcolor(orange) state_num = 1 elif state_num == 1: # transition from state 1 to state 2 tess.forward(70) tess.fillcolor(red) state_num = 2 else: # transition from state 2 to state 0 tess.back(140) tess.fillcolor(green) state_num = 0 # bind the event handler to the space key. wn.onkey(advance_state_machine, "space") wn.listen() wn.mainloop() # listen for events

The new Python statement is at line 46. The global keyword tells Python not to create a new local variable for state_num (in spite of the fact that the function assigns to this variable at lines 50, 54, and 58). Instead, in this function, state_num always refers to the variable that 140 Chapter 10. Event-Driven Programming

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

was created at line 42. What the code in advance_state_machine does is advance from whatever the current state is, to the next state. On the state change we move tess to her new position, change her color, and, of course, we assign to state num the number of the new state weve just entered. Each time the space bar is pressed, the event handler causes the trafc light machine to move to its new state.

10.5 Glossary
bind We bind a function (or associate it) with an event, meaning that when the event occurs, the function is called to handle it. event Something that happens outside the normal control ow of your program, usually from some user action. Typical events are mouse operations and keypresses. Weve also seen that a timer can be primed to create an event. handler A function that is called in response to an event.

10.6 Exercises
1. Add some new key bindings to the rst sample program: Pressing keys R, G or B should change tess color to Red, Green or Blue. Pressing keys + or - should increase or decrease the width of tess pen. Ensure that the pen size stays between 1 and 20 (inclusive). Handle some other keys to change some attributes of tess, or attributes of the window, or to give her new behaviour that can be controlled from the keyboard. 2. Change the trafc light program so that changes occur automatically, driven by a timer. 3. In an earlier chapter we saw two turtle methods, hideturtle and showturtle that can hide or show a turtle. This suggests that we could take a different approach to the trafc lights program. Modify the program so that we create three separate turtles for each of the green, orange and red lights, and instead of moving tess to different positions and changing her color, we just make one of the three turtles visible at any time. Once youve made the changes, sit back and ponder some deep thoughts: youve got two programs, both seem to do the same thing. Is one approach somehow preferable to the other? Which one more closely resembles reality i.e. the trafc lights in your town? 4. Now that youve got a trafc light program with different turtles for each light, perhaps the visibility / invisibilty trick wasnt such a great idea. If you watch the trafc lights, they turn on and off but when theyre off they are still there, perhaps just a darker color. Modify the program now so that the lights dont disappear: they are either on, or off. But when theyre off, theyre still visible.

10.5. Glossary

141

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

5. Your trafc light controller program has been patented, and youre about to become seriously rich. But your new client needs a change. They want four states in their state machine: Green, then Green and Orange together, then Orange only, and then Red. Additionally, they want different times spent in each state. The machine should spend 3 seconds in the Green state, followed by one second in the Green+Orange state, then one second in the Orange state, and then 2 seconds in the Red state. Change the logic in the state machine. 6. If you dont know how tennis scoring works, ask a friend or consult Wikipedia. A single game in tennis between player A and player B always has a score. We want to think about the state of the score as a state machine. The game starts in state (0, 0), meaning neither player has any score yet. Well assume the rst element in this pair is the score for player A. If player A wins the rst point, the score becomes (15, 0). If B wins the rst point, the state becomes (0, 15). Below are the rst few states and transitions for a state diagram. In this diagram, each state has two possible outcomes (A wins the next point, or B does), and the uppermost arrow is always the transition that happens when A wins the point. Complete the diagram, showing all transitions and all states. (Hint: there are twenty states, if you include the duece state, the advantage states, and the A wins and B wins states in your diagram.)

142

Chapter 10. Event-Driven Programming

CHAPTER

ELEVEN

LISTS
A list is an ordered collection of values. The values that make up a list are called its elements, or its items. We will use the term element or item to mean the same thing. Lists are similar to strings, which are ordered collections of characters, except that the elements of a list can be any type. Lists and strings and other collections that maintain the order of their items are called sequences.

11.1 List values


There are several ways to create a new list; the simplest is to enclose the elements in square brackets ( [ and ]):
1 2

ps = [10, 20, 30, 40] qs = ["spam", "bungee", "swallow"]

The rst example is a list of four integers. The second is a list of three strings. The elements of a list dont have to be the same type. The following list contains a string, a oat, an integer, and (amazingly) another list:
1

zs = ["hello", 2.0, 5, [10, 20]]

A list within another list is said to be nested. Finally, a list with no elements is called an empty list, and is denoted []. We have already seen that we can assign list values to variables or pass lists as parameters to functions:
1 2 3 4 5

>>> vocabulary = ["apple", "cheese", "dog"] >>> numbers = [17, 123] >>> an_empty_list = [] >>> print(vocabulary, numbers, an_empty_list) [apple, cheese, dog] [17, 123] []

143

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

11.2 Accessing elements


The syntax for accessing the elements of a list is the same as the syntax for accessing the characters of a stringthe index operator ( [] not to be confused with an empty list). The expression inside the brackets species the index. Remember that the indices start at 0:
>>> numbers[0] 17

Any integer expression can be used as an index:


>>> numbers[9-8] 5 >>> numbers[1.0] Traceback (most recent call last): File "<interactive input>", line 1, in <module> TypeError: list indices must be integers, not float

If you try to access or assign to an element that does not exist, you get a runtime error:
>>> numbers[2] Traceback (most recent call last): File "<interactive input>", line 1, in <module> IndexError: list index out of range

It is common to use a loop variable as a list index.


1 2 3 4

horsemen = ["war", "famine", "pestilence", "death"] for i in [0, 1, 2, 3]: print(horsemen[i])

Each time through the loop, the variable i is used as an index into the list, printing the ith element. This pattern of computation is called a list traversal. The above sample doesnt need or use the index i for anything besides getting the items from the list, so this more direct version where the for loop gets the items might be preferred:
1 2 3 4

horsemen = ["war", "famine", "pestilence", "death"] for h in horsemen: print(h)

11.3 List length


The function len returns the length of a list, which is equal to the number of its elements. If you are going to use an integer index to access the list, it is a good idea to use this value as the upper bound of a loop instead of a constant. That way, if the size of the list changes, you wont have to go through the program changing all the loops; they will work correctly for any size list: 144 Chapter 11. Lists

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2 3 4

horsemen = ["war", "famine", "pestilence", "death"] for i in range(len(horsemen)): print(horsemen[i])

The last time the body of the loop is executed, i is len(horsemen) - 1, which is the index of the last element. (But the version without the index looks even better now!) Although a list can contain another list, the nested list still counts as a single element in its parent list. The length of this list is 4:
>>> len([car makers, 1, [Ford, Toyota, BMW], [1, 2, 3]]) 4

11.4 List membership


in and not in are boolean operators that test membership in a sequence. We used them previously with strings, but they also work with lists and other sequences:
>>> horsemen = [war, famine, pestilence, death] >>> pestilence in horsemen True >>> debauchery in horsemen False >>> debauchery not in horsemen True

Using this produces a more elegant version of the nested loop program we previously used to count the number of students doing Computer Science in the section Nested Loops for Nested Data:
1 2 3 4 5 6 7 8 9 10 11 12 13 14

students = [ ("John", ["CompSci", "Physics"]), ("Vusi", ["Maths", "CompSci", "Stats"]), ("Jess", ["CompSci", "Accounting", "Economics", "Management"]), ("Sarah", ["InfSys", "Accounting", "Economics", "CommLaw"]), ("Zuki", ["Sociology", "Economics", "Law", "Stats", "Music"])] # Count how many students are taking CompSci counter = 0 for (name, subjects) in students: if "CompSci" in subjects: counter += 1 print("The number of students taking CompSci is", counter)

11.4. List membership

145

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

11.5 List operations


The + operator concatenates lists:
>>> >>> >>> >>> [1, a = [1, 2, 3] b = [4, 5, 6] c = a + b c 2, 3, 4, 5, 6]

Similarly, the * operator repeats a list a given number of times:


>>> [0, >>> [1, [0] * 4 0, 0, 0] [1, 2, 3] * 3 2, 3, 1, 2, 3, 1, 2, 3]

The rst example repeats [0] four times. The second example repeats the list [1, 2, 3] three times.

11.6 List slices


The slice operations we saw previously with strings lets us work with sublists:
>>> a_list = [a, b, c, d, e, f] >>> a_list[1:3] [b, c] >>> a_list[:4] [a, b, c, d] >>> a_list[3:] [d, e, f] >>> a_list[:] [a, b, c, d, e, f]

11.7 Lists are mutable


Unlike strings, lists are mutable, which means we can change their elements. Using the bracket operator on the left side of an assignment, we can update one of the elements:
>>> fruit = ["banana", "apple", "quince"] >>> fruit[0] = "pear" >>> fruit[2] = "orange" >>> fruit [pear, apple, orange]

The bracket operator applied to a list can appear anywhere in an expression. When it appears on the left side of an assignment, it changes one of the elements in the list, so the rst element

146

Chapter 11. Lists

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

of fruit has been changed from banana to pear, and the last from quince to orange. An assignment to an element of a list is called item assignment. Item assignment does not work for strings:
>>> my_string = TEST >>> my_string[2] = X Traceback (most recent call last): File "<interactive input>", line 1, in <module> TypeError: str object does not support item assignment

but it does for lists:


>>> my_list = [T, E, S, T] >>> my_list[2] = X >>> my_list [T, E, X, T]

With the slice operator we can update a whole sublist at once:


>>> a_list = [a, b, c, d, e, f] >>> a_list[1:3] = [x, y] >>> a_list [a, x, y, d, e, f]

We can also remove elements from a list by assigning the empty list to them:
>>> a_list = [a, b, c, d, e, f] >>> a_list[1:3] = [] >>> a_list [a, d, e, f]

And we can add elements to a list by squeezing them into an empty slice at the desired location:
>>> a_list = [a, d, f] >>> a_list[1:1] = [b, c] >>> a_list [a, b, c, d, f] >>> a_list[4:4] = [e] >>> a_list [a, b, c, d, e, f]

11.8 List deletion


Using slices to delete list elements can be error-prone. Python provides an alternative that is more readable. The del statement removes an element from a list:
>>> a = [one, two, three] >>> del a[1] >>> a [one, three]

11.8. List deletion

147

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

As you might expect, del causes a runtime error if the index is out of range. You can also use del with a slice to delete a sublist:
>>> a_list = [a, b, c, d, e, f] >>> del a_list[1:5] >>> a_list [a, f]

As usual, the sublist selected by slice contains all the elements up to, but not including, the second index.

11.9 Objects and references


If we execute these assignment statements,
1 2

a = "banana" b = "banana"

we know that a and b will refer to a string ojbect with the letters "banana". But we dont know yet whether they point to the same string object. There are two possible ways the Python interpreter could arrange its memory:

In one case, a and b refer to two different objects that have the same value. In the second case, they refer to the same object. We can test whether two names refer to the same object using the is operator:
>>> a is b True

This tells us that both a and b refer to the same object, and that it is the second of the two state snapshots that accurately describes the relationship. Since strings are immutable, Python optimizes resources by making two names that refer to the same string value refer to the same object. This is not the case with lists:
>>> a >>> b >>> a True >>> a False = [1, 2, 3] = [1, 2, 3] == b is b

The state snapshot here looks like this:

148

Chapter 11. Lists

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

a and b have the same value but do not refer to the same object.

11.10 Aliasing
Since variables refer to objects, if we assign one variable to another, both variables refer to the same object:
>>> a = [1, 2, 3] >>> b = a >>> a is b True

In this case, the state snapshot looks like this:

Because the same list has two different names, a and b, we say that it is aliased. Changes made with one alias affect the other:
>>> b[0] = 5 >>> a [5, 2, 3]

Although this behavior can be useful, it is sometimes unexpected or undesirable. In general, it is safer to avoid aliasing when you are working with mutable objects(i.e. lists at this point in our textbook, but well meet more mutable objects as we cover classes and objects, dictionaries and sets). Of course, for immutable objects (i.e. strings, tuples), theres no problem - it is just not possible to change something and get a surprise when you access an alias name. Thats why Python is free to alias strings (and any other immutable kinds of data) when it sees an opportunity to economize.

11.11 Cloning lists


If we want to modify a list and also keep a copy of the original, we need to be able to make a copy of the list itself, not just the reference. This process is sometimes called cloning, to avoid the ambiguity of the word copy. The easiest way to clone a list is to use the slice operator:
>>> a = [1, 2, 3] >>> b = a[:]

11.10. Aliasing

149

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

>>> b [1, 2, 3]

Taking any slice of a creates a new list. In this case the slice happens to consist of the whole list. So now the relationship is like this:

Now we are free to make changes to b without worrying that well inadvertently be changing a:
>>> b[0] = 5 >>> a [1, 2, 3]

11.12 Lists and for loops


The for loop also works with lists, as weve already seen. The generalized syntax of a for loop is:
for VARIABLE in LIST: BODY

So, as weve seen


1 2 3

friends = ["Joe", "Amy", "Brad", "Angelina", "Zuki", "Thandi", "Paris"] for friend in friends: print(friend)

It almost reads like English: For (every) friend in (the list of) friends, print (the name of the) friend. Any list expression can be used in a for loop:
1 2 3 4 5 6

for number in range(20): if number % 3 == 0: print(number) for fruit in ["banana", "apple", "quince"]: print("I like to eat " + fruit + "s!")

The rst example prints all the multiples of 3 between 0 and 19. The second example expresses enthusiasm for various fruits. Since lists are mutable, we often want to traverse a list, changing each of its elements. The following squares all the numbers in the list xs:
1 2

xs = [1, 2, 3, 4, 5]

150

Chapter 11. Lists

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

3 4

for i in range(len(xs)): xs[i] = xs[i]**2

Take a moment to think about range(len(xs)) until you understand how it works. In this example we are interested in both the value of an item, (we want to square that value), and its index (so that we can assign the new value to that position). This pattern is common enough that Python provides a nicer way to implement it:
1 2 3 4

xs = [1, 2, 3, 4, 5] for (i, val) in enumerate(xs): xs[i] = val**2

enumerate generates pairs of both (index, value) during the list traversal. Try this next example to see more clearly how enumerate works:
1 2

for (i, v) in enumerate([banana, apple, pear, lemon]): print(i, v) 0 1 2 3 banana apple pear lemon

11.13 List parameters


Passing a list as an argument actually passes a reference to the list, not a copy or clone of the list. So parameter passing creates an alias for you: the caller has one variable referencing the list, and the called function has an alias, but there is only one underlying list object. For example, the function below takes a list as an argument and multiplies each element in the list by 2:
1 2 3 4

def double_stuff(a_list): """ Overwrite each element in a_list with double its value. """ for (idx, val) in enumerate(a_list): a_list[idx] = 2 * val

If we add the following onto our script:


1 2 3

things = [2, 5, 9] double_stuff(things) print(things)

When we run it well get:


[4, 10, 18]

In the function above, the parameter a_list and the variable things are aliases for the same object. So before any changes to the elements in the list, the state snapshot looks like this:

11.13. List parameters

151

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Since the list object is shared by two frames, we drew it between them. If a function modies the items of a list parameter, the caller sees the change. Use the Python visualizer! Weve already mentioned the Python visualizer at http://netserv.ict.ru.ac.za/python3_viz. It is a very useful tool for building a good understanding of references, aliases, assignments, and passing arguments to functions. Pay special attention to cases where you clone a list or have two separate lists, and cases where there is only one underlying list, but more than one variable is aliased to reference the list.

11.14 List methods


The dot operator can also be used to access built-in methods of list objects. Well start with the most useful method for adding something onto the end of an existing list...
>>> >>> >>> >>> >>> >>> [5, >>> mylist = [] mylist.append(5) mylist.append(27) mylist.append(3) mylist.append(12) mylist 27, 3, 12]

append is a list method which adds the argument passed to it to the end of the list. Well use it heavily when were creating new lists. Continuing with this example, we show several other list methods:
>>> mylist.insert(1, 12) # insert 12 at pos 1, shift other items up >>> mylist [5, 12, 27, 3, 12] >>> mylist.count(12) # how many times is 12 in mylist? 2 >>> mylist.extend([5, 9, 5, 11]) # put whole list onto end of mylist >>> mylist [5, 12, 27, 3, 12, 5, 9, 5, 11]) >>> mylist.index(9) # find index of first 9 in mylist 6 >>> mylist.reverse() >>> mylist [11, 5, 9, 5, 12, 3, 27, 12, 5]

152

Chapter 11. Lists

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

>>> >>> [3, >>> >>> [3, >>>

mylist.sort() mylist 5, 5, 5, 9, 11, 12, 12, 27] mylist.remove(12) mylist 5, 5, 5, 9, 11, 12, 27]

# remove the first 12 in the list

Experiment and play with the list methods shown here, and read their documentation until you feel condent that you understand how they work.

11.15 Pure functions and modiers


Functions which take lists as arguments and change them during execution are called modiers and the changes they make are called side effects. A pure function does not produce side effects. It communicates with the calling program only through parameters, which it does not modify, and a return value. Here is double_stuff written as a pure function:
1 2 3 4 5 6 7 8 9 10

def double_stuff(a_list): """ Return a new list in which contains doubles of the elements in a_list. """ new_list = [] for value in a_list: new_elem = 2 * value new_list.append(new_elem) return new_list

This version of double_stuff does not change its arguments:


>>> >>> >>> [2, >>> [4, things = [2, 5, 9] xs = double_stuff(things) things 5, 9] xs 10, 18]

An early rule we saw for assignment said rst evaluate the right hand side, then assign the resulting value to the variable. So it is quite safe to assign the function result to the same variable that was passed to the function:
>>> >>> >>> [4, things = [2, 5, 9] things = double_stuff(things) things 10, 18]

Which style is better? 11.15. Pure functions and modiers 153

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Anything that can be done with modiers can also be done with pure functions. In fact, some programming languages only allow pure functions. There is some evidence that programs that use pure functions are faster to develop and less error-prone than programs that use modiers. Nevertheless, modiers are convenient at times, and in some cases, functional programs are less efcient. In general, we recommend that you write pure functions whenever it is reasonable to do so and resort to modiers only if there is a compelling advantage. This approach might be called a functional programming style.

11.16 Functions that produce lists


The pure version of double_stuff above made use of an important pattern for your toolbox. Whenever you need to write a function that creates and returns a list, the pattern is usually:
1 2 3 4 5

initialize a result variable to be an empty list loop create a new element append it to result return the result

Let us show another use of this pattern. Assume you already have a function is_prime(x) that can test if x is prime. Write a function to return a list of all prime numbers less than n:
1 2 3 4 5 6 7

def primes_lessthan(n): Return a list of all prime numbers less than n. result = [] for i in range(2, n): if is_prime(i): result.append(i) return result

11.17 Strings and lists


Two of the most useful methods on strings involve conversion to and from lists of substrings. The split method (which weve already seen) breaks a string into a list of words. By default, any number of whitespace characters is considered a word boundary:
>>> song = "The rain in Spain..." >>> wds = song.split() >>> wds [The, rain, in, Spain...]

An optional argument called a delimiter can be used to specify which string to use as the boundary marker between substrings. The following example uses the string ai as the delimiter:

154

Chapter 11. Lists

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

>>> song.split(ai) [The r, n in Sp, n...]

Notice that the delimiter doesnt appear in the result. The inverse of the split method is join. You choose a desired separator string, (often called the glue) and join the list with the glue between each of the elements:
>>> glue = ; >>> s = glue.join(wds) >>> s The;rain;in;Spain...

The list that you glue together (wds in this example) is not modied. Also, as these next examples show, you can use empty glue or multi-character strings as glue:
>>> --- . join(wds) The --- rain --- in --- Spain... >>> . join(wds) TheraininSpain...

11.18 list and range


Python has a built-in type conversion function called list that tries to turn whatever you give it into a list.
>>> xs = list("Crunchy Frog") >>> xs [C, r, u, n, c, h, y, , F, r, o, g] >>> .join(xs) Crunchy Frog

One particular feature of range is that it doesnt instantly compute all its values: it puts off the computation, and does it on demand, or lazily. Well say that it gives a promise to produce the values when they are needed. This is very convenient if your computation shortcircuits a search and returns early, as in this case:
1 2 3 4 5 6 7 8 9 10 11

def f(n): """ Find the first positive integer between 101 and less than n that is divisible by 21 """ for i in range(101, n): if (i % 21 == 0): return i

test(f(110), 105) test(f(1000000000), 105)

11.18. list and range

155

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

In the second test, if range were to eagerly go about building a list with all those elements, you would soon exhaust your computers available memory and crash the program. But it is cleverer than that! This computation works just ne, because the range object is just a promise to produce the elements if and when they are needed. Once the condition in the if becomes true, no further elements are generated, and the function returns. (Note: Before Python 3, range was not lazy. If you use an earlier versions of Python, YMMV!) YMMV: Your Mileage May Vary The acronym YMMV stands for your mileage may vary. American car advertisements often quoted fuel consumption gures for cars, e.g. that they would get 28 miles per gallon. But this always had to be accompanied by legal small-print warning the reader that they might not get the same. The term YMMV is now used idiomatically to mean your results may differ, e.g. The battery life on this phone is 3 days, but YMMV. Youll sometimes nd the lazy range wrapped in a call to list. This forces Python to turn the lazy promise into an actual list:
>>> range(10) # create a lazy promise range(0, 10) >>> list(range(10)) # Call in the promise, to produce a list. [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]

11.19 Nested lists


A nested list is a list that appears as an element in another list. In this list, the element with index 3 is a nested list:
>>> nested = ["hello", 2.0, 5, [10, 20]]

If we print(nested[3]), we get [10, 20]. To extract an element from the nested list, we can proceed in two steps:
>>> elem = nested[3] >>> elem[0] 10

Or we can combine them:


>>> nested[3][1] 20

Bracket operators evaluate from left to right, so this expression gets the three-th element of nested and extracts the one-th element from it.

156

Chapter 11. Lists

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

11.20 Matrices
Nested lists are often used to represent matrices. For example, the matrix:

might be represented as:


>>> mx = [[1, 2, 3], [4, 5, 6], [7, 8, 9]]

mx is a list with three elements, where each element is a row of the matrix. We can select an entire row from the matrix in the usual way:
>>> mx[1] [4, 5, 6]

Or we can extract a single element from the matrix using the double-index form:
>>> mx[1][2] 6

The rst index selects the row, and the second index selects the column. Although this way of representing matrices is common, it is not the only possibility. A small variation is to use a list of columns instead of a list of rows. Later we will see a more radical alternative using a dictionary.

11.21 Glossary
aliases Multiple variables that contain references to the same object. clone To create a new object that has the same value as an existing object. Copying a reference to an object creates an alias but doesnt clone the object. delimiter A character or string used to indicate where a string should be split. element One of the values in a list (or other sequence). The bracket operator selects elements of a list. Also called item. index An integer value that indicates the position of an item in a list. Indexes start from 0. item See element. list A collection of values, each in a xed position within the list. Like other types str, int, float, etc. there is also a list type-converter function that tries to turn whatever argument you give it into a list. list traversal The sequential accessing of each element in a list. modier A function which changes its arguments inside the function body. Only mutable types can be changed by modiers. 11.20. Matrices 157

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

mutable data type A data type in which the elements can be modied. All mutable types are compound types. Lists are mutable data types; strings are not. nested list A list that is an element of another list. object A thing to which a variable can refer. pattern A sequence of statements, or a style of coding something that has general applicability in a number of different situations. Part of becoming a mature Computer Scientist is to learn and establish the patterns and algorithms that form your toolkit. Patterns often correspond to your mental chunking. promise An object that promises to do some work or deliver some values if theyre eventually needed, but it lazily puts off doing the work immediately. Calling range produces a promise. pure function A function which has no side effects. Pure functions only make changes to the calling program through their return values. sequence Any of the data types that consist of an ordered collection of elements, with each element identied by an index. side effect A change in the state of a program made by calling a function. Side effects can only be produced by modiers. step size The interval between successive elements of a linear sequence. The third (and optional argument) to the range function is called the step size. If not specied, it defaults to 1.

11.22 Exercises
1. What is the Python interpreters response to the following?
>>> list(range(10, 0, -2))

The three arguments to the range function are start, stop, and step, respectively. In this example, start is greater than stop. What happens if start < stop and step < 0? Write a rule for the relationships among start, stop, and step. 2. Consider this fragment of code:
1 2 3 4 5

import turtle tess = turtle.Turtle() alex = tess alex.color("hotpink")

Does this fragment create one or two turtle instances? Does setting the colour of alex also change the colour of tess? Explain in detail. 3. Draw a state snapshot for a and b before and after the third line of the following python code is executed:

158

Chapter 11. Lists

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2 3

a = [1, 2, 3] b = a[:] b[0] = 5

4. What will be the output of the following program?


1 2 3 4 5

this = [I, am, not, a, crook] that = [I, am, not, a, crook] print("Test 1: {0}".format(this is that)) that = this print("Test 2: {0}".format(this is that))

Provide a detailed explaination of the results. 5. Lists can be used to represent mathematical vectors. In this exercise and several that follow you will write functions to perform standard operations on vectors. Create a script named vectors.py and write Python code to pass the tests in each case. Write a function add_vectors(u, v) that takes two lists of numbers of the same length, and returns a new list containing the sums of the corresponding elements of each:
1 2 3

test(add_vectors([1, 1], [1, 1]), [2, 2]) test(add_vectors([1, 2], [1, 4]), [2, 6]) test(add_vectors([1, 2, 1], [1, 4, 3]), [2, 6, 4])

6. Write a function scalar_mult(s, v) that takes a number, s, and a list, v and returns the scalar multiple of v by s. :
1 2 3

test(scalar_mult(5, [1, 2]), [5, 10]) test(scalar_mult(3, [1, 0, -1]), [3, 0, -3]) test(scalar_mult(7, [3, 0, 5, 11, 2]), [21, 0, 35, 77, 14])

7. Write a function dot_product(u, v) that takes two lists of numbers of the same length, and returns the sum of the products of the corresponding elements of each (the dot_product).
1 2 3

test(dot_product([1, 1], [1, 1]), 2) test(dot_product([1, 2], [1, 4]), 9) test(dot_product([1, 2, 1], [1, 4, 3]), 12)

8. Extra challenge for the mathematically inclined: Write a function cross_product(u, v) that takes two lists of numbers of length 3 and returns their cross product. You should write your own tests. 9. Describe the relationship between .join(song.split()) and song in the fragment of code below. Are they the same for all strings assigned to song? When would they be different? :
1

song = "The rain in Spain..."

10. Write a function replace(s, old, new) that replaces all occurrences of old with new in a string s:

11.22. Exercises

159

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2 3 4 5 6 7 8

test(replace(Mississippi, i, I), MIssIssIppI) s = I love spom! Spom is my favorite food. Spom, spom, yum! test(replace(s, om, am), I love spam! Spam is my favorite food. Spam, spam, yum!) test(replace(s, o, a), I lave spam! Spam is my favarite faad. Spam, spam, yum!)

Hint: use the split and join methods. 11. Cut and paste the following functions to a script called seqtools.py.
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

def def def def def def def def def def

make_empty(seq): pass insert_at_end(val, seq): pass insert_in_front(val, seq): pass index_of(val, seq, start=0): pass remove_at(index, seq): pass remove_val(val, seq): pass remove_all(val, seq): pass count(val, seq): pass reverse(seq): pass sort_sequence(seq): pass

def testsuite(): test(make_empty([1, 2, 3, 4]), []) test(make_empty((a, b, c)), ()) test(make_empty("No, not me!"), ) test(insert_at_end(5, [1, 3, 4, 6]), [1, 3, 4, 6, 5]) test(insert_at_end(x, abc), abcx) test(insert_at_end(5, (1, 3, 4, 6)), (1, 3, 4, 6, 5)) test(insert_in_front(5, [1, 3, 4, 6]), [5, 1, 3, 4, 6]) test(insert_in_front(5, (1, 3, 4, 6)), (5, 1, 3, 4, 6)) test(insert_in_front(x, abc), xabc) test(index_of(9, [1, 7, 11, 9, 10]), 3) test(index_of(5, (1, 2, 4, 5, 6, 10, 5, 5)), 3) test(index_of(5, (1, 2, 4, 5, 6, 10, 5, 5), 4), 6) test(index_of(y, happy birthday), 4) test(ndex_of(banana,[apple,banana,cherry,date]),1) test(index_of(5, [2, 3, 4]), -1) test(index_of(b, [apple,banana,cherry,date]),-1) test(remove_at(3, [1, 7, 11, 9, 10]), [1, 7, 11, 10]) test(remove_at(5, (1,4,6,7,0,9,3,5)), (1,4,6,7,0,3,5)) test(remove_at(2, "Yomrktown"), Yorktown) test(remove_val(11, [1, 7, 11, 9, 10]), [1, 7, 9, 10]) test(remove_val(15, (1, 15, 11, 4, 9)), (1, 11, 4, 9))

160

Chapter 11. Lists

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56

test(remove_val(what,(who,what,when,why,how)), (who, when, why, how)) test(remove_all(11, [1,7,11,9,11,10,2,11]),[1,7,9,10,2]) test(remove_all(i, Mississippi), Msssspp) test(count(5, (1, 5, 3, 7, 5, 8, 5)), 3) test(count(s, Mississippi), 4) test(count((1, 2), [1, 5, (1, 2), 7, (1, 2), 8, 5]), 2) test(reverse([1, 2, 3, 4, 5]), [5, 4, 3, 2, 1]) test(reverse((shoe,my,buckle,2,1)), (1,2,buckle,my,shoe)) test(reverse(Python), nohtyP) test(sort_sequence([3, 4, 6, 7, 8, 2]), [2, 3, 4, 6, 7, 8]) test(sort_sequence((3, 4, 6, 7, 8, 2)), (2, 3, 4, 6, 7, 8)) test(sort_sequence("nothappy"), ahnoppty)

As usual, add your test function and work on each of these one at a time until they pass all the tests. But do you really want to do this? Disclaimer. These exercises illustrate nicely that the sequence abstraction is general, (because slicing, indexing, and concatenation are so general), so it is possible to write general functions that work over all sequence types. A nice lesson indeed about generalization! Another view is that tuples are different from lists and strings precisely because you want to think about them very differently. It usually doesnt make sense to sort the elds of the julia tuple we saw in the chapter on tuples, or to cut bits out or insert bits into the middle, even if Python lets you do so! Tuple elds get their meaning from their position in the tuple. Dont mess with that. Use lists for many things of the same type, like an enrollment of many students for a course. Use tuples for elds of different types that make up a compound record. 12. Suppose you want to swap around the values in two variables. You decide to factor this out into a reusable function, and write this code:
1 2 3 4 5 6 7 8

def swap(x, y): # incorrect version print("before swap statement: x:", x, "y:", y) (x, y) = (y, x) print("after swap statement: x:", x, "y:", y) a = ["This", "is", "fun"] b = [2,3,4] print("before swap function call: a:", a, "b:", b)

11.22. Exercises

161

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

9 10

swap(a, b) print("after swap function call: a:", a, "b:", b)

Run this program and describe the results. Oops! So it didnt do what you intended! Explain why not. Using a Python visualizer like the one at http://netserv.ict.ru.ac.za/python3_viz may help you build a good conceptual model of what is going on. What will be the values of a and b after the call to swap?

162

Chapter 11. Lists

CHAPTER

TWELVE

MODULES
A module is a le containing Python denitions and statements intended for use in other Python programs. There are many Python modules that come with Python as part of the standard library. We have seen at least two of these already, the turtle module and the string module. We have also shown you how to access help. The help system contains a listing of all the standard modules that are available with Python. Play with help!

12.1 Random numbers


We often want to use random numbers in programs: here are a few typical uses: To play a game of chance where the computer needs to throw some dice, pick a number, or ip a coin, To shufe a deck of playing cards randomly, To randomly allow a new enemy spaceship to appear and shoot at you, To simulate possible rainfall when we make a computerized model for estimating the environmental impact of builing a dam, For encrypting your banking session on the Internet. Python provides a module random that helps with tasks like this. You can look it up using help, but here are the key things well do with it:
1 2 3 4 5 6 7

import random # create a black box object that generates random numbers rng = random.Random() dice_throw = rng.randrange(1,7) # return an int, one of 1,2,3,4,5,6 delay_in_seconds = rng.random() * 5.0

The randrange method call generates generates an integer between its lower and upper argument, using the same semantics as range so the lower bound is included, but the upper bound is excluded. All the values have an equal probability of occurring (i.e. the results are 163

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

uniformly distributed). Like range, randrange can also take an optional step argument. So lets assume we needed a random odd number less than 100, we could say:
1

r_odd = rng.randrange(1, 100, 2)

Other methods can also generate other distributions e.g. a bell-shaped, or normal distribution might be more appropriate for estimating seasonal rainfall, or the concentration of a compound in the body after taking a dose of medicine. The random() method returns a oating point number in the interval [0.0, 1.0) the square bracket means closed interval on the left and the round parenthesis means open interval on the right. In other words, 0.0 is possible, but all returned numbers will be strictly less than 1.0. It is usual to scale the results after calling this method, to get them into an interval suitable for your application. In the case shown here, weve converted the result of the method call to a number in the interval [0.0, 5.0). Once more, these are uniformly distributed numbers numbers close to 0 are just as likely to occur as numbers close to 0.5, or numbers close to 1.0. This example shows how to shufe a list. (shufe cannot work directly with a lazy promise, so notice that we had to convert the range object using the list type converter rst.)
1 2 3

cards = list(range(52)) rng.shuffle(cards)

# generate ints [0 .. 51] # representing a pack of cards. # shuffle the pack

12.1.1 Repeatability and Testing


Random number generators are based on a deterministic algorithm repeatable and predictable. So theyre called pseudo-random generators they are not genuinely random. They start with a seed value. Each time you ask for another random number, youll get one based on the current seed attribute, and the state of the seed (which is one of the attributes of the generator) will be updated. For debugging and for writing unit tests, it is convenient to have repeatability programs that do the same thing every time they are run. We can arrange this by forcing the random number generator to be initialized with a known seed every time. (Often this is only wanted during testing playing a game of cards where the shufed deck was always in the same order as last time you played would get boring very rapidly!)
1

drng = random.Random(123)

# create generator with known starting state

This alternative way of creating a random number generator gives an explicit seed value to the object. Without this argument, the system probably uses something based on the time. So grabbing some random numbers from drng today will give you precisely the same random sequence as it will tomorrow!

164

Chapter 12. Modules

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

12.1.2 Picking balls from bags, throwing dice, shufing a pack of cards
Here is an example to generate a list containing n random ints between a lower and an upper bound:
1 2 3 4 5 6 7 8 9 10 11 12

import random def make_random_ints(num, lower_bound, upper_bound): """ Generate a list containing num random ints between lower_bound and upper_bound. upper_bound is an open bound. """ result = [] rng = random.Random() for i in range(num): result.append(rng.randrange(lower_bound, upper_bound)) return result >>> make_random_ints(5, 1, 13) [8, 1, 8, 5, 6] # pick 5 random month numbers

Notice that we got a duplicate in the result. Often this is wanted, e.g. if we throw a die ve times, we would expect some duplicates. But what if you dont want duplicates? If you wanted 5 distinct months, then this algorithm is wrong. In this case a good algorithm is to generate the list of possibilities, shufe it, and slice off the number of elements you want:
1 2 3

xs = list(range(1,13)) random.shuffle(xs) result = xs[:5]

# make list 1..12. There are no duplicates # shuffle the list # take the first five elements.

In statistics courses, the rst case allowing duplicates is usually described as pulling balls out of a bag with replacement you put the drawn ball back in each time, so it can occur again. The latter case, with no duplicates, is usually described as pulling balls out of the bag without replacement. Once the ball is drawn, it doesnt go back to be drawn again. TV lotto games work like this. The second shufe and slice algorithm would not be so great if you only wanted a few elements, but from a very large domain. Suppose I wanted ve numbers between one and ten million, without duplicates. Generating a list of ten million items, shufing it, and then slicing off the rst ve would be a performance disaster! So let us have another try:
1 2 3 4 5 6 7 8

import random def make_random_ints_no_dups(num, lower_bound, upper_bound): """ Generate a list containing num random ints between lower_bound and upper_bound. upper_bound is an open bound. The result list cannot contain duplicates. """

12.1. Random numbers

165

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

9 10 11 12 13 14 15 16 17 18 19 20 21

result = [] rng = random.Random() for i in range(num): while True: candidate = rng.randrange(lower_bound, upper_bound) if candidate not in result: break; result.append(candidate) return result xs = make_random_ints_no_dups(5, 1, 10000000) print(xs)

This agreeably produces 5 random numbers, without duplicates:


[3344629, 1735163, 9433892, 1081511, 4923270]

Even this function has its pitfalls. Can you spot what is going to happen in this case?
1

xs = make_random_ints_no_dups(10, 1, 6)

12.2 The time module


As we start to work with more sophisticated algorithms and bigger programs, at natural concern is is our code efcient? One way to experiment is to time how long various operations take. The time module has a function called clock that is recommended for this purpose. Whenever clock is called, it returns a oating point number representing how many seconds have elapsed since your program started running. The way to use it is to call clock and assign the result to a variable, say t0, just before you start executing the code you want to measure. Then after execution, call clock again, (this time well save the result in variable t1). The difference t1-t0 is the time elapsed, and is a measure of how fast your program is running. Lets try a small example. Python has a built-in sum function that can sum the elements in a list. We can also write our own. How do we think they would compare for speed? Well try to do the summation of a list [0, 1, 2 ...] in both cases, and compare the results:
1 2 3 4 5 6 7 8 9 10 11

import time def do_my_sum(xs): sum = 0 for v in xs: sum += v return sum sz = 10000000 # lets have 10 million elements in the list testdata = range(sz)

166

Chapter 12. Modules

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

12 13 14 15 16 17 18 19 20 21 22

t0 = time.clock() my_result = do_my_sum(testdata) t1 = time.clock() print("my_result = {0} (time taken = {1:.4f} seconds)" .format(my_result, t1-t0)) t2 = time.clock() their_result = sum(testdata) t3 = time.clock() print("their_result = {0} (time taken = {1:.4f} seconds)" .format(their_result, t3-t2))

On a reasonably modest laptop, we get these results:


my_sum = 49999995000000 (time taken = 1.5567 seconds) their_sum = 49999995000000 (time taken = 0.9897 seconds)

So our function runs about 57% slower than their built-in one. And summing up ten million elements in under a second is not too shabby!

12.3 The math module


The math module contains the kinds of mathematical functions youd typically nd on your calculator (sin, cos, sqrt, asin, log, log10) and some mathematical constants like pi and e:
>>> import math >>> math.pi # constant attribute for pi 3.141592653589793 >>> math.e # constant natural log base 2.718281828459045 >>> math.sqrt(2.0) # square root function 1.4142135623730951 >>> math.radians(90) # convert 90 degrees to radians 1.5707963267948966 >>> math.sin(math.radians(90)) # find sin of 90 degrees. 1.0 >>> math.asin(1.0) * 2 # Double arcsin of 1.0 to get pi 3.141592653589793

Like almost all other programming languages, angles are expressed in radians rather than degrees. There are two functions radians and degrees to convert between the two popular ways of measuring angles. Notice another difference between this module and our use of random and turtle: in random and turtle we create objects and we call methods on the object. This is because objects have state a turtle has a colour, a position, a heading, etc., and every random number generator has a seed value that determines its next result.

12.3. The math module

167

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Mathematical functions are pure and dont have any state calculating the square root of 2.0 doesnt depend on any kind of state or history about what happened in the past. So the functions are not methods of an object they are simply functions that are grouped together in a module called math.

12.4 Creating your own modules


All we need to do to create our own modules is to save our script as a le with a .py extension on the lename. Suppose, for example, this script is saved as a le named seqtools.py:
1 2

def remove_at(pos, seq): return seq[:pos] + seq[pos+1:]

We can now use our module, both in scripts we write, or in the interactive Python interpreter. To do so, we must rst import the module.
>>> import seqtools >>> s = "A string!" >>> seqtools.remove_at(4, s) A sting!

We do not include the .py le extension when importing. Python expects the le names of Python modules to end in .py, so the le extention is not included in the import statement. The use of modules makes it possible to break up very large programs into managable sized parts, and to keep related parts together.

12.5 Namespaces
A namespace is a collection of identiers that belong to a module, or to a function, (and as we will see soon, in classes too). Generally, we like a namespace to hold related things, e.g. all the math functions, or all the typical things wed do with random numbers. Each module has its own namespace, so we can use the same identier name in multiple modules without causing an identication problem.
1 2 3 4

# module1.py question = "What is the meaning of Life, the Universe, and Everything?" answer = 42 # module2.py question = "What is your quest?" answer = "To seek the holy grail."

1 2 3 4

We can now import both modules and access question and answer in each:

168

Chapter 12. Modules

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2 3 4 5 6 7

import module1 import module2 print(module1.question) print(module2.question) print(module1.answer) print(module2.answer)

will output the following:


What is the meaning of Life, the Universe, and Everything? What is your quest? 42 To seek the holy grail.

Functions also have their own namespaces:


1 2 3 4 5 6 7 8 9 10 11 12 13 14

def f(): n = 7 print("printing n inside of f:", n) def g(): n = 42 print("printing n inside of g:", n) n = 11 print("printing n before calling f:", n) f() print("printing n after calling f:", n) g() print("printing n after calling g:", n)

Running this program produces the following output:


printing printing printing printing printing n n n n n before calling f: 11 inside of f: 7 after calling f: 11 inside of g: 42 after calling g: 11

The three ns here do not collide since they are each in a different namespace they are three names for three different variables, just like there might be three different instances of people, all called Bruce. Namespaces permit several programmers to work on the same project without having naming collisions. How are namespaces, les and modules related? Python has a convenient and simplifying one-to-one mapping, one module per le, giving rise to one namespace. Also, Python takes the module name from the le name, and this becomes the name of the namespace. math.py is a lename, the 12.5. Namespaces 169

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

module is called math, and its namespace is math. So in Python the concepts are more or less interchangeable. But you will encounter other languages (e.g. C#), that allow one module to span multiple les, or one le to have multiple namespaces, or many les to all share the same namespace. So the name of the le doesnt need to be the same as the namespace. So a good idea is to try to keep the concepts distinct in your mind. Files and directories organize where things are stored in our computer. On the other hand, namespaces and modules are a programming concept: they help us organize how we want to group related functions and attributes. They are not about where to store things, and should not have to coincide with the le and directory structures. So in Python, if you rename the le math.py, its module name also changes, your import statements would need to change, and your code that refers to functions or attributes inside that namespace would also need to change. In other languages this is not necessarily the case. So dont blur the concepts, just because Python blurs them!

12.6 Scope and lookup rules


The scope of an identier is the region of program code in which the identier can be accessed, or used. There are three important scopes in Python: Local scope refers to identiers declared within a function. These identiers are kept in the namespace that belongs to the function, and each function has its own namespace. Global scope refers to all the identiers declared within the current module, or le. Built-in scope refers to all the identiers built into Python those like range and min that can be used without having to import anything, and are (almost) always available. Python (like most other computer languages) uses precedence rules: the same name could occur in more than one of these scopes, but the innermost, or local scope, will always take precedence over the global scope, and the global scope always gets used in preference to the built-in scope. Lets start with a simple example:
1 2 3 4

def range(n): return 123*n print(range(10))

What gets printed? Weve dened our own function called range, so there is now a potential ambiguity. When we use range, do we mean our own one, or the built-in one? Using the scope lookup rules determines this: our own range function, not the built-in one, is called,

170

Chapter 12. Modules

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

because our function range is in the global namespace, which takes precedence over the builtin names. So although names likes range and min are built-in, they can be hidden from your use if you choose to dene your own variables or functions that reuse those names. (It is a confusing practice to redene built-in names so to be a good programmer you need to understand the scope rules and understand that you can do nasty things that will cause confusion, and then you avoid doing them!) Now, a slightly more complex example:
1 2 3 4 5 6 7

n = 10 m = 3 def f(n): m = 7 return 2*n+m print(f(5), n, m)

This prints 17 10 3. The reason is that the two variables m and n in lines 1 and 2 are outside the function in the global namespace. Inside the function, new variables called n and m are created just for the duration of the execution of f. These are created in the local namespace of function f. Within the body of f, the scope lookup rules determine that we use the local variables m and n. By contrast, after weve returned from f, the n and m arguments to the print function refer to the original variables on lines 1 and 2, and these have not been changed in any way by executing function f. Notice too that the def puts name f into the global namespace here. So it can be called on line 7. What is the scope of the variable n on line 1? Its scope the region in which it is visible is lines 1, 2, 6, 7. It is hidden from view in lines 3,4,5 because of the local variable n.

12.7 Attributes and the dot operator


Variables dened inside a module are called attributes of the module. Weve seen that objects have attributes too: for example, most objects have a __doc__ attribute, some functions have a __annotations__ attribute. Attributes are accessed by using the dot operator ( .). The question attribute of module1 and module2 are accessed using module1.question and module2.question. Modules contain functions as well as attributes, and the dot operator is used to access them in the same way. seqtools.remove_at refers to the remove_at function in the seqtools module. When we use a dotted name, we often refer to it as a fully qualied name, because were saying exactly which question attribute we mean.

12.7. Attributes and the dot operator

171

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

12.8 Three import statement variants


Here are three different ways to import names into the current namespace, and to use them:
1 2

import math x = math.sqrt(10)

Here just the single identier math is added to the current namespace. If you want to access one of the functions in the module, you need to use the dot notation to get to it. Here is a different arrangement:
1 2

from math import cos, sin, sqrt x = sqrt(10)

The names are added directly to the current namespace, and can be used without qualication. The name math is not itself imported, so trying to use the qualied form math.sqrt would give an error. Then we have a convenient shorthand:
1 2 3

from math import * x = sqrt(10)

# import all the identifiers from math, # adding them to the current namespace. # Use them without qualification.

Of these three, the rst method is generally preferred, even though it means a little more typing each time. (But hey, with nice editors that do auto-completion, and fast ngers, thats a small price!) Finally, observe this case:
1 2 3 4 5

def area(radius): import math return math.pi * radius * radius x = math.sqrt(10) # this gives an error

Here we imported math, but we imported it into the local namespace of area. So the name is usable within the function body, but not in the enclosing script, because it is not in the global namespace.

12.9 Turn your unit tester into a module


Near the end of Chapter 6 we introduced unit testing, and our own test function, and youve had to copy this into each module for which you wrote tests. Now we can put that denition into a module of its own, say my_own_unit_tester.py, and simply use one line in each new script instead:
1

from my_own_unit_tester import test

172

Chapter 12. Modules

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

12.10 Glossary
argv argv is short for argument vector and is a variable in the sys module which stores a list of command line arguments passed to a program at run time. attribute A variable dened inside a module (or class or instance as we will see later). Module attributes are accessed by using the dot operator ( .). command line The sequence of characters read into the command interpreter in a command line interface (see the Wikipedia article on command line interface for more information). command line argument A value passed to a program along with the programs invocation at the command prompt of a command line interface (CLI). command prompt A string displayed by a command line interface indicating that commands can be entered. continue statement A statement that causes the current iteration of a loop to be skipped. The ow of execution goes back to the top of the loop, evaluates the condition, and proceeds accordingly, so further execution of the loop body may still take place. dot operator The dot operator ( .) permits access to attributes and functions of a module (or attributes and methods of a class or instance as we have seen elsewhere). import statement A statement which makes the objects contained in a module available for use within another module. There are two forms for the import statement. Using hypothetical modules named mymod1 and mymod2 each containing functions f1 and f2, and variables v1 and v2, examples of these two forms include:
1 2

import mymod1 from mymod2 import f1, f2, v1, v2

The second form brings the imported objects into the namespace of the importing module, while the rst form preserves a seperate namespace for the imported module, requiring mymod1.v1 to access the v1 variable from that module. method Function-like attribute of an object. Methods are invoked (called) on an object using the dot operator. For example:
>>> s = "this is a string." >>> s.upper() THIS IS A STRING. >>>

We say that the method, upper is invoked on the string, s. s is implicitely the rst argument to upper. module A le containing Python denitions and statements intended for use in other Python programs. The contents of a module are made available to the other program by using the import statement. namespace A syntactic container providing a context for names so that the same name can reside in different namespaces without ambiguity. In Python, modules, classes, functions and methods all form namespaces. 12.10. Glossary 173

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

naming collision A situation in which two or more names in a given namespace cannot be unambiguously resolved. Using
1

import string

instead of
1

from string import *

prevents naming collisions. standard library A library is a collection of software used as tools in the development of other software. The standard library of a programming language is the set of such tools that are distributed with the core programming language. Python comes with an extensive standard library.

12.11 Exercises
1. Open help for the calendar module. (a) Try the following:
1 2 3

import calendar cal = calendar.TextCalendar() cal.pryear(2012)

# create an instance # What happens here?

(b) Observe that the week starts on Monday. An adventurous CompSci student believes that it is better mental chunking to have his week start on Thursday, because then there are only two working days to the weekend, and every week has a break in the middle. Read the documentation for TextCalendar, and see how you can help him print a calendar that suits his needs. (c) Find a function to print just the month in which your birthday occurs this year. (d) Try this:
1 2

d = calendar.LocaleTextCalendar(6, "SPANISH") d.pryear(2012)

Try a few other languages, including one that doesnt work, and see what happens. (e) Experiment with calendar.isleap. What does it expect as an argument? What does it return as a result? What kind of a function is this? Make detailed notes about what you learned from these exercises. 2. Open help for the math module. (a) How many functions are in the math module?

174

Chapter 12. Modules

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

(b) What does math.ceil do? What about math.floor? ( hint: both floor and ceil expect oating point arguments.) (c) Describe how we have been computing the same value as math.sqrt without using the math module. (d) What are the two data contstants in the math module? Record detailed notes of your investigation in this exercise. 3. Investigate the copy module. What does deepcopy do? In which exercises from last chapter would deepcopy have come in handy? 4. Create a module named mymodule1.py. Add attributes myage set to your current age, and year set to the current year. Create another module named mymodule2.py. Add attributes myage set to 0, and year set to the year you were born. Now create a le named namespace_test.py. Import both of the modules above and write the following statement:
1 2

print( (mymodule2.myage - mymodule1.myage) == (mymodule2.year - mymodule1.year) )

When you will run namespace_test.py you will see either True or False as output depending on whether or not youve already had your birthday this year. 5. Add the following statement to mymodule1.py, namespace_test.py from the previous exercise:
1

mymodule2.py,

and

print("My name is", __name__)

Run namespace_test.py. What happens? Why? Now add the following to the bottom of mymodule1.py:
1 2

if __name__ == __main__: print("This wont run if Im

imported.")

Run mymodule1.py and namespace_test.py again. In which case do you see the new print statement? 6. In a Python shell / interactive interpreter, try the following:
>>> import this

What does Tim Peters have to say about namespaces? 7. Give the Python interpreters response to each of the following from a continuous interpreter session:
>>> >>> >>> >>> >>> >>> s = "If we took the bones out, it wouldnt be crunchy, would it?" s.split() type(s.split()) s.split(o) s.split(i) 0.join(s.split(o))

12.11. Exercises

175

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Be sure you understand why you get each result. Then apply what you have learned to ll in the body of the function below using the split and join methods of str objects:
1 2 3 4 5 6 7 8 9 10

def myreplace(old, new, s): """ Replace all occurrences of old with new in s. """ ...

test(myreplace(,, ;, this, that, and some other thing), this; that; and some other thing) test(myreplace( , **, Words will now be separated by stars.), Words**will**now**be**separated**by**stars.)

Your solution should pass the tests. 8. Create a module named wordtools.py with our test scaffolding in place. Now add functions to these tests pass:
test(cleanword(what?), what) test(cleanword("now!"), now) test(cleanword(?+="w-o-r-d!,@$()"),

word)

test(has_dashdash(distance--but), True) test(has_dashdash(several), False) test(has_dashdash(spoke--), True) test(has_dashdash(distance--but), True) test(has_dashdash(-yo-yo-), False) test(extract_words(Now is the time! "Now", is the time? Yes, now.), [now,is,the,time,now,is,the,time,yes,now]) test(extract_words(she tried to curtsey as she spoke--fancy), [she,tried,to,curtsey,as,she,spoke,fancy]) test(wordcount(now, [now,is,time,is,now,is,is]), 2) test(wordcount(is, [now,is,time,is,now,the,is]), 3) test(wordcount(time, [now,is,time,is,now,is,is]), 1) test(wordcount(frog, [now,is,time,is,now,is,is]), 0) test(wordset([now, is, time, is, now, is, is]), [is, now, time]) test(wordset([I, a, a, is, a, is, I, am]), [I, a, am, is]) test(wordset([or, a, am, is, are, be, but, am]), [a, am, are, be, but, is, or]) test(longestword([a, apple, pear, grape]), 5) test(longestword([a, am, I, be]), 2) test(longestword([this,supercalifragilisticexpialidocious]), 34) test(longestword([ ]), 0)

Save this module so you can use the tools it contains in future programs. 176 Chapter 12. Modules

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

12.11. Exercises

177

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

178

Chapter 12. Modules

CHAPTER

THIRTEEN

FILES
13.1 About les
While a program is running, its data is stored in random access memory (RAM). RAM is fast and inexpensive, but it is also volatile, which means that when the program ends, or the computer shuts down, data in RAM disappears. To make data available the next time you turn on your computer and start your program, you have to write it to a non-volatile storage medium, such a hard drive, usb drive, or CD-RW. Data on non-volatile storage media is stored in named locations on the media called les. By reading and writing les, programs can save information between program runs. Working with les is a lot like working with a notebook. To use a notebook, you have to open it. When youre done, you have to close it. While the notebook is open, you can either write to it or read from it. In either case, you know where you are in the notebook. You can read the whole notebook in its natural order or you can skip around. All of this applies to les as well. To open a le, you specify its name and indicate whether you want to read or write.

13.2 Writing our rst le


Lets begin with a simple program that writes three lines of text into a le:
1 2 3 4 5

myfile = open(test.txt, w) myfile.write("My first file written from Python\n") myfile.write("---------------------------------\n") myfile.write("Hello, world!\n") myfile.close()

Opening a le creates what we call a le handle. In this example, the variable myfile refers to the new handle object. Our program calls methods on the handle, and this makes changes to the actual le which is usually located on our disk. On line 1, the open function takes two arguments. The rst is the name of the le, and the second is the mode. Mode w means that we are opening the le for writing.

179

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

If there is no le named test.txt on the disk, it will be created. If there already is one, it will be replaced by the le we are writing. To put data in the le we invoke the write method on the handle, shown in lines 2, 3 and 4 above. In bigger programs, lines 2-4 will usually be replaced by a loop that writes many more lines into the le. Closing the le handle (line 5) tells the system that we are done writing and makes the disk le available for reading by other programs (or by our own program). A handle is somewhat like a TV remote control Were all familiar with a remote control for a TV. You perform operations on the remote control switch channels, change the volume, etc. But the real action happens on the TV. So, by simple analogy, wed call the remote control your handle to the underlying TV. Sometimes we want to emphasize the difference the le handle is not the same as the le, and the remote control is not the same as the TV. But at other times we prefer to treat them as a single mental chunk, or abstraction, and well just say close the le, or ip the TV channel.

13.3 Reading a le line-at-a-time


Now that the le exists on our disk, we can open it, this time for reading, and read all the lines in the le, one at a time. This time, the mode argument is r for reading:
1 2 3 4 5 6 7 8 9 10

mynewhandle = open(test.txt, r) while True: theline = mynewhandle.readline() if len(theline) == 0: break

# keep reading forever # try to read next line # if there are no more lines # leave the loop

# Now process the line weve just read print(theline, end=) mynewhandle.close()

This is a handy pattern for our toolbox. In bigger programs, wed squeeze more extensive logic into the body of the loop at line 8 for example, if each line of the le contained the name and email address of one of our friends, perhaps wed split the line into some pieces and call a function to send the friend a party invitation. On line 8 we suppress the newline character that print usually appends to our strings. Why? This is because the string already has its own newline: the readline method in line 3 returns everything up to and including the newline character. This also explains the end-of-le detection logic: when there are no more lines to be read from the le, readline returns an empty string one that does not even have a newline at the end, hence its length is 0.

180

Chapter 13. Files

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Bump your head rst ... In our sample case here, we have three lines in the le, yet we enter the loop four times. In Python, you only learn that the le has no more lines by bumping your head against the end of le, i.e. when you try to read a line that doesnt exist. In some other programming languages (e.g. Pascal), things are different: there you read three lines, but you have what is called look ahead after reading the third line you already know that there are no more lines in the le. Youre not even allowed to try to read the fourth line. So the templates for working line-at-a-time in Pascal and Python are subtly different! When you transfer your Python skills to your next computer language, be sure to ask how youll know when the le has ended: is the style in the language try, and when you bump your head youll know, or is it look ahead? If we try to open a le that doesnt exist, we get an error:
>>> mynewhandle = open(wharrah.txt, r) IOError: [Errno 2] No such file or directory: wharrah.txt

13.4 Turning a le into a list of lines


It is often useful to fetch data from a disk le and turn it into a list of lines. Suppose we have a le containing our friends and their email addresses, one per line in the le. But wed like the lines sorted into alphabetical order. A good plan is to read everything into a list of lines, then sort the list, and then write the sorted list back to another le:
1 2 3 4 5 6 7 8 9 10

f = open(friends.txt, r) xs = f.readlines() f.close() xs.sort() g = open(sortedfriends.txt, w) for v in xs: g.write(v) g.close()

The readlines method in line 2 reads all the lines and returns a list of the strings. We could have used the template from the previous section to read each line one-at-a-time, and to build up the list ourselves, but it is a lot easier to use the method that the Python implementors gave us!

13.4. Turning a le into a list of lines

181

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

13.5 Reading the whole le at once


Another way of working with text les is to read the complete contents of the le into a string, and then to use our string-processing skills to work with the contents. Wed normally use this method of processing les if we were not interested in the line structure of the le. For example, weve seen the split method on strings which can break a string into words. So here is how we might count the number of words in a le:
1 2 3 4 5 6

f = open(somefile.txt) content = f.read() f.close() words = content.split() print("There are {0} words in the file.".format(len(words)))

You will notice here that we left out the r mode in line 1. By default, if you dont supply the mode, Python opens the le for reading.

13.6 Working with binary les


Files that hold photographs, videos, zip les, executable programs, etc. are called binary les: theyre not organized into lines, and cannot be opened with a normal text editor. Python works just as easily with binary les, but when you read from the le youre going to get bytes back rather than a string. Here well copy one binary le to another:
1 2 3 4 5 6 7 8 9 10 11

f = open(somefile.zip, rb) g = open(thecopy.zip, wb) while True: buf = f.read(1024) if len(buf) == 0: break g.write(buf) f.close() g.close()

There are a few new things here. In lines 1 and 2 we added a b to the mode to tell Python that the les are binary rather than text les. In line 5, we see read can take an argument which tells it how many bytes to attempt to read from the le. Here we chose to read and write up to 1024 bytes on each iteration of the loop. When we get back an empty buffer from our attempt to read, we know we can break out of the loop and close both the les. If you set a breakpoint at line 6, (or print type(buf) there) youll see that the type of buf is bytes. We dont do any detailed work with bytes objects in this textbook.

182

Chapter 13. Files

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

13.7 An example
Many useful line-processing programs will read a text le line-at-a-time and do some minor processing as they write the lines to an output le. They might number the lines in the output le, or insert extra blank lines after every 60 lines to make it convenient for printing on sheets of paper, or extract some specic columns only from each line in the source le, or only print lines that contain a specic substring. We call this kind of program a lter. Here is a lter that copies one le to another, omitting any lines that begin with #:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15

def filter(oldfile, newfile): infile = open(oldfile, r) outfile = open(newfile, w) while True: text = infile.readline() if len(text) == 0: break if text[0] == #: continue # put any more processing logic here outfile.write(text) infile.close() outfile.close()

The continue statement at line 9 skips over the remaining lines in the current iteration of the loop, but the loop will still iterate. This style looks a bit contrived here, but it is often useful to say get the lines were not concerned with out of the way early, so that we have cleaner more focussed logic in the meaty part of the loop that might be written around line 11. Thus, if text is the empty string, the loop exits. If the rst character of text is a hash mark, the ow of execution goes to the top of the loop, ready to start processing the next line. Only if both conditions fail do we fall through to do the processing at line 11, in this example, writing the line into the new le. Lets consider one more case: suppose your original le contained empty lines. At line 6 above, would this program nd the rst empty line in the le, and terminate immediately? No! Recall that readline always includes the newline character in the string it returns. It is only when we try to read beyond the end of the le that we get back the empty string of length 0.

13.8 Directories
Files on non-volatile storage media are organized by a set of rules known as a le system. File systems are made up of les and directories, which are containers for both les and other directories. When you create a new le by opening it and writing, the new le goes in the current directory (wherever you were when you ran the program). Similarly, when you open a le for reading, 13.7. An example 183

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Python looks for it in the current directory. If you want to open a le somewhere else, you have to specify the path to the le, which is the name of the directory (or folder) where the le is located:
>>> wordsfile = open(/usr/share/dict/words, r) >>> wordlist = wordsfile.readlines() >>> print(wordlist[:6]) [\n, A\n, "As\n", AOL\n, "AOLs\n", Aachen\n]

This (unix) example opens a le named words that resides in a directory named dict, which resides in share, which resides in usr, which resides in the top-level directory of the system, called /. It then reads in each line into a list using readlines, and prints out the rst 5 elements from that list. A Windows path might be "c:/temp/words.txt" or "c:\\temp\\words.txt". Because backslashes are used to escape things like newlines and tabs, you need to write two backslashes in a literal string to get one! So the length of these two strings is the same! You cannot use / or \ as part of a lename; they are reserved as a delimiter between directory and lenames. The le /usr/share/dict/words should exist on unix-based systems, and contains a list of words in alphabetical order.

13.9 What about fetching something from the web?


The Python libraries are pretty messy in places. But here is a very simple example that copies the contents at some web URL to a local le.
1 2 3 4 5 6

import urllib.request url = http://xml.resource.org/public/rfc/txt/rfc793.txt destination_filename = rfc793.txt urllib.request.urlretrieve(url, destination_filename)

The urlretrieve function just one call could be used to download any kind of content from the Internet. Well need to get a few things right before this works: The resource were trying to fetch must exist! Check this using a browser. Well need permission to write to the destination lename, and the le will be created in the current directory - i.e. the same folder that the Python program is saved in. If we are behind a proxy server that requires authentication, (as some students are), this may require some more special handling to work around our proxy. Use a local resource for the purpose of this demonstration!

184

Chapter 13. Files

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Here is a slightly different example. Rather than save the web resource to our local disk, we read it directly into a string, and return it:
1 2 3 4 5 6 7 8 9 10 11 12 13

import urllib.request def retrieve_page(url): Retrieve the contents of a web page. The contents is converted to a string before returning it. my_socket = urllib.request.urlopen(url) dta = str(my_socket.readall()) my_socket.close() return dta

the_text = retrieve_page("http://xml.resource.org/public/rfc/txt/rfc793.txt" print(the_text)

Opening the remote url returns what we call a socket. This is a handle to our end of the connection between our program and the remote web server. We can call read, write, and close methods on the socket object in much the same way as we can work with a le handle.

13.10 Glossary
delimiter A sequence of one or more characters used to specify the boundary between separate parts of text. directory A named collection of les, also called a folder. Directories can contain les and other directories, which are refered to as subdirectories of the directory that contains them. le A named entity, usually stored on a hard drive, oppy disk, or CD-ROM, that contains a stream of characters. le system A method for naming, accessing, and organizing les and the data they contain. handle An object in our program that is connected to an underlying resource (e.g. a le). The le handle lets our program manipulate / read/ write / close the actual le that is on our disk. fully qualied name A name that is prexed by some namespace identier and the dot operator, or by an instance object, e.g. math.sqrt or tess.forward(10). mode A distinct method of operation within a computer program. Files in Python can be openned in one of three modes: read (r), write (w), and append (a). non-volatile memory Memory that can maintain its state without power. Hard drives, ash drives, and rewritable compact disks (CD-RW) are each examples of non-volatile memory. path A sequence of directory names that species the exact location of a le.

13.10. Glossary

185

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

text le A le that contains printable characters organized into lines separated by newline characters. socket One end of a connection allowing one to read and write information to or from another computer. volatile memory Memory which requires an electrical current to maintain state. The main memory or RAM of a computer is volatile. Information stored in RAM is lost when the computer is turned off.

13.11 Exercises
1. Write a program that reads a le and writes out a new le with the lines in reversed order (i.e. the rst line in the old le becomes the last one in the new le.) 2. Write a program that reads a le and prints only those lines that contain the substring snake. 3. Write a program that reads a text le and produces an output le which is a copy of the le, except the rst ve columns of each line contain a four digit line number, followed by a space. Start numbering the rst line in the output le at 1. Ensure that every line number is formatted to the same width in the output le. Use one of your Python programs as test data for this exercise: your output should be a printed and numbered listing of the Python program. 4. Write a program that undoes the numbering of the previous exercise: it should read a le with numbered lines and produce another le without line numbers.

186

Chapter 13. Files

CHAPTER

FOURTEEN

LIST ALGORITHMS
This chapter is a bit different from what weve done so far: rather than introduce more new Python syntax and features, were going to focus on the program development process, and some algorithms that work with lists. As in all parts of this book, our expectation is that you, the reader, will copy our code into your Python environment, play and experiment, and work along with us.

14.1 Test-driven development


Early in our Fruitful functions chapter we introduced the idea of incremental development, where we added small fragments of code to slowly build up the whole, so that we could easily nd problems early. Later in that same chapter we introduced unit testing and gave code for our testing framework so that we could capture, in code, appropriate tests for the functions we were writing. Test-driven development (TDD) is a software development practice which takes these practices one step further. The key idea is that automated tests should be written rst. This technique is called test-driven because if we are to believe the extremists non-testing code should only be written when there is a failing test to make pass. We can still retain our mode of working in small incremental steps, but now well dene and express those steps in terms of a sequence of increasingly sophisticated unit tests that demand more from our code at each stage. Well turn our attention to some standard algorithms that process lists now, but as we proceed through this chapter well attempt to do so in the spirit envisaged by TDD.

14.2 The linear search algorithm


Wed like to know the index where a specic item occurs within in a list of items. Specically, well return the index of the item if it is found, or well return -1 if the item doesnt occur in the list. Let us start with some tests:

187

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2 3 4 5

friends = ["Joe", "Amy", "Brad", "Angelina", "Zuki", "Thandi", "Paris"] test(search_linear(friends, Amy), 1) test(search_linear(friends, Joe), 0) test(search_linear(friends, Paris), 6) test(search_linear(friends, Bill), -1)

Motivated by the fact that our tests dont even run, let alone pass, we now write the function:
1 2 3 4 5 6

def search_linear(xs, target): Find and return the index of target in sequence xs for (i, v) in enumerate(xs): if v == target: return i return -1

There are a some points to learn here: Weve seen a similar algorithm in section 8.10 when we searched for a character in a string. There we used a while loop, here weve used a for loop, coupled with enumerate to extract the (i, v) pair on each iteration. There are other variants for example, we could have used range and made the loop run only over the indexes, or we could have used the idiom of returning None when the item was not found in the list. But the essential similarity in all these variations is that we test every item in the list in turn, from rst to last, using the pattern of the short-circuit eureka traversal that we introduced in section 8.10 that we return from the function as soon as we nd the target that were looking for. Searching all items of a sequence from rst to last is called a linear search. Each time we check whether v == target well call it a probe. We like to count probes as a measure of how efcient our algorithm is, and this will be a good enough indication of how long our algorithm will take to execute. Linear searching is characterized by the fact that the number of probes needed to nd some target depends directly on the length of the list. So if the list becomes ten times bigger, we can expect to wait ten times longer when searching for things. Notice too, that if were searching for a target that is not present in the list, well have to go all the way to the end before we can return the negative value. So this case needs N probes, where N is the length of the list. However, if were searching for a target that does exist in the list, we could be lucky and nd it immediately in position 0, or we might have to look further, perhaps even all the way to the last item. On average, when the target is present, were going to need to go about halfway through the list, or N/2 probes. We say that this search has linear performance (linear meaning straight line) because, if we were to measure the search times for different sizes of lists (N), and then plot a graph of timeto-search against N, wed get a more-or-less straight line graph. Analysis like this is pretty meaningless for small lists the computer is quick enough not to bother if the list only has a handful of items. So generally, were interested in the scalability of our algorithms how do they perform if we throw bigger problems at them. Would this search be a sensible one to use if we had a million or ten million items (perhaps the catalog of books in your local library) in our list? What happens for really large datasets, e.g. how does Google search so brilliantly well?

188

Chapter 14. List Algorithms

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

14.3 A more realistic problem


As children learn to read, there are expectations that their vocabulary will grow. So a child of age 14 is expected to know more words than a child of age 8. When prescribing reading books for a grade, an important question might be which words in this book are not in the expected vocabulary at this level? Let us assume we can read a vocabulary of words into our program, and read the text of a book, and split it into words. Let us write some tests for what we need to do next. Test data can usually be very small, even if we intend to nally use our program for larger cases:
1 2 3 4 5 6

vocab = [apple, boy, dog, down, fell, girl, grass, the, tree] book_words = the apple fell from the tree to the grass.split() test(find_unknown_words(vocab, book_words), [from, to]) test(find_unknown_words([], book_words), book_words) test(find_unknown_words(vocab, [the, boy, fell]), [])

Notice we were a bit lazy, and used split to create our list of words it is easier than typing out the list, and very convenient if you want to input a sentence into the program and turn it into a list of words. We now need to implement the function for which weve written tests, and well make use of our linear search. The basic strategy is to run through each of the words in the book, look it up in the vocabulary, and if it is not in the vocabulary, save it into a new resulting list which we return from the function:
1 2 3 4 5 6 7

def find_unknown_words(vocab, wds): """ Return a list of words in wds that do not occur in vocab """ result = [] for w in wds: if (search_linear(vocab, w) < 0): result.append(w) return result

We can happily report now that the tests all pass. Now let us look at the scalability. We have more realistic vocabulary in a text le, so let us read in the le (as a single string) and split it into a list of words. For convenience, well create a function to do this for us, and test it on a le we happen to have available:
1 2 3 4 5 6 7 8 9 10 11

def load_words_from_file(filename): """ Read words from filename, return list of words. """ f = open(filename, r) file_content = f.read() f.close() wds = file_content.split() return wds bigger_vocab = load_words_from_file("vocab.txt") print("There are {0} words in the vocab, starting with\n {1} " .format(len(bigger_vocab), bigger_vocab[:6]))

14.3. A more realistic problem

189

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Python responds with:


There are 19469 words in the vocab, starting with [a, aback, abacus, abandon, abandoned, abandonment]

So weve got a more sensible size vocabulary. Now let us load up a book. Loading a book is much like loading words from a le, but were going to do a little extra black magic. Books are full of punctuation, and have mixtures of lowercase and uppercase letters. We need to clean up the contents of the book. This will involve removing punctuation, and converting everything to the same case (lowercase, because our vocabulary is all in lowercase). So well want a more sophisticated way of converting text to words.
1 2 3

test(text_to_words("My name is Earl!"), [my, name, is, earl]) test(text_to_words("Well, I never!", said Alice.), [well, i, never, said, alice])

There is a powerful translate method available for strings. The idea is that one sets up desired substitutions for every character, we can give a corresponding replacement character. The translate method will apply these replacements throughout the whole string. So here we go:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15

def text_to_words(the_text): """ return a list of words with all punctuation removed, and all in lowercase. """

my_substitutions = the_text.maketrans( # if you find any of these ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789!"#$%&()*+,-./:;<=>?@[]^_{|}~\\ # replace them by these abcdefghijklmnopqrstuvwxyz # Translate the text now. cleaned_text = the_text.translate(my_substitutions) wds = cleaned_text.split() return wds

The translation turns all uppercase characters into lowercase, and all punctuation characters and digits into spaces. Then, of course, split will get rid of the spaces as it breaks the text into a list of words. The tests pass. Now were ready to read in our book:
1 2 3 4 5 6 7 8 9

def get_words_in_book(filename): """ Read a book from filename, and return a list of its words. """ f = open(filename, r) content = f.read() f.close() wds = text_to_words(content) return wds book_words = get_words_in_book("AliceInWonderland.txt")

190

Chapter 14. List Algorithms

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

10 11

print("There are {0} words in the book, starting with \n{1} ". format(len(book_words), book_words[:100]))

Python prints the following (all on one line, weve cheated a bit for the textbook):
There are 27336 words in the book, starting with [alice, s, adventures, in, wonderland, lewis, carroll, chapter, i, down, the, rabbit, hole, alice, was, beginning, to, get, very, tired, of, sitting, by, her, sister, on, the, bank, and, of, having, nothing, to, do, once, or, twice, she, had, peeped, into, the, book, her, sister, was, reading, but, it, had, no, pictures, or, conversations, in, it, and, what, is, the, use, of, a, book, thought, alice, without, pictures, or, conversation, so, she, was, considering, in, her, own, mind, as, well, as, she, could, for, the, hot, day, made, her, feel, very, sleepy, and, stupid, whether, the, pleasure, of, making, a]

Well now we have all the pieces ready. Let us see what words in this book are not in the vocabulary:
>>> missing_words = find_unknown_words(bigger_vocab, book_words)

We wait a considerable time now, something like a minute, before Python nally works its way through this, and prints a list of 3398 words in the book that are not in the vocabulary. Mmm... This is not particularly scaleable. For a vocabularly that is twenty times larger (youll often nd school dictionaries with 300 000 words, for example), and longer books, this is going to be slow. So let us make some timing measurements while we think about how we can improve this in the next section.
1 2 3 4 5 6 7

import time t0 = time.clock() missing_words = find_unknown_words(bigger_vocab, book_words) t1 = time.clock() print("There are {0} unknown words.".format(len(missing_words))) print("That took {0:.4f} seconds.".format(t1-t0))

We get the results and some timing that we can refer back to later:
There are 3398 unknown words. That took 49.8014 seconds.

14.4 Binary Search


If you think about what weve just done, it is not how we work in real life. If you were given a vocabulary and asked to tell if some word was present, youd probably start in the middle. You can do this because the vocabulary is ordered so you can probe some word in the middle, 14.4. Binary Search 191

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

and immediately realize that your target was before (or perhaps after) the one you had probed. Applying this principle repeatedly leads us to a very much better algorithm for searching in a list of items that are already ordered. (Note that if the items are not ordered, you have little choice other than to look through all of them. But, if we know the items are in order, we can improve our searching technique). Lets start with some tests. Remember, the list needs to be sorted:
xs = [2,3,5,7,11,13,17,23,29,31,37,43,47,53] test(search_binary(xs, 20), -1) test(search_binary(xs, 99), -1) test(search_binary(xs, 1), -1) for (i, v) in enumerate(xs): test(search_binary(xs, v), i)

Even our test cases are interesting this time: notice that we start with items not in the list and look at boundary conditions - in the middle of the list, less than all items in the list, bigger than the biggest. Then we use a loop to use every list item as a target, and to conrm that our binary search returns the corresponding index of that item in the list. It is useful to think about having a region-of-interest (roi) within the list being searched. This roi will be the portion of the list in which it is still possible that our target might be found. Our algorithm will start with the roi set to all the items in the list. On the rst probe in the middle of the roi, there are three possible outcomes: either we nd the target, or we learn that we can discard the top half of the roi, or we learn that we can discard the bottom half of the roi. And we keep doing this repeatedly, until we nd our target, or until we end up with no more items in our region of interest. We can code this as follows:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24

def search_binary(xs, target): """ Find and return the index of key in sequence xs """ lb = 0 ub = len(xs) while True: if lb == ub: # if region of interest (roi) becomes empty return -1 # next probe should be in the middle of the roi mid_index = (lb + ub) // 2 # fetch the item at that position item_at_mid = xs[mid_index] # print("Roi[{0}:{1}](size={2}), probed={3}, target={4}" # .format(lb, ub, ub-lb, item_at_mid, target)) # how does the probed item compare to the target? if item_at_mid == target: return mid_index # found it! if item_at_mid < target: lb = mid_index + 1 # use upper half of roi next time else: ub = mid_index # use lower half of roi next time

192

Chapter 14. List Algorithms

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

The region of interest is represented by two variables, a lower bound lb and an upper bound ub. It is important to be precise about what values these indexes have. Well make lb hold the index of the rst item in the roi, and make ub hold the index just beyond the last item of interest. So these semantics are similar to a Python slice semantics: the region of interest is exactly the slice xs[lb:ub]. (The algorithm never actually takes any array slices!) With this code in place, our tests pass. Great. Now if we substitute a call to this search algorithm instead of calling the search_linear in find_unknown_words, can we improve our performance? Lets do that, and again run this test:
1 2 3 4 5

t0 = time.clock() missing_words = find_unknown_words(bigger_vocab, book_words) t1 = time.clock() print("There are {0} unknown words.".format(len(missing_words))) print("That took {0:.4f} seconds.".format(t1-t0))

What a spectacular difference! More than 200 times faster!


There are 3398 unknown words. That took 0.2262 seconds.

Why is this binary search so much faster than the linear search? If we uncomment the print statement on lines 15 and 16, well get a trace of the probes done during a search. Lets go ahead, and try that:
>>> search_binary(bigger_vocab, "magic") Roi[0:19469](size=19469), probed=known, target=magic Roi[9735:19469](size=9734), probed=retailer, target=magic Roi[9735:14602](size=4867), probed=overthrow, target=magic Roi[9735:12168](size=2433), probed=mission, target=magic Roi[9735:10951](size=1216), probed=magnificent, target=magic Roi[9735:10343](size=608), probed=liken, target=magic Roi[10040:10343](size=303), probed=looks, target=magic Roi[10192:10343](size=151), probed=lump, target=magic Roi[10268:10343](size=75), probed=machete, target=magic Roi[10306:10343](size=37), probed=mafia, target=magic Roi[10325:10343](size=18), probed=magnanimous, target=magic Roi[10325:10334](size=9), probed=magical, target=magic Roi[10325:10329](size=4), probed= maggot, target=magic Roi[10328:10329](size=1), probed=magic, target=magic 10328

Here we see that nding the target word magic needed just 14 probes before it was found at index 10328. The important thing is that each probe more or less halves the remaining region of interest. By contrast, the linear search would have needed 10329 probes. The word binary means two. Binary search gets its name from the fact that each probe splits the list into two pieces and discards the one half from the region of interst. The beauty of the algorithm is that we could double the size of the vocabulary, and it would only need one more probe! And after another doubling, just another one probe. So as the vocabulary gets bigger, this algorithms performance becomes even more impressive. Can we put a formula to this? If our list size is N, what is the most number of probes k we 14.4. Binary Search 193

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

could need? The maths is a bit easier if we turn the question around: how big a list N could we deal with if we were only allowed to make k probes? With 1 probe, we can only search a list of size 1. With two probes we could cope with lists up to size 3 - (test the middle item with the rst probe, then test either the left or right sublist with the remaining probe). With one more probe, we could cope with 7 items (the middle item, and two sublists of size 3). With four probes, we can search 15 items, and 5 probes lets us search up to 31 items. So the general relationship is given by the formula N = 2k - 1 where k is the number of probes were allowed to make, and N is the maximum size of the list that can be searched in that many probes. This function is exponential in k - because k occurs as the exponent. If we wanted to turn the formula around and solve for k in terms of N, we need to move the constant 1 to the other side, and take a log (base 2) on each side. (The log is the inverse of an exponent.) So the formula for k in terms of N is now:

The square-only-on-top brackets are called ceiling brackets: this means that you must round the number up to the next whole integer. Let us try this on a calculator, or in Python, which is the mother of all calculators: suppose I have 1000 elements to be searched, what is the maximum number of probes Ill need? (There is a pesky +1 in the formula, so let us not forget to add it on...):
>>> from math import log >>> log(1000 + 1, 2) 9.967226258835993

Telling us that well need 9.96 probes maximum, to search 1000 items is not quite what we want. We forgot to take the ceiling. The ceil function in the math module does exactly this. So more accurately, now:
1 2 3 4 5 6 7

>>> >>> 10 >>> 20 >>> 30

from math import log, ceil ceil(log(1000 + 1, 2)) ceil(log(1000000 + 1, 2)) ceil(log(1000000000 + 1, 2))

This tells us that searching 1000 items needs 10 probes. (Well technically, with 10 probes we can search exactly 1023 items, but the easy and useful stuff to remember here are that 1000 items needs 10 probes, a million needs 20 probes, and a billion items only needs 30 probes). You will rarely encounter algorithms that scale to large datasets as beautifully as binary search does!

194

Chapter 14. List Algorithms

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

14.5 Removing adjacent duplicates from a list


We often want to get the unique elements in a list, i.e. produce a new list in which each different element occurs just once. Consider our case of looking for words in Alice in Wonderland that are not in our vocabulary. We had a report that there are 3398 such words, but there are duplicates in that list. In fact, the word alice occurs 398 times in the book, and it is not in our vocabulary! How should we remove these duplicates? A good approach is to sort the list, then remove all adjacent duplicates. Let us start with removing adjacent duplicates
1 2 3 4

test(remove_adjacent_dups([1,2,3,3,3,3,5,6,9,9]), [1,2,3,5,6,9]) test(remove_adjacent_dups([]), []) test(remove_adjacent_dups([a, big, big, bite, dog]), [a, big, bite, dog])

The algorithm is easy and efcient. We simply have to remember the most recent item that was inserted into the result, and avoid inserting it again:
1 2 3 4 5 6 7 8 9 10 11 12

def remove_adjacent_dups(xs): """ Return a new list in which all adjacent duplicates from xs have been removed. """ result = [] most_recent_elem = None for e in xs: if e != most_recent_elem: result.append(e) most_recent_elem = e return result

The amount of work done in this algorithm is linear each item in xs causes the loop to execute exactly once, and there are no nested loops. So doubling the number of elements in xs should cause this program to run twice as long: the relationship between the size of the list and the time to run will be graphed as a straight (linear) line. Let us go back now to our analysis of Alice in Wonderland. Before checking the words in the book against the vocabulary, well sort those words into order, and eliminate duplicates. So our new code looks like this:
1 2 3 4 5 6 7

all_words = get_words_in_book("AliceInWonderland.txt") all_words.sort() book_words = remove_adjacent_dups(all_words) print("There are {0} words in the book. Only {1} are unique.". format(len(all_words), len(book_words))) print("The first 100 words are\n{0}". format(book_words[:100]))

Almost magically, we get the following output:

14.5. Removing adjacent duplicates from a list

195

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

There are 27336 words in the book. Only 2570 are unique. The first 100 words are [_i_, a, abide, able, about, above, absence, absurd, acceptance, accident, accidentally, account, accounting, accounts, accusation, accustomed, ache, across, act, actually, ada, added, adding, addressed, addressing, adjourn, adoption, advance, advantage, adventures, advice, advisable, advise, affair, affectionately, afford, afore, afraid, after, afterwards, again, against, age, ago, agony, agree, ah, ahem, air, airs, alarm, alarmed, alas, alice, alive, all, allow, almost, alone, along, aloud, already, also, altered, alternately, altogether, always, am, ambition, among, an, ancient, and, anger, angrily, angry, animal, animals, ann, annoy, annoyed, another, answer, answered, answers, antipathies, anxious, anxiously, any, anything, anywhere, appealed, appear, appearance, appeared, appearing, applause, apple, apples, arch]

It should surprise you that Lewis Carroll was able to write a classic piece of literature using only 2570 different words!

14.6 Merging sorted lists


Suppose we have two sorted lists. Devise an algorithm to merge them together into a single sorted list. A simple but inefcient algorithm could be to simply append the two lists together, and sort the result:
1 2

newlist = (xs + ys) newlist.sort()

But this doesnt take advantage of the fact that the two lists are already sorted, and is going to have poor scaleability and performance for very large lists. Lets get some tests together rst:
1 2 3 4 5 6 7 8 9 10 11

xs = [1,3,5,7,9,11,13,15,17,19] ys = [4,8,12,16,20,24] zs = xs+ys zs.sort() test(merge(xs, []), xs) test(merge([], ys), ys) test(merge([], []), []) test(merge(xs, ys), zs) test(merge([1,2,3], [3,4,5]), [1,2,3,3,4,5]) test(merge([a, big, cat], [big, bite, dog]), [a, big, big, bite, cat, dog])

196

Chapter 14. List Algorithms

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Here is our merge algorithm:


1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22

def merge(xs, ys): """ merge sorted lists xs and ys. Return a sorted result """ result = [] xi = 0 yi = 0 while True: if xi >= len(xs): # if xs list is finished, result.extend(ys[yi:]) # add remaining items from ys return result; # and were done. if yi >= len(ys): # Same again, but swap roles result.extend(xs[xi:]) return result; # Both lists still have items, copy smaller item to result. if xs[xi] <= ys[yi]: result.append(xs[xi]) xi += 1 else: result.append(ys[yi]) yi += 1

The algorithm works as follows: we create a result list, and keep two indexes, one into each list (lines 3-5). On each iteration of the loop, whichever list item is smaller is copied to the result list, and that lists index is advanced. As soon as either index reaches the end of its list, we copy all the remaining items from the other list into the result, which we return.

14.7 Alice in Wonderland, again!


Underlying the algorithm for merging sorted lists is a deep pattern of computation that is widely reusable. The pattern essence is Run through the lists always processing the smallest remaining items from each, with these cases to consider: What should we do when either list has no more items? What should we do if the smallest items from each list are equal to each other? What should we do if the smallest item in the rst list is smaller than the smallest one the second list? What should we do in the remaining case? Lets assume we have two sorted lists. Exercise your algorithmic skills by adapting the merging algorithm pattern for each of these cases: Return only those items that are present in both lists. Return only those items that are present in the rst list, but not in the second.

14.7. Alice in Wonderland, again!

197

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Return only those items that are present in the second list, but not in the rst. Return items that are present in either the rst or the second list. Return items from the rst list that are not eliminated by a matching element in the second list. In this case, an item in the second list knocks out just one matching item in the rst list. This operation is sometimes called bagdiff. For example bagdiff([5,7,11,11,11,12,13], [7,8,11]) would return [5,11,11,12,13] In the previous section we sorted the words from the book, and elimated duplicates. Our vocabulary is also sorted. So third case above nd all items in the second list that are not in the rst list, would be another way to implement find_unknown_words. Instead of searching for every word in the dictionary (either by linear or binary search), why not use a variant of the merge to return the words that occur in the book, but not in the vocabulary.
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

def find_unknowns_merge_pattern(vocab, wds): """ Both the vocab and wds must be sorted. Return a new list of words from wds that do not occur in vocab. """ result = [] xi = 0 yi = 0 while True: if xi >= len(vocab): result.extend(wds[yi:]) return result if yi >= len(wds): return result if vocab[xi] == wds[yi]: yi += 1 # good, word exists in vocab

elif vocab[xi] < wds[yi]: # move past this vocab word, xi += 1 else: # got word that is not in vocab result.append(wds[yi]) yi += 1

Now we put it all together:


1 2 3 4 5 6 7

all_words = get_words_in_book("AliceInWonderland.txt") t0 = time.clock() all_words.sort() book_words = remove_adjacent_dups(all_words) missing_words = find_unknowns_merge_pattern(bigger_vocab, book_words) t1 = time.clock() print("There are {0} unknown words.".format(len(missing_words)))

198

Chapter 14. List Algorithms

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

print("That took {0:.4f} seconds.".format(t1-t0))

Even more stunning performance here:


There are 828 unknown words. That took 0.0410 seconds.

Lets review what weve done. We started with a word-by-word linear lookup in the vocabulary that ran in about 50 seconds. We implemented a clever binary search, and got that down to 0.22 seconds, more than 200 times faster. But then we did something even better: we sorted the words from the book, eliminated duplicates, and used a merging pattern to nd words from the book that were not in the dictionary. This was about ve times faster than even the binary lookup algorithm. At the end of the chapter our algorithm is more than a 1000 times faster than our rst attempt! That is what we can call a good day at the ofce!

14.8 Eight queens puzzle, part 1


As told by Wikipedia, The eight queens puzzle is the problem of placing eight chess queens on an 8x8 chessboard so that no two queens attack each other. Thus, a solution requires that no two queens share the same row, column, or diagonal.

Please try this yourself, and nd a few more solutions by hand. Wed like to write a program to nd solutions to the this puzzle. In fact, the puzzle generalizes to placing N queens on an NxN board, so were going to think about the general case, not just the 8x8 case. Perhaps we can nd solutions for 12 queens on a 12x12 board, or 20 queens on a 20x20 board. How do we approach a complex problem like this? A good starting point is to think about our data structures how exactly do we plan to represent the state of the chessboard and its queens in our program? Once we have some handle on what our puzzle is going to look like 14.8. Eight queens puzzle, part 1 199

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

in memory, we can begin to think about the functions and logic well need to solve the puzzle, i.e. how do we put another queen onto the board somewhere, and to check whether it clashes with any of the queens already on the board. The steps of nding a good representation, and then nding a good algorithm to operate on the data cannot always be done independently of each other. As you think about the operations you require, you may want to change or reorganize the data somewhat to make it easier to do the operations you need. This relationship between algorithms and data was elegantly expressed in the title of a book Algorithms + Data Structures = Programs, written by one of the pioneers in Computer Science, Niklaus Wirth, the inventor of Pascal. Lets brainstorm some ideas about how a chessboard and queens could be represented in memory. A two dimensional matrix (a list of 8 lists, each containing 8 squares) is one possibility. At each square of the board would like to know whether it contains a queen or not just two possible states for each square so perhaps each element in the lists could be True or False, or, more simply, 0 or 1. Our state for the solution above could then have this data representation:
1 2 3 4 5 6 7 8

bd1 = [[0,0,0,1,0,0,0,0], [0,0,0,0,0,0,1,0], [0,0,1,0,0,0,0,0], [0,0,0,0,0,0,0,1], [0,1,0,0,0,0,0,0], [0,0,0,0,1,0,0,0], [1,0,0,0,0,0,0,0], [0,0,0,0,0,1,0,0]]

You should also be able to see how the empty board would be represented, and you should start to imagine what operations or changes youd need to make to the data to place another queen somewhere on the board. Another idea might be to keep a list of coordinates of where the queens are. Using the notation in the illustration, for example, we could represent the state of that solution as:
1

bd2 = [ a6, b4, c2, d0, e5, f7, g1, h3 ]

We could make other tweaks to this perhaps each element in this list should rather be a tuple, with integer coordinates for both axes. And being good computer scientists, wed probably start numbering each axis from 0 instead of at 1. Now our representation could be:
1

bd3 = [(0,6), (1,4), (2,2), (3,0), (4,5), (5,7), (6,1), (7,3)]

Looking at this representation, we cant help but notice that the rst coordinates are 0,1,2,3,4,5,6,7 and they correspond exactly to the index position of the pairs in the list. So we could discard them, and come up with this really compact alternative representation of the solution:

200

Chapter 14. List Algorithms

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

bd4 = [6, 4, 2, 0, 5, 7, 1, 3]

This will be what well use, lets see where that takes us. This representation is not general Weve come up with a great representation. But will it work for other puzzles? Our list representation has the constraint that one can only put one queen in each column. But that is a puzzle constraint anyway no two queens are allowed to share the same column. So puzzle and data representation are well matched. But if we were trying to solve a different puzzle on a chessboard, perhaps play a game of checkers, where many pieces could occupy the same column, our representation would not work. Let us now take some grand insight into the problem. Do you think it is a coincidence that there are no repeated numbers in the solution? The solution [6,4,2,0,5,7,1,3] contains the numbers 0,1,2,3,4,5,6,7, but none are duplicated! Could other solutions contain duplicate numbers, or not? A little thinking should convince you that there can never be duplicate numbers in a solution: the numbers represent the row on which the queen is placed, and because we are never permitted to put two queens in the same row, no solution will ever have duplicate row numbers in it. Our key insight In our representation, any solution to the N queens problem must therefore be a permutation of the numbers [0 .. N-1]. Note that not all permutations are solutions. For example, [0,1,2,3,4,5,6,7] has all queens on the same diagonal. Wow, we seem to be making progress on this problem merely by thinking, rather than coding! Our algorithm should start taking shape now. We can start with the list [0..N-1], generate various permutations of that list, and check each permutation to see if it has any clashes (queens that are on the same diagonal). If it has no clashes, it is a solution, and we can print it. Let us be precise and clear on this issue: if we only use permutations of the rows, and were using our compact representation, no queens can clash on either rows or columns, and we dont even have to concern ourselves with those cases. So the only clashes we need to test for are clashes on the diagonals. It sounds like a useful function will be one that can test if two queens share a diagonal. Each queen is on some (x,y) position. So does the queen at (5,2) share a diagonal with the one at (2,0)? Does (5,2) clash with (3,0)?
1 2 3 4

test(share_diagonal(5,2,2,0), test(share_diagonal(5,2,3,0), test(share_diagonal(5,2,4,3), test(share_diagonal(5,2,4,1),

False) True) True) True)

14.8. Eight queens puzzle, part 1

201

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

A little geometry will help us here. A diagonal has a slope of either 1 or -1. The question we really want to ask is is their distance between them the same in the x and the y direction? If it is, they share a diagnonal. Because diagonals can be to the left or right, it will make sense for this program to use the absolute distance in each direction:
1 2 3 4 5

def share_diagonal(x0, y0, x1, y1): """ Is (x0, y0) on a shared diagonal with (x1, dy = abs(y1 - y0) # calc the absolute y dx = abs(x1 - x0) # calc the absolute x return dx == dy # they clash if dx ==

y1)? """ distance distance dy

If you copy the code and run it, youll be happy to learn that the tests pass! Now lets consider how we construct a solution by hand. Well put a queen somewhere in the rst column, then place one in the second column, only if it does not clash with the one already on the board. And then well put a third one on, checking it against the two queens already to its left. When we consider the queen on column 6, well need to check for clashes against those in all the columns to its left, i.e. in columns 0,1,2,3,4,5. So the next building block is a function that, given a partially completed puzzle, can check whether the queen at column c clashes with any of the queens to its left, at columns 0,1,2,..c-1:
1 2 3 4 5 6 7 8 9 10 11

# solutions should not have any clashes test(col_clashes([6,4,2,0,5], 4), False) test(col_clashes([6,4,2,0,5,7,1,3], 7), False) # Now some test cases that should mostly clash test(col_clashes([0,1], 1), True) test(col_clashes([5,6], 1), True) test(col_clashes([6,5], 1), True) test(col_clashes([0,6,4,3], 3), True) test(col_clashes([5,0,7], 2), True) test(col_clashes([2,0,1,3], 1), False) test(col_clashes([2,0,1,3], 2), True)

Here is our function that makes them all pass:


1 2 3 4 5 6 7 8 9

def col_clashes(bs, c): """ Return True if the queen at column c clashes with any queen to its left. """ for i in range(c): # look at all columns to the left of c if share_diagonal(i, bs[i], c, bs[c]): return True return False # no clashes - col c has a safe placement.

Finally, were going to give our program one of our permutations i.e. all queens placed somewhere, one on each row, one on each column. But does the permutation have any diagonal clashes?
1 2

test(has_clashes([6,4,2,0,5,7,1,3]), False) # solution from above test(has_clashes([4,6,2,0,5,7,1,3]), True) # swap rows of first two

202

Chapter 14. List Algorithms

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

3 4

test(has_clashes([0,1,2,3]), True) test(has_clashes([2,0,3,1]), False)

# try small 4x4 board # solution to 4x4 case

And the code to make the tests pass:


1 2 3 4 5 6 7 8 9

def has_clashes(the_board): """ Determine whether we have any queens clashing on the diagonals. Were assuming here that the_board is a permutation of column numbers, so were not explicitly checking row or column clashes. """ for col in range(1,len(the_board)): if col_clashes(the_board, col): return True return False

Summary of what weve done so far: we now have a powerful function called has_clashes that can tell if a conguration is a solution to the queens puzzle. Lets get on now with generating lots of permutations and nding solutions!

14.9 Eight queens puzzle, part 2


This is the fun, easy part. We could try to nd all permutations of [0,1,2,3,4,5,6,7] that might be algorithmically challenging, and would be a brute force way of tackling the problem. We just try everything, and nd all possible solutions. Of course we know there are N! permutations of N things, so we can get an early idea of how long it would take to search all of them for all solutions. Not too long at all, actually - 8! is only 40320 different cases to check out. This is vastly better than starting with 64 places to put eight queens. If you do the sums for how many ways can you choose 8 of the 64 squares for your queens, the formula (called N choose k where youre choosing k=8 squares of the available N=64) yieds a whopping 4426165368, obtained from (64! / (8! x 56!)). So our earlier key insight that we only need to consider permutations has reduced what we call the problem space from about 4.4 billion cases to just 40320! Were not even going to explore all those, however. When we introduced the random number module, we learnt that it had a shuffle method that randomly permuted a list of items. So were going to write a random algorithm to nd solutions to the N queens problem. Well begin with the permutation [0,1,2,3,4,5,6,7] and well repeatedly shufe the list, and test each to see if it works! Along the way well count how many attempts we need before we nd each solution, and well nd 10 solutions (we could hit the same solution more than once, because shufe is random!):
1 2 3 4 5 6 7

def main(): import random rng = random.Random() bd = list(range(8)) numFound = 0 tries = 0

# instantiate a generator # generate the initial permutation

14.9. Eight queens puzzle, part 2

203

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

8 9 10 11 12 13 14 15 16

while numFound < 10: rng.shuffle(bd) tries += 1 if not has_clashes(bd): print(Found solution {0} in {1} tries..format(bd, tries)) tries = 0 numFound += 1 main()

Almost magically, and at great speed, we get this:


Found Found Found Found Found Found Found Found Found Found solution solution solution solution solution solution solution solution solution solution [3, [5, [3, [1, [6, [3, [4, [3, [5, [1, 6, 7, 0, 6, 1, 0, 1, 5, 1, 6, 2, 1, 4, 4, 3, 4, 7, 0, 6, 2, 7, 3, 7, 7, 0, 7, 0, 4, 0, 5, 1, 0, 1, 0, 7, 5, 3, 1, 3, 7, 4, 6, 6, 3, 4, 2, 6, 7, 7, 4, 0, 4, 2, 5, 2, 6, 2, 2, 4, 0, 5] 2] 5] 2] 5] 1] 5] 6] 2] 3] in in in in in in in in in in 693 tries. 82 tries. 747 tries. 428 tries. 376 tries. 204 tries. 98 tries. 64 tries. 177 tries. 478 tries.

Here is an interesting fact about this algorithm. There are know to be 92 solutions to this puzzle, and we know there are 40320 permutations, all in all. So on average, you probably need to try 40320/92 shufes which is about 438, before you stumble across a solution. The number of tries we printed looks like our experimental data agrees quite nicely with our theory!

14.10 Glossary
binary search A famous algorithm that searches for a target in a sorted list. Each probe in the list allows us to discard half the remaining items, so the algorithm is very efcient. linear Relating to a straight line. Here, we talk about graphing how the time taken by an algorithm depends on the size of the data it is processing. Linear algorithms have straightline graphs that can describe this relationship. linear search A search that probes each item in a list or sequence, from rst, until it nds what it is looking for. It is used for searching for a target in unordered lists of items. Merge algorithm An efcient algorithm that merges two already sorted lists, to produce a sorted list result. The merge algorithm is really a pattern of computation that can be adapted and reused for various other scenarios, such as nding words that are in a book, but not in a vocabulary. probe Each time we take a look when searching for an item is called a probe. In our chapter on Iteration we also played a guessing game where the computer tried to guess the users secret number. Each of those tries would also be called a probe. test-driven development (TDD) A software development practice which arrives at a desired feature through a series of small, iterative steps motivated by automated tests which are 204 Chapter 14. List Algorithms

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

written rst that express increasing renements of the desired feature. (see the Wikipedia article on Test-driven development for more information.)

14.11 Exercises
1. The section in this chapter called Alice in Wonderland, again! started with the observation that the merge algorithm uses a pattern that can be reused in other situations. Adapt the merge algorithm to write each of these functions, as was suggested there: (a) Return only those items that are present in both lists. (b) Return only those items that are present in the rst list, but not in the second. (c) Return only those items that are present in the second list, but not in the rst. (d) Return items that are present in either the rst or the second list. (e) Return items from the rst list that are not eliminated by a matching element in the second list. In this case, an item in the second list knocks out just one matching item in the rst list. This operation is sometimes called bagdiff. For example bagdiff([5,7,11,11,11,12,13], [7,8,11]) would return [5,11,11,12,13] 2. Modify the queens program to solve some boards of size 4, 12, and 16. What is the maximum size puzzle you can usually solve in under a minute? 3. Adapt the queens program so that we keep a list of solutions that have already printed, so that we dont print the same solution more than once. 4. Chess boards are symmetric: if we have a solution to the queens problem, its mirror solution either ipping the board on the X or in the Y axis, is also a solution. And giving the board a 90 degree, 180 degree, or 270 degree rotation is also a solution. In some sense, solutions that are just mirror images or rotations of other solutions in the same family are less interesting than the unique core cases. Of the 92 solutions for the 8 queens problem, there are only 12 unique families if you take rotations and mirror images into account. Wikipedia has some fascinating stuff about this. (a) Write a function to mirror a solution in the Y axis, (b) Write a function to mirror a solution in the X axis, (c) Write a function to rotate a solution by 90 degrees anti-clockwise, and use this to provide 180 and 270 degree rotations too. (d) Write a function which is given a solution, and it generates the family of symmetries for that solution. For example, the symmetries of [0,4,7,5,2,6,1,3] are
[[0,4,7,5,2,6,1,3],[7,1,3,0,6,4,2,5], [4,6,1,5,2,0,3,7],[2,5,3,1,7,4,6,0], [3,1,6,2,5,7,4,0],[0,6,4,7,1,3,5,2], [7,3,0,2,5,1,6,4],[5,2,4,6,0,3,1,7]]

14.11. Exercises

205

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

(e) Now adapt the queens program so it wont list solutions that are in the same family. It only prints solutions from unique families. 5. Every week a computer scientist buys four lotto tickets. He always choses the same prime numbers, with the hope that he ever hits the jackpot, he will be able to go on TV and tell everyone his secret and this will suddenly create widespread public interest in prime numbers. He represents his weekly tickets in Python as a list of lists:
my_tickets = [ [ 7, 17, 37, 19, 23, [ 7, 2, 13, 41, 31, [ 2, 5, 7, 11, 13, [13, 17, 37, 19, 23, 43], 43], 17], 43] ]

Complete these exercises. (a) Each lotto draw takes six random balls, numbered from 1 to 49. Write a function to return a lotto draw. (b) Write a function that returns compares a single ticket and a draw, and returns the number of correct picks on that ticket:
test(lotto_match([42,4,7,11,1,13], [2,5,7,11,13,17]), 3)

(c) Write a function that takes a list of tickets and a draw, and returns a list telling how many picks were correct on each ticket:
test(lotto_matches([42,4,7,11,1,13], my_tickets), [1,2,3,1])

(d) Write a function that takes a list of integers, and returns the number of primes in the list:
test(primes_in([42, 4, 7, 11, 1, 13]), 3)

(e) Write a function to discover whether the computer scientist has missed any prime numbers in his selection of the four tickets. Return a list of all primes that he has missed:
test(prime_misses(my_tickets), [3, 29, 47])

(f) Write a function that repeatedly makes a new draw, and compares the draw to the four tickets. i. Count how many draws are needed until one of the computer scientists tickets has at least 3 correct picks. Try the experiment twenty times, and average out the number of draws needed. ii. How many draws are needed, on average, before he gets at least 4 picks correct? iii. How many draws are needed, on average, before he gets at least 5 correct? (Hint: this might take a while. It would be nice if you could print some dots, like a progress bar, to show when each of the 20 experiments has completed.) Notice that we have difculty constructing test cases here, because our random numbers are not deterministic. Automated testing only really works if you already know what the answer should be! 206 Chapter 14. List Algorithms

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

6. Read Alice in Wonderland. You can read the plain text version we have with this textbook, or if you have e-book reader software on your PC, or a Kindle, iPhone, Android, etc. youll be able to nd a suitable version for your device at http://www.gutenberg.org/. They also have html and pdf versions, with pictures, and thousands of other classic books!

14.11. Exercises

207

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

208

Chapter 14. List Algorithms

CHAPTER

FIFTEEN

CLASSES AND OBJECTS - THE BASICS


15.1 Object-oriented programming
Python is an object-oriented programming language, which means that it provides features that support object-oriented programming (OOP). Object-oriented programming has its roots in the 1960s, but it wasnt until the mid 1980s that it became the main programming paradigm used in the creation of new software. It was developed as a way to handle the rapidly increasing size and complexity of software systems, and to make it easier to modify these large and complex systems over time. Up to now, most of the programs we have been writing use a procedural programming paradigm. In procedural programming the focus is on writing functions or procedures which operate on data. In object-oriented programming the focus is on the creation of objects which contain both data and functionality together. (We have seen turtle objects, string objects, and random number generators, to name a few places where weve already worked with objects.) Usually, each object denition corresponds to some object or concept in the real world, and the functions that operate on that object correspond to the ways real-world objects interact.

15.2 User-dened compound data types


Weve already seen classes like str, int, float and Turtle. We are now ready to create our own user-dened class: the Point. Consider the concept of a mathematical point. In two dimensions, a point is two numbers (coordinates) that are treated collectively as a single object. Points are often written in parentheses with a comma separating the coordinates. For example, (0, 0) represents the origin, and (x, y) represents the point x units to the right and y units up from the origin. Some of the typical operations that one associates with points might be calculating the distance of a point from the origin, or from another point, or nding a midpoint of two points, or asking if a point falls within a given rectangle or circle. Well shortly see how we can organize these together with the data. 209

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

A natural way to represent a point in Python is with two numeric values. The question, then, is how to group these two values into a compound object. The quick and dirty solution is to use a tuple, and for some applications that might be a good choice. An alternative is to dene a new class. This approach involves a bit more effort, but it has advantages that will be apparent soon. Well want our points to each have an x and a y attribute, so our rst class denition looks like this:
1 2 3 4 5 6 7

class Point: """ Point class represents and manipulates x,y coords. """ def __init__(self): """ Create a new point at the origin """ self.x = 0 self.y = 0

Class denitions can appear anywhere in a program, but they are usually near the beginning (after the import statements). Some programmers and languages prefer to put every class in a module of its own we wont do that here. The syntax rules for a class denition are the same as for other compound statements. There is a header which begins with the keyword, class, followed by the name of the class, and ending with a colon. Youll know where the class ends by looking at the indentation. If the rst line after the class header is a string, it becomes the docstring of the class, and will be recognized by various tools. (This is also the way docstrings work in functions.) Every class should have a method with the special name __init__. This initializer method is automatically called whenever a new instance of Point is created. It gives the programmer the opportunity to set up the attributes required within the new instance by giving them their initial state / values. The self parameter (you could choose any other name, but nobody ever does!) is automatically set to reference the newly created object that needs to be initialized. So lets use our new Point class now...
1 2 3 4

p = Point() q = Point()

# Instantiate an object of type Point # and make a second point # Each point object has its own x and y

print(p.x, p.y, q.x, q.y)

This program prints:


0 0 0 0

because during the initialization of the objects, we created two attributes called x and y for each, and gave them both the value 0. This should look familiar weve used classes before to create more than one object:
1 2 3 4

from turtle import Turtle tess = Turtle() alex = Turtle() # Instantiate objects of type Turtle

The variables p and q are assigned references to two new Point objects. A function like 210 Chapter 15. Classes and Objects - the Basics

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Turtle or Point that creates a new object instance is called a constructor, and every class automatically provides a constructor function which is named the same as the class. It may be helpful to think of a class as a factory for making objects. The class itself isnt an instance of a point, but it contains the machinery to make point instances. Every time you call the constructor, youre asking the factory to make you a new object. As the object comes off the production line, its initialization method is executed to get the object properly set up with its factory default settings. The combined process of make me a new object and get its settings initialized to the factory default settings is called instantiation.

15.3 Attributes
Like real world objects, object instances have both attributes and methods. We can modify the attributes in an instance using dot notation:
>>> p.x = 3 >>> p.y = 4

Both modules and instances create their own namespaces, and the syntax for accessing names contained in each, called attributes, is the same. In this case the attribute we are selecting is a data item from an instance. The following state diagram shows the result of these assignments:

The variable p refers to a Point object, which contains two attributes. Each attribute refers to a number. We can access the value of an attribute using the same syntax:
>>> print(p.y) 4 >>> x = p.x >>> print(x) 3

The expression p.x means, Go to the object p refers to and get the value of x. In this case, we assign that value to a variable named x. There is no conict between the variable x (in the global namespace here) and the attribute x (in the namespace belonging to the instance). The purpose of dot notation is to fully qualify which variable you are referring to unambiguously. You can use dot notation as part of any expression, so the following statements are legal:
1 2

print((x={0}, y={1}).format(p.x, p.y)) distance_squared_from_origin = p.x * p.x + p.y * p.y

The rst line outputs (x=3, y=4). The second line calculates the value 25. 15.3. Attributes 211

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

15.4 Improving our initializer


To create a point at position (7, 6) currently needs three lines of code:
1 2 3

p = Point() p.x = 7 p.y = 6

We can make our class constructor more general by putting extra parameters into the __init__ method, as shown in this example:
1 2 3 4 5 6 7 8 9

class Point: """ Point class represents and manipulates x,y coords. """ def __init__(self, x=0, y=0): """ Create a new point at x, y """ self.x = x self.y = y # Other statements outside the class continue below here.

The x and y parameters here are both optional. If the caller does not supply arguments, theyll get the default values of 0. Here is our improved class in action:
>>> >>> >>> >>> 4 3 p = Point(4, 2) q = Point(6, 3) r = Point() # r represents the origin (0, 0) print(p.x, q.y, r.x) 0

Technically speaking ... If we are really fussy, we would argue that the __init__ methods docstring is inaccurate. __init__ doesnt create the object (i.e. set aside memory for it), it just initializes the object to its factory-default settings after its creation. But tools like PyScripter understand that instantiation creation and initialization happen together, and they choose to display the initializers docstring as the tooltip to guide the programmer when you call the class constructor. So were writing the docstring so that it makes the most sense when it pops up to help the programmer who is using our Point class:

212

Chapter 15. Classes and Objects - the Basics

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

15.5 Adding other methods to our class


The key advantage of using a class like Point rather than a simple tuple (6, 7) now becomes apparent. We can add methods to the Point class that are sensible operations for points, but which may not be appropriate for other tuples like (25, 12) which might represent, say, a day and a month, e.g. Christmas day. So being able to calculate the distance from the origin is sensible for points, but not for (day, month) data. For (day, month) data, wed like different operations, perhaps to nd what day of the week it will fall on in 2020. Creating a class like Point brings an exceptional amount of organizational power to our programs, and to our thinking. We can group together the sensible operations, and the kinds of data they apply to, and each instance of the class can have its own state. A method behaves like a function but it is invoked on a specic instance, e.g. tess.right(90). Like a data attribute, methods are accessed using dot notation. Lets add another method, distance_from_origin, to see better how methods work:
1 2 3 4 5 6 7 8 9 10 11

class Point: """ Create a new Point, at coordinates x, y """ def __init__(self, x=0, y=0): """ Create a new point at x, y """ self.x = x self.y = y def distance_from_origin(self): """ Compute my distance from the origin """ return ((self.x ** 2) + (self.y ** 2)) ** 0.5

Lets create a few point instances, look at their attributes, and call our new method on them:
>>> p = Point(3, 4) >>> p.x 3 >>> p.y

15.5. Adding other methods to our class

213

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

4 >>> p.distance_from_origin() 5.0 >>> q = Point(5, 12) >>> q.x 5 >>> q.y 12 >>> q.distance_from_origin() 13.0 >>> r = Point() >>> r.x 0 >>> r.y 0 >>> r.distance_from_origin() 0.0

When dening a method, the rst parameter refers to the instance being manipulated. As already noted, it is customary to name this parameter self. Notice that the caller of distance_from_origin does not explicitly supply an argument to match the self parameter this is done for you, behind your back.

15.6 Instances as arguments and parameters


You can pass an object as an argument in the usual way. Weve already seen this in some of the turtle examples, where we passed the turtle to some function like draw_bar in the chapter titled Conditionals, so that the function could control and use whatever turtle instance we passed to it. Be aware that your variable only holds a reference to an object, so passing tess into a function creates an alias: both the caller and the called function now have a reference, but there is only one turtle! Here is a simple function involving our new Point objects:
1 2

def print_point(pt): print(({0}, {1}).format(pt.x, pt.y))

print_point takes a point as an argument and formats the output in whichever way we choose. If you call print_point(p) with point p as dened previously, the output is (3, 4).

15.7 Converting an instance to a string


Most object-oriented programmers probably would not do what weve just done in print_point. When were working with classes and objects, a preferred alternative is to 214 Chapter 15. Classes and Objects - the Basics

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

add a new method to the class. And we dont like chatterbox methods that call print. A better approach is to have a method so that every instance can produce a string representation of itself. Lets initially call it to_string:
1 2 3 4 5

class Point: # ... def to_string(self): return ({0}, {1}).format(self.x, self.y)

Now we can say:


>>> p = Point(3, 4) >>> print(p.to_string()) (3, 4)

But, you ask, dont we already have an str type converter that can turn our object into a string? Yes! And doesnt print automatically use this when printing things? Yes again! But these automatic mechanisms do not yet do exactly what we want:
>>> str(p) <__main__.Point object at 0x01F9AA10> >>> print(p) <__main__.Point object at 0x01F9AA10>

Python has a clever trick up its sleeve to x this. If we call our new method __str__ instead of to_string, the Python interpreter will use our code whenever it needs to convert a Point to a string. Lets re-do this again, now:
1 2 3 4 5

class Point: # ... def __str__(self): # all we have done is renamed the method return ({0}, {1}).format(self.x, self.y)

and now things are looking great!


>>> (3, >>> (3, str(p) 4) print(p) 4) # python now uses the __str__ method that we wrote.

15.8 Instances as return values


Functions and methods can return instances. For example, given two Point objects, nd their midpoint. First well write this as a regular function:
1 2 3

def midpoint(p1, p2): """ Return the midpoint of points p1 and p2 """ mx = (p1.x + p2.x)/2

15.8. Instances as return values

215

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

4 5

my = (p1.y + p2.y)/2 return Point(mx, my)

The function creates and returns a new Point object:


>>> p >>> q >>> r >>> r (4.0, = Point(3, 4) = Point(5, 12) = midpoint(p, q) 8.0)

Now let us do this as a method instead. Suppose you have a point object, and wish to nd the midpoint halfway between it and some other target point:
1 2 3 4 5 6 7 8

class Point: # ... def halfway(self, target): """ Return the halfway point between myself and the target """ mx = (self.x + target.x)/2 my = (self.y + target.y)/2 return Point(mx, my)

This method is identical to the function, aside from some renaming. Its usage might be like this:
>>> p >>> q >>> r >>> r (4.0, = Point(3, 4) = Point(5, 12) = p.halfway(q) 8.0)

While this example assigns each point to a variable, this need not be done. Just as function calls are composable, method calls and object instantiation are also composable, leading to this alternative that uses no variables:
>>> print(Point(3, 4).halfway(Point(5, 12))) (4.0, 8.0)

15.9 A change of perspective


The original syntax for a function call, print_time(current_time), suggests that the function is the active agent. It says something like, Hey, print_time! Heres an object for you to print. In object-oriented programming, the objects are considered the active agents. An invocation like current_time.print_time() says Hey current_time! Please print yourself! In our early introduction to turtles, we used an object-oriented style, so that we said tess.forward(100), which asks the turtle to move itself forward by the given number of steps. 216 Chapter 15. Classes and Objects - the Basics

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

This change in perspective might be more polite, but it may not initially be obvious that it is useful. But sometimes shifting responsibility from the functions onto the objects makes it possible to write more versatile functions, and makes it easier to maintain and reuse code. The most important advantage of the object-oriented style is that it ts our mental chunking and real-life experience more accurately. In real life our cook method is part of our microwave oven we dont have a cook function sitting in the corner of the kitchen, into which we pass the microwave! Similarly, we use the cellphones own methods to send an sms, or to change its state to silent. The functionality of real-world objects tends to be tightly bound up inside the objects themselves. OOP allows us to accurately mirror this when we organize our programs.

15.10 Objects can have state


Objects are most useful when we also need to keep some state that is updated from time to time. Consider a turtle object. Its state consists of things like its position, its heading, its color, and its shape. A method like left(90) updates the turtles heading, forward changes its position, and so on. For a bank account object, a main component of the state would be the current balance, and perhaps a log of all transactions. The methods would allow us to query the current balance, deposit new funds, or make a payment. Making a payment would include an amount, and a description, so that this could be added to the transaction log. Wed also want a method to show the transaction log.

15.11 Glossary
attribute One of the named data items that makes up an instance. class A user-dened compound type. A class can also be thought of as a template for the objects that are instances of it. (The iPhone is a class. By December 2010, estimates are that 50 million instances had been sold!) constructor Every class has a factory, called by the same name as the class, for making new instances. If the class has an initializer method, this method is used to get the attributes (i.e. the state) of the new object properly set up. initializer method A special method in Python (called __init__) that is invoked automatically to set a newly created objects attributes to their initial (factory-default) state. instance An object whose type is of some class. Instance and object are used interchangeably. instantiate To create an instance of a class, and to run its initializer. method A function that is dened inside a class denition and is invoked on instances of that class. object A compound data type that is often used to model a thing or concept in the real world. It bundles together the data and the operations that are relevant for that kind of data. Instance and object are used interchangeably. 15.10. Objects can have state 217

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

object-oriented programming A powerful style of programming in which data and the operations that manipulate it are organized into classes and methods. object-oriented language A language that provides features, such as user-dened classes and inheritance, that facilitate object-oriented programming.

15.12 Exercises
1. Rewrite the distance function from the chapter titled fruitful functions so that it takes two Points as parameters instead of four numbers. 2. Add a method reflect_x to Point which returns a new Point, one which is the reection of the point about the x-axis. For example, Point(3, 5).reflect_x() is (3, -5) 3. Add a method slope_from_origin which returns the slope of the line joining the origin to the point. For example,
>>> Point(4, 10).slope_from_origin() 2.5

What cases will cause your method to fail? 4. The equation of a straight line is y = ax + b, (or perhaps y = mx + c). The coefcients a and b completely describe the line. Write a method in the Point class so that if a point instance is given another point, it will compute the equation of the straight line joining the two points. It must return the two coefcients as a tuple of two values. For example,
>>> print(Point(4, 11).get_line_to(Point(6, 15))) >>> (2, 3)

This tells us that the equation of the line joining the two points is y = 2x + 3. When will your method fail? 5. Given four points that fall on the circumference of a circle, nd the midpoint of the circle. When will you function fail? Hint: You must know how to solve the geometry problem before you think of going anywhere near programming. You cannot program a solution to a problem if you dont understand what you want the computer to do! 6. Create a new class, SMS_store. The class will instantiate SMS_store objects, like your inbox or your outbox on your cellphone:
my_inbox = SMS_store()

This store can hold multiple SMS messages (i.e. its internal state will just be a list of messages). Each message will be represented as a tuple:
(has_been_viewed, from_number, time_arrived, text_of_SMS)

Your inbox object should provide these methods:

218

Chapter 15. Classes and Objects - the Basics

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

my_inbox.add_new_arrival(from_number, time_arrived, text_of_SMS) # Makes new SMS tuple, inserts it after other messages # in the store. When creating this message, its # has_been_viewed status is set False. my_inbox.message_count() # returns the number of sms messages in my_inbox my_inbox.get_unread_indexes() # returns list of indexes of all not-yet-viewed SMS messages my_inbox.get_message(i) # return (from_number, time_arrived, text_of_sms) for message[i] # Also change its state to "has been viewed". # If there is no message at position i, return None my_inbox.delete(i) my_inbox.clear() # delete the message at index i # delete all messages from inbox

Write the class, create a message store object, write tests for these methods, and implement the methods.

15.12. Exercises

219

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

220

Chapter 15. Classes and Objects - the Basics

CHAPTER

SIXTEEN

CLASSES AND OBJECTS - DIGGING A LITTLE DEEPER


16.1 Rectangles
Lets say that we want a class to represent a rectangle which is located somewhere in the XY plane. The question is, what information do we have to provide in order to specify such a rectangle? To keep things simple, assume that the rectangle is oriented either vertically or horizontally, never at an angle. There are a few possibilities: we could specify the center of the rectangle (two coordinates) and its size (width and height); or we could specify one of the corners and the size; or we could specify two opposing corners. A conventional choice is to specify the upper-left corner of the rectangle, and the size. Again, well dene a new class, and provide it with an initializer and a string converter method:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17

class Rectangle: """ A class to manufacture rectangle objects """ def __init__(self, posn, w, h): """ Initialize rectangle at posn, with width w, height h """ self.corner = posn self.width = w self.height = h def __str__(self): return "({0}, {1}, {2})" .format(self.corner, self.width, self.height) box = Rectangle(Point(0, 0), 100, 200) bomb = Rectangle(Point(100, 80), 5, 10) print("box: ", box) print("bomb: ", bomb)

# in my video game

To specify the upper-left corner, we have embedded a Point object (as we used it in the previous chapter) within our new Rectangle object! We create two new Rectangle objects, and then print them producing: 221

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

box: ((0, 0), 100, 200) bomb: ((100, 80), 5, 10)

The dot operator composes. The expression box.corner.x means, Go to the object that box refers to and select its attribute named corner; then go to that object and select its attribute named x. The gure shows the state of this object:

16.2 Objects are mutable


We can change the state of an object by making an assignment to one of its attributes. For example, to grow the size of a rectangle without changing its position, we could modify the values of width and height:
box.width += 50 box.height += 100

Of course, wed probably like to provide a method to encapsulate this inside the class. We will also provide another method to move the position of the rectangle elsewhere:
1 2 3 4 5 6 7 8 9 10 11 12

class Rectangle: # ... def grow(self, delta_width, delta_height): """ Grow (or shrink) this object by the deltas """ self.width += delta_width self.height += delta_height def move(self, dx, dy): """ Move this object by the deltas """ self.corner.x += dx self.corner.y += dy

Let us try this:


>>> r = Rectangle(Point(10,5), 100, 50) >>> print(r) ((10, 5), 100, 50) >>> r.grow(25, -10) >>> print(r) ((10, 5), 125, 40) >>> r.move(-10, 10) print(r) ((0, 15), 125, 40)

222

Chapter 16. Classes and Objects - Digging a little deeper

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

16.3 Sameness
The meaning of the word same seems perfectly clear until you give it some thought, and then you realize there is more to it than you expected. For example, if you say, Chris and I have the same car, you mean that his car and yours are the same make and model, but that they are two different cars. If you say, Chris and I have the same mother, you mean that his mother and yours are the same person. When you talk about objects, there is a similar ambiguity. For example, if two Points are the same, does that mean they contain the same data (coordinates) or that they are actually the same object? Weve already seen the is operator in the chapter on lists, where we talked about aliases: it allows us to nd out if two references refer to the same object:
>>> p1 = Point(3, 4) >>> p2 = Point(3, 4) >>> p1 is p2 False

Even though p1 and p2 contain the same coordinates, they are not the same object. If we assign p1 to p3, then the two variables are aliases of the same object:
>>> p3 = p1 >>> p1 is p3 True

This type of equality is called shallow equality because it compares only the references, not the contents of the objects. To compare the contents of the objects deep equality we can write a function called same_coordinates:
1 2

def same_coordinates(p1, p2): return (p1.x == p2.x) and (p1.y == p2.y)

Now if we create two different objects that contain the same data, we can use same_point to nd out if they represent points with the same coordinates.
>>> p1 = Point(3, 4) >>> p2 = Point(3, 4) >>> same_coordinates(p1, p2) True

Of course, if the two variables refer to the same object, they have both shallow and deep equality. Beware of == When I use a word, Humpty Dumpty said, in a rather scornful tone, it means just what I choose it to mean neither more nor less. Alice in Wonderland

16.3. Sameness

223

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Python has a powerful feature that allows a designer of a class to decide what an operation like == or < should mean. (Weve just shown how we can control how our own objects are converted to strings, so weve already made a start!) Well cover more detail later. But sometimes the implementors will attach shallow equality semantics, and sometimes deep equality, as shown in this little experiment:
1 2 3 4 5 6 7 8 9

p = Point(4, 2) s = Point(4, 2) print("== on Points returns", p == s) # by default, == on Point objects does a shallow equality test a = [2,3] b = [2,3] print("== on lists returns", a == b) # but by default, == does a deep equality test on lists

This outputs:
== on Points returns False == on lists returns True

So we conclude that even though the two lists (or tuples, etc.) are distinct objects with different memory addresses, for lists the == operator tests for deep equality, while in the case of points it makes a shallow test.

16.4 Copying
Aliasing can make a program difcult to read because changes made in one place might have unexpected effects in another place. It is hard to keep track of all the variables that might refer to a given object. Copying an object is often an alternative to aliasing. The copy module contains a function called copy that can duplicate any object:
>>> import copy >>> p1 = Point(3, 4) >>> p2 = copy.copy(p1) >>> p1 is p2 False >>> same_coordinates(p1, p2) True

Once we import the copy module, we can use the copy function to make a new Point. p1 and p2 are not the same point, but they contain the same data. To copy a simple object like a Point, which doesnt contain any embedded objects, copy is sufcient. This is called shallow copying. For something like a Rectangle, which contains a reference to a Point, copy doesnt do quite the right thing. It copies the reference to the Point object, so both the old Rectangle 224 Chapter 16. Classes and Objects - Digging a little deeper

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

and the new one refer to a single Point. If we create a box, b1, in the usual way and then make a copy, b2, using copy, the resulting state diagram looks like this:

This is almost certainly not what we want. In this case, invoking grow on one of the Rectangles would not affect the other, but invoking move on either would affect both! This behavior is confusing and error-prone. The shallow copy has created an alias to the Point that represents the corner. Fortunately, the copy module contains a function named deepcopy that copies not only the object but also any embedded objects. You will not be surprised to learn that this operation is called a deep copy.
>>> b2 = copy.deepcopy(b1)

Now b1 and b2 are completely separate objects.

16.5 Glossary
deep copy To copy the contents of an object as well as any embedded objects, and any objects embedded in them, and so on; implemented by the deepcopy function in the copy module. deep equality Equality of values, or two references that point to objects that have the same value. shallow copy To copy the contents of an object, including any references to embedded objects; implemented by the copy function in the copy module. shallow equality Equality of references, or two references that point to the same object.

16.6 Exercises
1. Add a method area to the Rectangle class that returns the area of any instance:
r = Rectangle(Point(0, 0), 10, 5) test(r.area(), 50)

2. Write a perimeter method in the Rectangle class so that we can nd the perimeter of any rectangle instance:
r = Rectangle(Point(0, 0), 10, 5) test(r.perimeter(), 30)

16.5. Glossary

225

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

3. Write a flip method in the Rectangle class that swaps the width and the height of any rectangle instance:
r = Rectangle(Point(100, 50), 10, 5) test(r.width, 10) test(r.height, 5) r.flip() test(r.width, 5) test(r.height, 10)

4. Write a new method in the Rectangle class to test if a Point falls within the rectangle. For this exercise, assume that a rectangle at (0,0) with width 10 and height 5 has open upper bounds on the width and height, i.e. it stretches in the x direction from [0 to 10), where 0 is included but 10 is excluded, and from [0 to 5) in the y direction. So it does not contain the point (10, 2). These tests should pass:
r = Rectangle(Point(0, 0), 10, 5) test(r.contains(Point(0, 0)), True) test(r.contains(Point(3, 3)), True) test(r.contains(Point(3, 7)), False) test(r.contains(Point(3, 5)), False) test(r.contains(Point(3, 4.99999)), True) test(r.contains(Point(-3, -3)), False)

5. In games, we often put a rectangular bounding box around our sprites. (A sprite is an object that can move about in the game, as we will see shortly.) We can then do collision detection between, say, bombs and spaceships, by comparing whether their rectangles overlap anywhere. Write a function to determine whether two rectangles collide. Hint: this might be quite a tough exercise! Think carefully about all the cases before you code.

226

Chapter 16. Classes and Objects - Digging a little deeper

CHAPTER

SEVENTEEN

PYGAME
PyGame is a package that is not part of the standard Python distribution, so if you do not already have it installed (i.e. import pygame fails), download and install a suitable version from http://pygame.org/download.shtml. These notes are based on PyGame 1.9.1, the most recent version at the time of writing. PyGame comes with a substantial set of tutorials, examples, and help, so there is ample opportunity to stretch yourself on the code. You may need to look around a bit to nd these resources, though: if youve installed PyGame on a Windows machine, for example, theyll end up in a folder like C:\Python31\Lib\site-packages\pygame\ where you will nd directories for docs and examples.

17.1 The game loop


The structure of the games well consider always follows this xed pattern:

227

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

In every game, in the setup section well create a window, load and prepare some content, and then enter the game loop. The game loop continuously does four main things: it polls for events i.e. asks the system whether events have occurred and responds appropriately, it updates whatever internal data structures or objects need changing, it draws the current state of the game into a (non-visible) surface, it puts the just-drawn surface on display.
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

import pygame def main(): """ Set up the game and run the main game loop """ pygame.init() # prepare the pygame module for use surfaceSz = 480 # Desired physical surface size, in pixels. # Create surface of (width, height), and its window. main_surface = pygame.display.set_mode((surfaceSz, surfaceSz))

# set up some data to describe a small rectangle and its color small_rect = (300, 200, 150, 90) some_color = (255, 0, 0) # A color is a mix of (Red, Green, Blue) while True: ev = pygame.event.poll() if ev.type == pygame.QUIT: break

# look for any event # window close button clicked? # ... leave game loop

# Update your game objects and data structures here... # We draw everything from scratch on each frame. # So first fill everything with the background color main_surface.fill((0, 200, 255)) # Overpaint a smaller rectangle on the main surface main_surface.fill(some_color, small_rect) # Now the surface is ready, tell pygame to display it! pygame.display.flip() pygame.quit() main() # once we leave the loop, close the window.

This program pops up a window which stays there until we close it:

228

Chapter 17. PyGame

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

PyGame does all its drawing onto rectangular surfaces. After initializing PyGame at line 5, we create a window holding our main surface. The main loop of the game extends from line 15 to 30, with the following key bits of logic: First (line 16) we poll to fetch the next event that might be ready for us. This step will always be followed by some conditional statements that will determine whether any event that were interested in has happened. Polling for the event consumes it, as far as PyGame is concerned, so we only get one chance to fetch and use each event. On line 17 we test whether the type of the event is the predened constant called pygame.QUIT. This is the event that well see when the user clicks the close button on the PyGame window. In response to this event, we leave the loop. Once weve left the loop, the code at line 32 closes window, and well return from function main. Your program could go on to do other things, or reinitialize pygame and create another window, but it will usually just end too. There are different kinds of events key presses, mouse motion, mouse clicks, joystick movement, and so on. It is usual that we test and handle all these cases with new code squeezed in before line 19. The general idea is handle events rst, then worry about the other stuff. At line 20 wed update objects or data for example, if we wanted to vary the color, position, or size of the rectangle were about to draw, wed re-assign some_color, and small_rect here. A modern way to write games (now that we have fast computers and fast graphics cards) is to redraw everything from scratch on every iteration of the game loop. So the rst thing we do at line 24 is ll the entire surface with a background color. The fill method of a surface takes two arguments the color to use for lling, and the rectangle to be lled. But the second argument is optional, and if it is left out the entire surface is lled. In line 27 we ll a second rectangle, this time using some_color. The placement and size of the rectangle are given by the tuple small_rect, a 4-element tuple (x, y, width, height). It is important to understand that the origin of the PyGames surface is at the top left corner (unlike the turtle module that puts its origin in the middle of the screen). So, 17.1. The game loop 229

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

if you wanted the rectangle closer to the top of the window, you need to make its y coordinate smaller. If your graphics display hardware tries to read from memory at the same time as the program is writing to that memory, they will interfere with each other, causing video noise and icker. To get around this, PyGame keeps two buffers in the main surface the back buffer that the program draws to, while the front buffer is being shown to the user. Each time the program has fully prepared its back buffer, it ips the back/front role of the two buffers. So the drawing on lines 24 and 27 does does not change what is seen on the screen until we flip the buffers, on line 30.

17.2 Displaying images and text


To draw an image on the main surface, we load the image, say a beach ball, into its own new surface. The main surface has a blit method that copies pixels from the beach ball surface into its own surface. When we call blit, we can specify where the beach ball should be placed on the main surface. The term blit is widely used in computer graphics, and means to make a fast copy of pixels from one area of memory to another. So in the setup section, before we enter the game loop, wed load the image, like this:
1

ball = pygame.image.load("ball.png")

and after line 28 in the program above, wed add this code to display our image at position (100,120):
1

main_surface.blit(ball, (100, 120))

To display text, we need do do three things. Before we enter the game loop, we instantiate a font object:
1 2

# Instantiate 16 point Courier font to draw text. my_font = pygame.font.SysFont(Courier, 16)

and after line 28, again, we use the fonts render method to create a new surface containing the pixels of the drawn text, and then, as in the case for images, we blit our new surface onto the main surface. Notice that render takes two extra parameters the second tells it whether to carefully smooth edges of the text while drawing (this process is called anti-aliasing), and the second is the color that we want the text text be. Here weve used (0,0,0) which is black:
1 2

the_text = my_font.render(Hello, world!, True, (0,0,0)) surface.blit(the_text, (10, 10))

Well demonstrate these two new features by counting the frames the iterations of the game loop and keeping some timing information. On each frame, well display the frame count, and the frame rate. We will only update the frame rate after every 500 frames, when well look at the timing interval and can do the calculations.
1 2 3

import pygame import time

# for graphics

230

Chapter 17. PyGame

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

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

def main(): pygame.init() # prepare the PyGame module for use main_surface = pygame.display.set_mode((480, 240)) # Load an image to draw. Substitute your own. # PyGame handles gif, jpg, png, etc. image types. ball = pygame.image.load("ball.png") # Create a font for rendering text my_font = pygame.font.SysFont(Courier, 16) frame_count = 0 frame_rate = 0 t0 = time.clock() while True: # look for an event from keyboard, mouse, joystick, etc. ev = pygame.event.poll() if ev.type == pygame.QUIT: # window close button clicked? break # leave game loop # do other bits of logic for the game here frame_count += 1 if frame_count % 500 == 0: t1 = time.clock() frame_rate = 500 / (t1-t0) t0 = t1 # Completely redraw the surface, starting with background main_surface.fill((0, 200, 255)) # Put a red rectangle somewhere on the surface main_surface.fill((255,0,0), (300, 100, 150, 90)) # Copy our image to the surface, at this (x,y) posn main_surface.blit(ball, (100, 120)) # Make a new surface with an image of the text the_text = my_font.render(Frame = {0}, rate = {1:.2f} fps .format(frame_count, frame_rate), True, (0,0,0)) # Copy the text surface to the main surface main_surface.blit(the_text, (10, 10)) # Now that everything is drawn, put it on display! pygame.display.flip() pygame.quit()

17.2. Displaying images and text

231

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

55

main()

The frame rate is close to ridiculous a lot faster than ones eye can process frames. (Commercial video games usually plan their action for 60 frames per second (fps).) Of course, our rate will drop once we start doing something a little more strenuous inside our game loop.

17.3 Drawing a board for the N queens puzzle


In section Eight queens puzzle, part 1 we solved our N queens puzzle. For the 8x8 board, one of the solutions was the list [6,4,2,0,5,7,1,3]. Lets now use PyGame to draw that chessboard with its queens. We begin with a background of black and red squares for the board. Perhaps we could create an image that we could load and draw, but that approach would need different background images for different size boards. Just drawing our own red and black rectangles of the appropriate size sounds like much more fun!
1 2 3 4 5 6 7 8 9 10 11 12 13

def draw_board(the_board): """ Draw a chess board with queens, from the_board. """ pygame.init() colors = [(255,0,0), (0,0,0)] n = len(the_board) surfaceSz = 480 sq_sz = surfaceSz // n surfaceSz = n * sq_sz # # # #

# set up colors [red, black]

this is an NxN chess board. Proposed physical surface size. sq_sz is length of a square. Adjust to exactly fit n squares.

# Create the surface of (width, height), and its window. surface = pygame.display.set_mode((surfaceSz, surfaceSz))

Here we precompute sq_sz, the integer size that each square will be, so that we can t the squares nicely into the available window. So if wed like the board to be 480x480, and were 232 Chapter 17. PyGame

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

drawing an 8x8 chessboard, then each square will need to have a size of 60 units. But we notice that a 7x7 board cannot t nicely into 480 were going to get some ugly border that our squares dont ll exactly. So we recompute the surface size to exactly t our squares before we create the window. Now lets draw the squares, in the game loop. Well need a nested loop: the outer loop will run over the rows of the chessboard, the inner loop over the columns:
1 2 3 4 5 6 7 8

# Draw a fresh background (a blank chess board) for row in range(n): # Draw each row of the board. c_indx = row % 2 # Change starting color on each row for col in range(n): # Run through cols drawing squares the_square = (col*sq_sz, row*sq_sz, sq_sz, sq_sz) surface.fill(colors[c_indx], the_square) # now flip the color index for the next square c_indx = (c_indx + 1) % 2

There are two important ideas in this code: rstly, we compute the rectangle to be lled from the row and col loop variables, multiplying them by the size of the square to get their position. And, of course, each square is a xed width and height. So the_square represents the rectangle to be lled on the current iteration of the loop. The second idea is that we have to alternate colors on every square. In the earlier setup code we created a list containing two colors, here we manipulate c_indx (which will always either have the value 0 or 1) to start each row on a color that is different from the previous rows starting color, and to switch colors each time a square is lled. This (together with the other fragments not shown to ip the surface onto the display) leads to the pleasing backgrounds like this, for different size boards:

Now, on to drawing the queens! Recall that our solution [6,4,2,0,5,7,1,3] means that in column 0 of the board we want a queen at row 6, at column 1 we want a queen at row 4, and so on. So we need a loop running over each queen:
1 2

for (col, row) in enumerate(the_board): # draw a queen at col, row...

In this chapter we already have a beach ball image, so well use that for our queens. In the setup code before our game loop, we load the ball image (as we did before), and in the body of the loop, we add the line:

17.3. Drawing a board for the N queens puzzle

233

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

surface.blit(ball, (col * sq_sz, row * sq_sz))

Were getting there, but those queens need to be centred in their squares! Our problem arises from the fact that both the ball and the rectangle have their upper left corner as their reference points. If were going to centre this ball in the square, we need to give it an extra offset in both the x and y direction. (Since the ball is round and the square is square, the offset in the two directions will be the same, so well just compute a single offset value, and use it in both directions.) The offset we need is half the (size of the square less the size of the ball). So well precompute this in the games setup section, after weve loaded the ball and determined the square size:
1

ball_offset = (sq_sz - ball.get_width()) // 2

Now we touch up the drawing code for the ball and were done:
1

surface.blit(ball, (col * sq_sz + ball_offset, row * q_sz + ball_offset))

We might just want to think about what would happen if the ball was bigger than the square. In that case, ball_offset would become negative. So it would still be centered in the square - it would just spill over the boundaries, or perhaps obscure the square entirely! Here is the complete program:
1 2 3 4 5 6 7 8 9 10 11 12 13

import pygame def draw_board(the_board): """ Draw a chess board with queens, as determined by the the_board. """ pygame.init() colors = [(255,0,0), (0,0,0)] n = len(the_board) surfaceSz = 480 sq_sz = surfaceSz // n surfaceSz = n * sq_sz # # # #

# set up colors [red, black]

this is an NxN chess board. Proposed physical surface size. sq_sz is length of a square. Adjust to exactly fit n squares.

234

Chapter 17. PyGame

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

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

# Create the surface of (width, height), and its window. surface = pygame.display.set_mode((surfaceSz, surfaceSz)) ball = pygame.image.load("ball.png") # Use an extra offset to centre the ball in its square. # If the square is too small, offset becomes negative, # but it will still be centered :-) ball_offset = (sq_sz-ball.get_width()) // 2 while True: # look for an event from keyboard, mouse, etc. ev = pygame.event.poll() if ev.type == pygame.QUIT: break; # Draw a fresh background (a blank chess board) for row in range(n): # Draw each row of the board. c_indx = row % 2 # Alternate starting color for col in range(n): # Run through cols drawing squares the_square = (col*sq_sz, row*sq_sz, sq_sz, sq_sz) surface.fill(colors[c_indx], the_square) # now flip the color index for the next square c_indx = (c_indx + 1) % 2 # Now that squares are drawn, draw the queens. for (col, row) in enumerate(the_board): surface.blit(ball, (col*sq_sz+ball_offset,row*sq_sz+ball_offset)) pygame.display.flip()

pygame.quit() if __name__ == __main__: draw_board([0, 5, 3, 1, 6, 4, 2]) # 7 x 7 to test window size draw_board([6, 4, 2, 0, 5, 7, 1, 3]) draw_board([9, 6, 0, 3, 10, 7, 2, 4, 12, 8, 11, 5, 1]) # 13 x 13 draw_board([11, 4, 8, 12, 2, 7, 3, 15, 0, 14, 10, 6, 13, 1, 5, 9])

There is one more thing worth reviewing here. The conditional statement on line 50 tests whether the name of the currently executing program is __main__. This allows us to distinguish whether this module is being run as a main program, or whether it has been imported elsewhere, and used as a module. If we run this module in Python, the test cases in lines 51-54 will be executed. However, if we import this module into another program (i.e. our N queens solver from earlier) the condition at line 50 will be false, and the statements on lines 51-54 wont run. In the section Eight queens puzzle, part 2 our main program looked like this: 17.3. Drawing a board for the N queens puzzle 235

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2 3 4 5 6 7 8 9 10 11 12 13 14

def main(): bd = list(range(8)) # generate the initial permutation numFound = 0 tries = 0 while numFound < 10: random.shuffle(bd) tries += 1 if not has_clashes(bd): print(Found solution {0} in {1} tries..format(bd, tries)) tries = 0 numFound += 1 main()

Now we just need two changes. At the top of that program, we import the module that weve been working on here (assume we called it draw_queens). (Youll have to ensure that the two modules are saved in the same folder.) Then after line 10 here we add a call to draw the solution that weve just discovered:
draw_queens.draw_board(bd)

And that gives a very satisfying combination of program that can search for solutions to the N queens problem, and when it nds each, it pops up the board showing the solution.

17.4 Sprites
A sprite is an object that can move about in a game, and has internal behaviour and state of its own. For example, a spaceship would be a sprite, the player would be a sprite, and bullets and bombs would all be sprites. Object oriented programming (OOP) is ideally suited to a situation like this: each object can have its own attributes and internal state, and a couple of methods. Lets have some fun with our N queens board. Instead of placing the queen in her nal position, wed like to drop her in from the top of the board, and let her fall into position, perhaps bouncing along the way. The rst encapsulation we need is to turn each of our queens into an object. Well keep a list of all the active sprites (i.e. a list of queen objects), and arrange two new things in our game loop: After handling events, but before drawing, call an update method on every sprite. This will give each sprite a chance to modify its internal state in some way perhaps change its image, or change its position, or rotate itself, or make itself grow a bit bigger or a bit smaller. Once all the sprites have updated themselves, the game loop can begin drawing - rst the background, and then call a draw method on each sprite in turn, and delegate (hand off) the task of drawing to the object itself. This is in line with the OOP idea that we dont say Hey, draw, show this queen!, but we prefer to say Hey, queen, draw youself!.

236

Chapter 17. PyGame

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

We start with a simple object, no movement or animation yet, just scaffolding, to see how to t all the pieces together:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15

class Queen_sprite: def __init__(self, img, target_posn): """ Create and initialize a queen for this target position on the board """ self.image = img self.target_posn = target_posn self.posn = target_posn def update(self): return

# do nothing for the moment.

def draw(self, target_surface): target_surface.blit(self.image, self.posn)

Weve given the sprite three attributes: an image to be drawn, a target position, and a current position. If were going to move the spite about, the current position may need to be different from the target, which is where we want the queen nally to end up. In this code at this time weve done nothing in the update method, and our draw method (which can probably remain this simple in future) simply draws itself at its current position on the surface that is provided by the caller. With its class denition in place, we now instantiate our N queens, put them into a list of sprites, and arrange for the game loop to call the update and draw methods on each frame. The new bits of code, and the revised game loop look like this:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22

all_sprites = []

# Keep a list of all sprites in the game

# Create a sprite object for each queen, and populate our list. for (col, row) in enumerate(the_board): a_queen = Queen_sprite(ball, (col*sq_sz+ball_offset, row*sq_sz+ball_offset)) all_sprites.append(a_queen) while True: # look for an event from keyboard, mouse, etc. ev = pygame.event.poll() if ev.type == pygame.QUIT: break; # Ask every sprite to update itself. for sprite in all_sprites: sprite.update() # Draw a fresh background (a blank chess board) # ... same as before ... # Ask every sprite to draw itself.

17.4. Sprites

237

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

23 24 25 26

for sprite in all_sprites: sprite.draw(surface) pygame.display.flip()

This works just like it did before, but our extra work in making objects for the queens has prepared the way for some more ambitious extensions. Let us begin with a falling queen object. At any instant, it will have a velocity i.e. a speed, in a certain direction. (We are only working with movement in the y direction, but use your imagination!) So in the objects update method, we want to change its current position by its velocity. If our N queens board is oating in space, velocity would stay constant, but hey, here on Earth we have gravity! Gravity changes the velocity on each time interval, so well want a ball that speeds up as it falls further. Gravity will be constant for all queens, so we wont keep it in the instances well just make it a variable in our module. Well make one other change too: we will start every queen at the top of the board, so that it can fall towards its target position. With these changes, we now get the following:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19

gravity = 0.0001 class Queen_sprite: def __init__(self, img, target_posn): self.image = img self.target_posn = target_posn (x, y) = target_posn self.posn = (x, 0) # start ball at top of its column self.y_velocity = 0 # with zero initial velocity def update(self): self.y_velocity += gravity # gravity changes velocity (x, y) = self.posn new_y_pos = y + self.y_velocity # velocity moves the ball self.posn = (x, new_y_pos) # to this new position. def draw(self, target_surface): # same as before. target_surface.blit(self.image, self.posn)

Making these changes gives us a new chessboard in which each queen starts at the top of its column, and speeds up, until it drops off the bottom of the board and disappears forever. A good start we have movement! The next step is to get the ball to bounce when it reaches its own target position. It is pretty easy to bounce something you just change the sign of its velocity, and it will move at the same speed in the opposite direction. Of course, if it is travelling up towards the top of the board it will be slowed down by gravity. (Gravity always sucks down!) And youll nd it bounces all the way up to where it began from, reaches zero velocity, and starts falling all over again. So well have bouncing balls that never settle. A realistic way to settle the object is to lose some energy (probably to friction) each time it bounces so instead of simply reversing the sign of the velocity, we multiply it by some 238 Chapter 17. PyGame

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

fractional factor say -0.65. This means the ball only retains 65% of its energy on each bounce, so it will, as in real life, stop bouncing after a short while, and settle on its ground. The only changes are in the update method, which now looks like this:
1 2 3 4 5 6 7 8 9 10 11 12

def update(self): self.y_velocity += gravity (x, y) = self.posn new_y_pos = y + self.y_velocity (target_x, target_y) = self.target_posn dist_to_go = target_y - new_y_pos

# unpack the position # how far to our floor?

if dist_to_go < 0: # Are we under floor? self.y_velocity = -0.65 * self.y_velocity # bounce new_y_pos = target_y + dist_to_go # move back above floor self.posn = (x, new_y_pos) # set our new position.

Heh, heh, heh! Were not going to show animated screenshots, so copy the code into your Python environment and see for yourself.

17.5 Events
The only kind of event were handled so far has been the QUIT event. But we can also detect keydown and keyup events, mouse motion, and mousebutton down or up events. Consult the PyGame documentation and follow the link to Event. When your program polls for and receives an event object from PyGame, its event type will determine what secondary information is available. Each event object carries a dictionary (which you may only cover in due course in these notes). The dictionary holds certain keys and values that make sense for the type of event. For example, if the type of event is MOUSEMOTION, well be able to nd the mouse position and information about the state of the mouse buttons in the dictionary attached to the event. Similarly, if the event is KEYDOWN, we can learn from the dictionary whick key went down, and whether any modier keys (shift, control, alt, etc.) are also down. You also get events when the game window becomes active (i.e. gets focus) or loses focus. The event object with type NOEVENT is returned if there are no events waiting. Events can be printed, allowing you to experiment and play around. So dropping these lines of code into the game loop directly after polling for any event is quite informative:
1 2

if ev.type != NOEVENT: print(ev)

# only print if it is interesting!

With this is place, hit the space bar and the escape key, and watch the events you get. Click your three mouse buttons. Move your mouse over the window. (This causes a vast cascade of events, so you may also need to lter those out of the printing.) Youll get output that looks something like this:

17.5. Events

239

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

<Event(17-VideoExpose {})> <Event(1-ActiveEvent {state: 1, gain: 0})> <Event(2-KeyDown {scancode: 57, key: 32, unicode: , mod: 0})> <Event(3-KeyUp {scancode: 57, key: 32, mod: 0})> <Event(2-KeyDown {scancode: 1, key: 27, unicode: \x1b, mod: 0})> <Event(3-KeyUp {scancode: 1, key: 27, mod: 0})> ... <Event(4-MouseMotion {buttons: (0, 0, 0), pos: (323, 194), rel: (-3, -1)}) <Event(4-MouseMotion {buttons: (0, 0, 0), pos: (322, 193), rel: (-1, -1)}) <Event(4-MouseMotion {buttons: (0, 0, 0), pos: (321, 192), rel: (-1, -1)}) <Event(4-MouseMotion {buttons: (0, 0, 0), pos: (319, 192), rel: (-2, 0)})> <Event(5-MouseButtonDown {button: 1, pos: (319, 192)})> <Event(6-MouseButtonUp {button: 1, pos: (319, 192)})> <Event(4-MouseMotion {buttons: (0, 0, 0), pos: (319, 191), rel: (0, -1)})> <Event(5-MouseButtonDown {button: 2, pos: (319, 191)})> <Event(5-MouseButtonDown {button: 5, pos: (319, 191)})> <Event(6-MouseButtonUp {button: 5, pos: (319, 191)})> <Event(6-MouseButtonUp {button: 2, pos: (319, 191)})> <Event(5-MouseButtonDown {button: 3, pos: (319, 191)})> <Event(6-MouseButtonUp {button: 3, pos: (319, 191)})> ... <Event(1-ActiveEvent {state: 1, gain: 0})> <Event(12-Quit {})>

So let us now make these changes to the code near the top of our game loop:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20

while True: # look for an event from keyboard, ev = pygame.event.poll() if ev.type == pygame.QUIT: break; if ev.type == pygame.KEYDOWN: key = ev.dict[key] if key == 27: break if key == ord(r): colors[0] = (255, 0, 0) elif key == ord(g): colors[0] = (0, 255, 0) elif key == ord(b): colors[0] = (0, 0, 255) mouse, etc.

# on Escape key # leave the game loop # change to red + black # change to green + black # change to blue + black

if ev.type == pygame.MOUSEBUTTONDOWN: # mouse gone down? posn_of_click = ev.dict[pos] # get the coordinates print(posn_of_click) # just print them

Lines 7-16 show typical processing for a KEYDOWN event if a key has gone down, we test which key it is, and take some action. With this in place, we have another way to quit our queens program by hitting the escape key. Also, we can use keys to change the color of the board that is drawn. 240 Chapter 17. PyGame

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Finally, at line 20, we respond (pretty lamely) to the mouse button going down. As a nal exercise in this section, well write a better response handler to mouse clicks. What we will do is gure out if the user has clicked the mouse on one of our sprites. If there is a sprite under the mouse when the click occurs, well send the click to the sprite and let it respond in some sensible way. Well begin with some code that nds out which sprite is under the clicked position, perhaps none! We add a method to the class, contains_point, which returns True if the point is within the rectangle of the sprite:
1 2 3 4 5 6 7 8

def contains_point(self, pt): """ Return True if my sprite rectangle contains point pt """ (my_x, my_y) = self.posn my_width = self.image.get_width() my_height = self.image.get_height() (x, y) = pt return ( x >= my_x and x < my_x + my_width and y >= my_y and y < my_y + my_height)

Now in the game loop, once weve seen the mouse event, we determine which queen, if any, should be told to respond to the event:
1 2 3 4 5 6

if ev.type == pygame.MOUSEBUTTONDOWN: posn_of_click = ev.dict[pos] for sprite in all_sprites: if sprite.contains_point(posn_of_click): sprite.handle_click() break

And the nal thing is to write a new method called handle_click in the Queen_sprite class. When a sprite is clicked, well just add some velocity in the up direction, i.e. kick it back into the air.
1 2

def handle_click(self): self.y_velocity += -0.3

# kick it up

With these changes we have a playable game! See if you can keep all the balls on the move, not allowing any one to settle!

17.6 A wave of animation


Many games have sprites that are animated: they crouch, jump and shoot. How do they do that? Consider this sequence of 10 images: if we display them in quick succession, Duke will wave at us. (Duke is a friendly visitor from the kingdom of Javaland.)

17.6. A wave of animation

241

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

A compound image containing smaller patches which are intended for animation is called a sprite sheet. Download this sprite sheet by right-clicking in your browser and saving it in your working directory with the name duke_sprintesheet.png. The sprite sheet has been quite carefully prepared: each of the 10 patches are spaced exactly 50 pixels apart. So, assuming we want to draw patch number 4 (numbering from 0), we want to draw only the rectangle that starts at x position 200, and is 50 pixels wide, within the sprite sheet. Here weve shown the patches and highlighted the patch we want to draw.

The blit method weve been using for copying pixels from one surface to another can copy a sub-rectangle of the source surface. So the grand idea here is that each time we draw Duke, we wont blit the whole sprite sheet. Instead well provide an extra rectangle argument that determines which portion of the sprite sheet will be blitted. Were going to add new code in this section to our existing N queens drawing game. What we want is to put some instances of Duke on the chessboard somewhere. If the user clicks on one of them, well get him to respond by waving back, for one cycle of his animation. But before we do that, we need another change. Up until now, our game loop has been running at really fast frame rates that are unpredictable. So weve chosen some magic numbers for gravity and for bouncing and kicking the ball on the basis of trial-and-error. If were going to start animating more sprites, we need to tame our game loop to operate at a xed, known frame rate. This will allow us to plan our animation better. PyGame gives us the tools to do this in just two lines of code. In the setup section of the game, we instantiate a new Clock object:
1

my_clock = pygame.time.Clock()

and right at the bottom of the game loop, we call a method on this object that limits the frame rate to whatever we specify. So lets plan our game and animation for 60 frames per second, by adding this line at the bottom of our game loop:
1

my_clock.tick(60)

# Waste time so that frame rate becomes 60 fps

Youll nd that you have to go back and adjust the numbers for gravity and kicking the ball now, to match this much slower frame rate. When we plan an animation so that it only works sensibly at a xed frame rate, we say that weve baked the animation. In this case were baking our animations for 60 frames per second. To t into the existing framework that we already have for our queens board, we want to create a Duke_sprite class that has all the same methods as the Queens_sprite class. Then we can add one or more Duke instances onto our list of all_sprites, and our existing game loop will then call methods of the Duke instance. Let us start with skeleton scaffolding for the new class:
1 2

class Duke_sprite:

242

Chapter 17. PyGame

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18

def __init__(self, img, target_posn): self.image = img self.posn = target_posn def update(self): return def draw(self, target_surface): return def handle_click(self): return def contains_point(self, pt): # use code from Queens_sprite return

The only changes well need to the existing game are all in the setup section. We load up the new sprite sheet and instantiate a couple of instances of Duke, at the positions we want on the chessboard. So before entering the game loop, we add this code:
1 2 3 4 5 6 7 8 9 10

# Load the sprite sheet duke_sprite_sheet = pygame.image.load("duke_spritesheet.png") # Instantiate two duke instances, put them on the chessboard duke1 = Duke_sprite(duke_sprite_sheet,(sq_sz*2, 0)) duke2 = Duke_sprite(duke_sprite_sheet,(sq_sz*5, sq_sz)) # Add them to the list of sprites which our game loop manages all_sprites.append(duke1) all_sprites.append(duke2)

Now the game loop will test if each instance has been clicked, will call the click handler for that instance. It will also call update and draw for all sprites. All the remaining changes we need to make will be made in the methods of the Duke_sprite class. Lets begin with drawing one of the patches. Well introduce a new attribute curr_patch_num into the class. It holds a value between 0 and 9, and determines which patch to draw. So the job of the draw method is to compute the sub-rectangle of the patch to be drawn, and to blit only that portion of the spritesheet:
1 2 3 4

def draw(self, target_surface): patch_rect = (self.curr_patch_num * 50, 0, 50, self.image.get_height()) target_surface.blit(self.image, self.posn, patch_rect)

Now on to getting the animation to work. We need to arrange logic in update so that if were busy animating, we change the curr_patch_num every so often, and we also decide when to bring Duke back to his rest position, and stop the animation. An important issue is that the game loop frame rate in our case 60 fps is not the same as the animation rate the rate at which we want to change Dukes animation patches. So well plan Duke waves animation cycle for a duration of 1 second. In other words, we want to play out Dukes 10 animation 17.6. A wave of animation 243

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

patches over 60 calls to update. (This is how the baking of the animation takes place!) So well keep another animation frame counter in the class, which will be zero when were not animating, and each call to update will increment the counter up to 59, and then back to 0. We can then divide that animation counter by 6, to set the curr_patch_num variable to select the patch we want to show.
1 2 3 4

def update(self): if self.anim_frame_count > 0: self.anim_frame_count = (self.anim_frame_count + 1 ) % 60 self.curr_patch_num = self.anim_frame_count // 6

Notice that if anim_frame_count is zero, i.e. Duke is at rest, nothing happens here. But if we start the counter running, it will count up to 59 before settling back to zero. Notice also, that because anim_frame_count can only be a value between 0 and 59, the curr_patch_num will always stay between 0 and 9. Just what we require! Now how do we trigger the animation, and start it running? On the mouse click.
1 2 3

def handle_click(self): if self.anim_frame_count == 0: self.anim_frame_count = 5

Two things of interest here. We only start the animation if Duke is at rest. Clicks on Duke while he is already waving get ignored. And when we do start the animation, we set the counter to 5 this means that on the very next call to update the counter becomes 6, and the image changes. If we had set the counter to 1, we would have needed to wait for 5 more calls to update before anything happened a slight lag, but enough to make things feel sluggish. The nal touch-up is to initialize our two new attributes when we instantiate the class. Here is the code for the whole class now:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21

class Duke_sprite: def __init__(self, img, target_posn): self.image = img self.posn = target_posn self.anim_frame_count = 0 self.curr_patch_num = 0 def update(self): if self.anim_frame_count > 0: self.anim_frame_count = (self.anim_frame_count + 1 ) % 60 self.curr_patch_num = self.anim_frame_count // 6 def draw(self, target_surface): patch_rect = (self.curr_patch_num * 50, 0, 50, self.image.get_height()) target_surface.blit(self.image, self.posn, patch_rect) def contains_point(self, pt): """ Return True if my sprite rectangle contains (my_x, my_y) = self.posn

pt """

244

Chapter 17. PyGame

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

22 23 24 25 26 27 28 29 30

my_width = self.image.get_width() my_height = self.image.get_height() (x, y) = pt return ( x >= my_x and x < my_x + my_width and y >= my_y and y < my_y + my_height) def handle_click(self): if self.anim_frame_count == 0: self.anim_frame_count = 5

Now we have two extra Duke instances on our chessboard, and clicking on either causes that instance to wave.

17.7 Aliens - a case study


Find the example games with the PyGame package, (On a windows system, something like C:\Python3\Lib\site-packages\pygame\examples) and play the Aliens game. Then read the code, in an editor or Python environment that shows line numbers. It does a number of much more advanced things that we do, and relies on the PyGame framework for more of its logic. Here are some of the points that youll notice: The frame rate is deliberately constrained near the bottom of the game loop at line 311. If you change that number you can make the game very slow or unplayably fast! There are different kinds of sprites: Explosions, Shots, Bombs, Aliens and a Player. Some of these have more than one image by swapping the images, we get animation of the sprites, i.e. the Alien spacecraft lights change, and this is done at line 112. Different kinds of objects are referenced in different groups of sprites, and PyGame helps maintain these. This lets the program check for collisions between, say, the list of shots red by the player, and the list of spaceships that are attacking. PyGame does a lot of the hard work for you. Unlike our game, objects in the Aliens game have a limited lifetime, and have to get killed. For example, if you shoot, a Shot object is created if it reaches the top of the screen without expoding against anything, it has to be removed from the game. Lines

17.7. Aliens - a case study

245

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

141-142 do this. Similarly, when a falling bomb gets close to the ground (line 156), it instantiates a new Explosion sprite, and the bomb kills itself. There are random timings that add to the fun when to spawn the next Alien, when an Alien drops the next bomb, etc. The game plays sounds too: a less-than-relaxing loop sound, plus sounds for the shots and explosions.

17.8 Reections
Object oriented programming is a good organizational tool for software. In the examples in this chapter, weve started to use (and hopefully appreciate) these benets. Here we had N queens each with its own state, falling to its own oor level, bouncing, getting kicked, etc. We might have managed without the organizational power of objects perhaps we could have kept lists of velocities for each queen, and lists of target positions, and so on our code would likely have been much more complicated, ugly, and a lot poorer!

17.9 Glossary
animation rate The rate at which we play back successive patches to create the illusion of movement. In the sample we considered in this chapter, we played Dukes 10 patches over the duration of one second. Not the same as the frame rate. baked animation An animation that is designed to look good at a predetermined xed frame rate. This reduces the amount of computation that needs to be done when the game is running. High-end commercial games usually bake their animations. blit A verb used in computer graphics, meaning to make a fast copy of an image or pixels from a sub-rectangle of one image or surface to another surface or image. frame rate The rate at which the game loop executes and updates the display. game loop A loop that drives the logic of a game. It will usually poll for events, then update each of the objects in the game, then get everything drawn, and then put the newly drawn frame on display. pixel A single picture element, or dot, from which images are made. poll To ask whether something like a keypress or mouse movement has happened. Game loops usually poll to discover what events have occured. This is different from eventdriven programs like the ones seen in the chapter titled Events. In those cases, the button click or keypress event triggers the call of a handler function in your program, but this happens behind your back. sprite An active agent or element in a game, with its own state, position and behaviour. surface This is PyGames term for what the Turtle module calls a canvas. A surface is a rectangle of pixels used for displaying shapes and images.

246

Chapter 17. PyGame

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

17.10 Exercises
1. Have fun with Python, and with PyGame. 2. We deliberately left a bug in the code for animating Duke. If you click on one of the chessboard squares to the right of Duke, he waves anyway. Why? Find a one-line x for the bug. 3. Search Google for sprite sheet playing cards. Create a list [0..51] to represent an encoding of the 52 cards in a deck. Shufe the cards, slice off the top ve as your hand in a poker deal. Display the hand you have been dealt. 4. So the Aliens game is in outer space, without gravity. Shots y away forever, and bombs dont speed up when they fall. Add some gravity to the game. Decide if youre going to allow your own shots to fall back on your head and kill you. 5. Those pesky Aliens seem to pass right through each other! Change the game so that they collide, and destroy each other in a mighty explosion.

17.10. Exercises

247

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

248

Chapter 17. PyGame

CHAPTER

EIGHTEEN

RECURSION
18.1 Drawing Fractals
Recursion means dening something in terms of itself usually at some smaller scale, perhaps multiple times, to achieve your objective. For example, we might say A human being is someone whose mother is a human being. For our purposes, a fractal is a drawing which also has self-similar structure, where it can be dened in terms of itself. Let us start by looking at the famous Koch fractal. An order 0 Koch fractal is simply a straight line of a given size.

An order 1 Koch fractal is obtained like this: instead of drawing just one line, draw instead four smaller segments, in the pattern shown here:

Now what would happen if we repeated this Koch pattern again on each of the order 1 segments? Wed get this order 2 Koch fractal:

Repeating our pattern again gets us an order 3 Koch fractal:

249

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Now let us think about it the other way around. To draw a Koch fractal of order 3, we can simply draw four order 2 Koch fractals. But each of these in turn needs four order 1 Koch fractals, and each of those in turn needs four order 0 fractals. Ultimately, the only drawing that will take place is at order 0. This is very simple to code up in Python:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16

def koch(t, order, size): """ Make turtle t draw a Koch fractal of order and size. Leave the turtle facing the same direction. """ if order == 0: t.forward(size) else: koch(t, order-1, t.left(60) koch(t, order-1, t.right(120) koch(t, order-1, t.left(60) koch(t, order-1, # The base case is just a straight line

size/3) size/3) size/3) size/3)

# go 1/3 of the way

The key thing that is new here is that if order is not zero, koch calls itself recursively to get its job done. Lets make a simple observation and tighten up this code. Remember that turning right by 120 is the same as turning left by -120. So with a bit of clever rearrangement, we can use a loop instead of lines 10-16:
1 2 3 4 5 6 7

def koch(t, order, size): if order == 0: t.forward(size) else: for angle in [60, -120, 60, 0]: koch(t, order-1, size/3) t.left(angle)

The nal turn is 0 degrees so it has no effect. But it has allowed us to nd a pattern and reduce seven lines of code to three, which will make things easier for our next observations. Recursion, the high-level view One way to think about this is to convince yourself that the function works correctly when you call it for an order 0 fractal. Then do a mental leap of faith, saying the fairy godmother (or Python, if you can think of Python as your fairy godmother) knows how to handle the recursive level 0 calls for me on lines 11, 13, 15, and 17, so I dont need to think about that detail! All 250 Chapter 18. Recursion

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

I need to focus on is how to draw an order 1 fractal if I can assume the order 0 one is already working. Youre practicing mental abstraction ignoring the subproblem while you solve the big problem. If this mode of thinking works (and you should practice it!), then take it to the next level. Aha! now can I see that it will work when called for order 2 under the assumption that it is already working for level 1. And, in general, if I can assume the order n-1 case works, can I just solve the level n problem? Students of mathematics who have played with proofs of induction should see some very strong similarities here.

Recursion, the low-level operational view Another way of trying to understand recursion is to get rid of it! If we had separate functions to draw a level 3 fractal, a level 2 fractal, a level 1 fractal and a level 0 fractal, we could simplify the above code, quite mechanically, to a situation where there was no longer any recursion, like this:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17

def koch_0(t, size): t.forward(size) def koch_1(t, size): for angle in [60, -120, 60, 0]: koch_0(t, size/3) t.left(angle) def koch_2(t, size): for angle in [60, -120, 60, 0]: koch_1(t, size/3) t.left(angle) def koch_3(t, size): for angle in [60, -120, 60, 0]: koch_2(t, size/3) t.left(angle)

This trick of unrolling the recursion gives us an operational view of what happens. You can trace the program into koch_3, and from there, into koch_2, and then into koch_1, etc., all the way down the different layers of the recursion. This might be a useful hint to build your understanding. The mental goal is, however, to be able to do the abstraction!

18.1. Drawing Fractals

251

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

18.2 Recursive data structures


All of the Python data types we have seen can be grouped inside lists and tuples in a variety of ways. Lists and tuples can also be nested, providing a myriad possibilities for organizing data. The organization of data for the purpose of making it easier to use is called a data structure. Its election time and we are helping to compute the votes as they come in. Votes arriving from individual wards, precincts, municipalities, counties, and states are sometimes reported as a sum total of votes and sometimes as a list of subtotals of votes. After considering how best to store the tallies, we decide to use a nested number list, which we dene as follows: A nested number list is a list whose elements are either: 1. numbers 2. nested number lists Notice that the term, nested number list is used in its own denition. Recursive denitions like this are quite common in mathematics and computer science. They provide a concise and powerful way to describe recursive data structures that are partially composed of smaller and simpler instances of themselves. The denition is not circular, since at some point we will reach a list that does not have any lists as elements. Now suppose our job is to write a function that will sum all of the values in a nested number list. Python has a built-in function which nds the sum of a sequence of numbers:
>>> sum([1, 2, 8]) 11

For our nested number list, however, sum will not work:
>>> sum([1, 2, [11, 13], 8]) Traceback (most recent call last): File "<interactive input>", line 1, in <module> TypeError: unsupported operand type(s) for +: int and list >>>

The problem is that the third element of this list, [11, 13], is itself a list, so it cannot just be added to 1, 2, and 8.

18.3 Recursion
To sum all the numbers in our recursive nested number list we need to traverse the list, visiting each of the elements within its nested structure, adding any numeric elements to our sum, and repeating this process with any elements which are lists. Modern programming languages generally support recursion, which means that functions can call themselves within their denitions. Thanks to recursion, the Python code needed to sum the values of a nested number list is surprisingly short:

252

Chapter 18. Recursion

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2 3 4 5 6 7 8

def r_sum(nested_num_list): tot = 0 for element in nested_num_list: if type(element) == type([]): tot += r_sum(element) else: tot += element return tot

The body of r_sum consists mainly of a for loop that traverses nested_num_list. If element is a numerical value (the else branch), it is simply added to tot. If element is a list, then r_sum is called again, with the element as an argument. The statement inside the function denition in which the function calls itself is known as the recursive call. Recursion is truly one of the most beautiful and elegant tools in computer science. A slightly more complicated problem is nding the largest value in our nested number list:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23

def r_max(nxs): """ Find the maximum in a recursive structure of lists within other lists. Precondition: No lists or sublists are empty. """ largest = None first_time = True for e in nxs: if type(e) == type([]): val = r_max(e) else: val = e if first_time or val > largest: largest = val first_time = False return largest test(r_max([2, 9, [1, 13], 8, 6]), 13) test(r_max([2, [[100, 7], 90], [1, 13], 8, 6]), 100) test(r_max([[[13, 7], 90], 2, [1, 100], 8, 6]), 100) test(r_max(["joe", ["sam", "ben"]]), "sam")

Tests are included to provide examples of r_max at work. The added twist to this problem is nding a value for initializing largest. We cant just use nxs[0], since that may be either a element or a list. To solve this problem (at every recursive call) we initialize a boolean ag. When weve found the value of interest, we check to see whether this is the initializing (rst) value for largest, or a value that could potentially change largest. The two examples above each have a base case which does not lead to a recursive call: the case where the element is a number and not a list. Without a base case, youll have innite

18.3. Recursion

253

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

recursion, and your program will not work. Python stops after reaching a maximum recursion depth and returns a runtime error. See how this happens, by running this little script which we will call innite_recursion.py:
1 2 3 4 5

def recursion_depth(number): print("{0}, ".format(number), end=) recursion_depth(number + 1) recursion_depth(0)

After watching the messages ash by, you will be presented with the end of a long traceback that ends with a message like the following:
RuntimeError: maximum recursion depth exceeded ...

We would certainly never want something like this to happen to a user of one of our programs, so in the next chapter well see how errors, any kinds of errors, are handled in Python.

18.4 Case study: Fibonacci numbers


The famous Fibonacci sequence 0, 1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89, 134, ... was devised by Fibonacci (1170-1250), who used this to model the breeding of (pairs) of rabbits. If, in generation 7 you had 21 pairs in total, of which 13 were adults, then next generation the adults will all have bred new children, and the previous children will have grown up to become adults. So in generation 8 youll have 13+21=34, of which 21 are adults. This model to explain rabbit breeding made the simplifying assumption that rabbits never died. Scientists often make (unrealistic) simplifying assumptions and restrictions to make some headway with the problem. If we number the terms of the sequence from 0, we can describe each term recursively as the sum of the previous two terms:
fib(0) = 0 fib(1) = 1 fib(n) = fib(n-1) + fib(n-2)

for n >= 2

This translates very directly into some Python:


1 2 3 4 5

def fib(n): if n <= 1: return n t = fib(n-1) + fib(n-2) return t

This is a particularly inefcient algorithm, and well show one way of xing it when we learn about dictionaries:
1 2 3

import time t0 = time.clock() n = 35

254

Chapter 18. Recursion

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

4 5 6 7

result = fib(n) t1 = time.clock() print(fib({0}) = {1}, ({2:.2f} secs).format(n, result, t1-t0))

We get the correct result, but an exploding amount of work!


fib(35) = 9227465, (10.54 secs)

18.5 Example with recursive directories and les


The following program lists the contents of a directory and all its subdirectories.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23

import os def get_dirlist(path): """ Return a sorted list of all entries in path. This returns just the names, not the full path to the names. """ dirlist = os.listdir(path) dirlist.sort() return dirlist def print_files(path, prefix = ""): """ Print recursive listing of contents of path """ if prefix == "": # detect outermost call, print a heading print(Folder listing for, path) prefix = "| " dirlist = get_dirlist(path) for f in dirlist: print(prefix+f) fullE = os.path.join(path, f) if os.path.isdir(fullE): print_files(fullE, prefix +

# print the line # turn name into full path # if a directory, recurse. "| ")

Calling the function print_files with some folder name will produce output similar to this:
Folder listing for c:\python31\Lib\site-packages\pygame\examples | __init__.py | aacircle.py | aliens.py | arraydemo.py | blend_fill.py | blit_blends.py | camera.py | chimp.py

18.5. Example with recursive directories and les

255

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

| cursors.py | data | | alien1.png | | alien2.png | | alien3.png ...

18.6 Glossary
base case A branch of the conditional statement in a recursive function that does not give rise to further recursive calls. innite recursion A function that calls itself recursively without ever reaching any base case. Eventually, innite recursion causes a runtime error. recursion The process of calling a function that is already executing. recursive call The statement that calls an already executing function. Recursion can also be indirect function f can call g which calls h, and h could make a call back to f. recursive denition A denition which denes something in terms of itself. To be useful it must include base cases which are not recursive. In this way it differs from a circular denition. Recursive denitions often provide an elegant way to express complex data structures, like a directory that can contain other directories, or a menu that can contain other menus.

18.7 Exercises
1. Modify the Koch fractal program so that it draws a Koch snowake, like this:

2.

(a) Draw a Cesaro torn line fractal, of the order given by the user. We show four different lines of orders 0,1,2,3. In this example, the angle of the tear is 10 degrees.

256

Chapter 18. Recursion

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

(b) Four lines make a square. Use the code in part a) to draw cesaro squares. Varying the angle gives interesting effects experiment a bit, or perhaps let the user input the angle of the tear.

(a) (For the mathematically inclined). In the squares shown here, the higher-order drawings become a little larger. (Look at the bottom lines of each square - theyre not aligned.) This is because we just halved the drawn part of the line for each recursive subproblem. So weve grown the overall square by the width of the tear(s). Can you solve the geometry problem so that the total size of the subproblem case (including the tear) remains exactly the same size as the original?

3. A Sierpinski triangle of order 0 is an equilateral triangle. An order 1 triangle can be drawn by drawing 3 smaller triangles (shown slightly disconnected here, just to help our understanding). Higher order 2 and 3 triangles are also shown. Draw Sierpinski triangles of any order input by the user.

4. Adapt the above program to change the color of its three sub-triangles at some depth of recursion. The illustration below shows two cases: on the left, the color is changed at depth 0 (the outmost level of recursion), on the right, at depth 2. If the user supplies a negative depth, the color never changes. (Hint: add a new optional parameter colorChangeDepth (which defaults to -1), and make this one smaller on each recursive subcall. Then, in the section of code before you recurse, test whether the parameter 18.7. Exercises 257

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

is zero, and change color.)

5. Write a function, recursive_min, that returns the smallest value in a nested number list. Assume there are no empty lists or sublists:
test(recursive_min([2, 9, [1, 13], 8, 6]), 1) test(recursive_min([2, [[100, 1], 90], [10, 13], 8, 6]), 1) test(recursive_min([2, [[13, -7], 90], [1, 100], 8, 6]), -7) test(recursive_min([[[-13, 7], 90], 2, [1, 100], 8, 6]), -13)

6. Write a function count that returns the number of occurrences of target in a nested list:
test(count(2, []), 0) test(count(2, [2, 9, [2, 1, 13, 2], 8, [2, 6]]), 4) test(count(7, [[9, [7, 1, 13, 2], 8], [7, 6]]), 2) test(count(15, [[9, [7, 1, 13, 2], 8], [2, 6]]), 0) test(count(5, [[5, [5, [1, 5], 5], 5], [5, 6]]), 6) test(count(a, [[this,[a,[thing,a],a],is], [a,easy]]), 4)

7. Write a function flatten that returns a simple list containing all the values in a nested list:
test(flatten([2,9,[2,1,13,2],8,[2,6]]),[2,9,2,1,13,2,8,2,6]) test(flatten([[9,[7,1,13,2],8],[7,6]]),[9,7,1,13,2,8,7,6]) test(flatten([[9,[7,1,13,2],8],[2,6]]),[9,7,1,13,2,8,2,6]) test(flatten([[this,[a,[thing],a],is],[a,easy]]), [this,a,thing,a,is,a,easy]) test(flatten([]), [])

8. Rewrite the bonacci algorithm without using recursion. Can you nd bigger terms of the sequence? Can you nd fib(200)? 9. Use help to nd out what sys.getrecursionlimit() and sys.setrecursionlimit(n) do. Create several experiments similar to what was done in innite_recursion.py to test your understanding of how these module functions work. 10. Write a program that walks a directory structure (as in the last section of this chapter), but instead of printing lenames, it returns a list of all the full paths of les in the directory or the subdirectories. (Dont include directories in this list just les.) For example, the output list might have elements like this: 258 Chapter 18. Recursion

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

[C:\Python31\Lib\site-packages\pygame\docs\ref\mask.html, C:\Python31\Lib\site-packages\pygame\docs\ref\midi.html, ... C:\Python31\Lib\site-packages\pygame\examples\aliens.py, ... C:\Python31\Lib\site-packages\pygame\examples\data\boom.wav, ... ]

11. Write a program named litter.py that creates an empty le named trash.txt in each subdirectory of a directory tree given the root of the tree as an argument (or the current directory as a default). Now write a program named cleanup.py that removes all these les. Hint: Use the program from the example in the last section of this chapter as a basis for these two recursive programs. Because youre going to destroy les on your disks, you better get this right, or you risk losing les you care about. So excellent advice is that initially you should fake the deletion of the les just print the full path names of each le that you intend to delete. Once youre happy that your logic is correct, and you can see that youre not deleting the wrong things, you can replace the print statement with the real thing.

18.7. Exercises

259

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

260

Chapter 18. Recursion

CHAPTER

NINETEEN

EXCEPTIONS
19.1 Catching exceptions
Whenever a runtime error occurs, it creates an exception object. The program stops running at this point and Python prints out the traceback, which ends with a message describing the exception that occurred. For example, dividing by zero creates an exception:
>>> print(55/0) Traceback (most recent call last): File "<interactive input>", line 1, in <module> ZeroDivisionError: integer division or modulo by zero >>>

So does accessing a non-existent list item:


>>> a = [] >>> print(a[5]) Traceback (most recent call last): File "<interactive input>", line 1, in <module> IndexError: list index out of range >>>

Or trying to make an item assignment on a tuple:


>>> tup = (a, b, d, d) >>> tup[2] = c Traceback (most recent call last): File "<interactive input>", line 1, in <module> TypeError: tuple object does not support item assignment >>>

In each case, the error message on the last line has two parts: the type of error before the colon, and specics about the error after the colon. Sometimes we want to execute an operation that might cause an exception, but we dont want the program to stop. We can handle the exception using the try statement to wrap a region of code.

261

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

For example, we might prompt the user for the name of a le and then try to open it. If the le doesnt exist, we dont want the program to crash; we want to handle the exception:
1 2 3 4 5

filename = input(Enter a file name: ) try: f = open(filename, r) except: print(There is no file named, filename)

The try statement has three separate clauses, or parts, introduced by the keywords try ... except ... finally. The finally clause can be omitted, so well consider the two-clause version of the try statement rst. The try statement executes and monitors the statements in the rst block. If no exceptions occur, it skips the block under the except clause. If any exception occurs, it executes the statements in the except clause and then continues. We could encapsulate this capability in a function: exists which takes a lename and returns true if the le exists, false if it doesnt:
1 2 3 4 5 6 7

def exists(filename): try: f = open(filename) f.close() return True except: return False

A template to test if a le exists, without using exceptions The function weve just shown is not one wed recommend. It opens and closes the le, which is semantically different from asking does it exist?. How? Firstly, it might update some timestamps on the le. Secondly, it might tell you that there is no such le if some other program already happens to have the le open, or if your permissions settings dont allow you to open the le. Python provides a module called os.path within the os module. It provides a number of useful functions to work with paths, les and directories, so you should check out the help.
1 2 3 4 5

import os # This is the preferred way to check if a file exists. if os.path.isfile("c:/temp/testdata.txt"): ...

You can use multiple except clauses to handle different kinds of exceptions (see the Errors and Exceptions lesson from Python creator Guido van Rossums Python Tutorial for a more complete discussion of exceptions). So your program could do one thing if the le does not exist, but do something else if the le was in use by another program.

262

Chapter 19. Exceptions

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

19.2 Raising your own exceptions


Can your program deliberately cause its own exceptions? If your program detects an error condition, you can raise an exception. Here is an example that gets input from the user and checks that the number is non-negative:
1 2 3 4 5 6 7

def get_age(): age = int(input(Please enter your age: )) if age < 0: # Create a new instance of an exception myError = ValueError({0} is not a valid age.format(age)) raise myError return age

Line 5 creates an exception object, in this case, a ValueError object, which encapsulates your specic information about the error. Assume that in this case function A called B which called C which called D which called get_age. The raise statement on line 6 carries this object out as a kind of return value, and immediately exits from get_age() to its caller D. Then D again exits to its caller C, and C exits to B and so on, each returning the exception object to their caller, until it encounters a try ... except that can handle the exception. We call this unwinding the call stack. ValueError is one of the built-in exception types which most closely matches the kind of error we want to raise. The complete listing of built-in exceptions is found in the Built-in Exceptions section of the Python Library Reference, again by Pythons creator, Guido van Rossum. If the function that called get_age (or its caller, or their caller, ...) handles the error, then the program can carry on running; otherwise, Python prints the traceback and exits:
>>> get_age() Please enter your age: 42 42 >>> get_age() Please enter your age: -2 Traceback (most recent call last): File "<interactive input>", line 1, in <module> File "learn_exceptions.py", line 4, in get_age raise ValueError({0} is not a valid age.format(age)) ValueError: -2 is not a valid age >>>

The error message includes the exception type and the additional information that was provided when the exception object was rst created. It is often the case that lines 5 and 6 (creating the exception object, then raising the exception) are combined into a single statement, but there are really two different and independent things happening, so perhaps it makes sense to keep the two steps separate when we rst learn to work with exceptions. Here we show it all in a single statement:

19.2. Raising your own exceptions

263

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

raise ValueError({0} is not a valid age.format(age))

19.3 Revisiting an earlier example


Using exception handling, we can now modify our recursion_depth example from the previous chapter so that it stops when it reaches the maximum recursion depth allowed:
1 2 3 4 5 6 7 8

def recursion_depth(number): print("Recursion depth number", number) try: recursion_depth(number + 1) except: print("I cannot go any deeper into this wormhole.") recursion_depth(0)

Run this version and observe the results.

19.4 The finally clause of the try statement


A common programming pattern is to grab a resource of some kind e.g. we create a window for turtles to draw on, or we dial up a connection to our internet service provider, or we may open a le for writing. Then we perform some computation which may raise an exception, or may work without any problems. Whatever happens, we want to clean up the resources we grabbed e.g. close the window, disconnect our dial-up connection, or close the le. The finally clause of the try statement is the way to do just this. Consider this (somewhat contrived) example:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18

import turtle, time def show_poly(): try: win = turtle.Screen() tess = turtle.Turtle()

# Grab/create a resource, eg a window

# This dialog could be cancelled, # or the conversion to int might fail. n = int(input("How many sides do you want in your polygon?")) angle = 360 / n for i in range(n): # Draw the polygon tess.forward(10) tess.left(angle) time.sleep(3) # make program wait a few seconds finally: win.bye() # close the turtles window.

264

Chapter 19. Exceptions

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

19 20 21 22

show_poly() show_poly() show_poly()

In lines 19-21, show_poly is called three times. Each one creates a new window for its turtle, and draws a polygon with the number of sides input by the user. But what if the user enters a string that cannot be converted to an int? What if they close the dialog? Well get an exception, but even though weve had an exception, we still want to close the turtles window. Lines 16-17 does this for us. Whether we complete the statements in the try clause successfully or not, the finally block will always be executed. Notice that the exception is still unhandled only an except clause can handle an exception, so your program will still crash. But at least its turtle window will be closed before it crashes!

19.5 Glossary
exception An error that occurs at runtime. handle an exception To prevent an exception from causing your program to crash, by wrapping the block of code in a try / except construct. immutable data type A data type which cannot be modied. Assignments to elements or slices (sub-parts) of immutable types cause a runtime error. mutable data type A data type which can be modied. All mutable types are compound types. Lists and dictionaries (see next chapter) are mutable data types; strings and tuples are not. raise To create a deliberate exception by using the raise statement.

19.6 Exercises
1. Write a function named readposint that uses the input dialog to prompt the user for a positive integer and then checks the input to conrm that it meets the requirements. It should be able to handle inputs that cannot be converted to int, as well as negative ints, and edge cases (e.g. when the user closes the dialog, or does not enter anything at all.)

19.5. Glossary

265

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

266

Chapter 19. Exceptions

CHAPTER

TWENTY

DICTIONARIES
All of the compound data types we have studied in detail so far strings, lists, and tuples are sequence types, which use integers as indices to access the values they contain within them. Dictionaries are yet another kind of compound type. They are Pythons built-in mapping type. They map keys, which can be any immutable type, to values, which can be any type, just like the values of a list or tuple. As an example, we will create a dictionary to translate English words into Spanish. For this dictionary, the keys are strings. One way to create a dictionary is to start with the empty dictionary and add key-value pairs. The empty dictionary is denoted {}:
>>> eng2sp = {} >>> eng2sp[one] = uno >>> eng2sp[two] = dos

The rst assignment creates a dictionary named eng2sp; the other assignments add new keyvalue pairs to the dictionary. We can print the current value of the dictionary in the usual way:
>>> print(eng2sp) {two: dos, one: uno}

The key-value pairs of the dictionary are seperated by commas. Each pair contains a key and a value separated by a colon. The order of the pairs may not be what you expected. Python uses complex algorithms, designed for very fast access, to determine where the key-value pairs are stored in a dictionary. For our purposes we can think of this ordering as unpredicatable. Another way to create a dictionary is to provide a list of key-value pairs using the same syntax as the previous output:
>>> eng2sp = {one: uno, two: dos, three: tres}

It doesnt matter what order we write the pairs. The values in a dictionary are accessed with keys, not with indices, so there is no need to care about ordering. Here is how we use a key to look up the corresponding value:

267

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

>>> print(eng2sp[two]) dos

The key two yields the value dos. Lists, tuples, and strings have been called sequences, because their items occur in order. The dictionary is the rst compound type that weve seen that is not a sequence, so we cant index or slice a dictionary.

20.1 Dictionary operations


The del statement removes a key-value pair from a dictionary. For example, the following dictionary contains the names of various fruits and the number of each fruit in stock:
>>> inventory={apples:430,bananas:312,oranges:525,pears:217} >>> print(inventory) {pears: 217, apples: 430, oranges: 525, bananas: 312}

If someone buys all of the pears, we can remove the entry from the dictionary:
>>> del inventory[pears] >>> print(inventory) {apples: 430, oranges: 525, bananas: 312}

Or if were expecting more pears soon, we might just change the value associated with pears:
>>> inventory[pears] = 0 >>> print(inventory) {pears: 0, apples: 430, oranges: 525, bananas: 312}

A new shipment of bananas arriving could be handled like this:


>>> inventory[bananas] += 200 >>> print(inventory) {pears: 0, apples: 430, oranges: 525, bananas: 512}

The len function also works on dictionaries; it returns the number of key-value pairs:
>>> len(inventory) 4

20.2 Dictionary methods


Dictionaries have a number of useful built-in methods. The keys method returns what Python 3 calls a view of its underlying keys. A view object has some similarities to the range object we saw earlier it is a lazy promise, to deliver its elements when theyre needed by the rest of the program. We can iterate over the view, or turn the view into a list like this: 268 Chapter 20. Dictionaries

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2 3 4 5

for k in eng2sp.keys(): # the order of the ks is not defined print("Got key", k, "which maps to value", eng2sp[k]) ks = list(eng2sp.keys()) print(ks)

This produces this output:


Got key three which maps to value tres Got key two which maps to value dos Got key one which maps to value uno [three, two, one]

It is so common to iterate over the keys in a dictionary that you can omit the keys method call in the for loop iterating over a dictionary implicitly iterates over its keys:
1 2

for k in eng2sp: print("Got key", k)

The values method is similar; it returns a view object which can be turned into a list:
>>> list(eng2sp.values()) [tres, dos, uno]

The items method also returns a view, which promises a list of tuples one tuple for each key-value pair:
>>> list(eng2sp.items()) [(three, tres), (two, dos), (one, uno)]

Tuples are often useful for getting both the key and the value at the same time while you are looping:
1 2

for (k,v) in eng2sp.items(): print("Got",k,"that maps to",v)

This produces:
Got three that maps to tres Got two that maps to dos Got one that maps to uno

The in and not in operators can test if a key is in the dictionary:


>>> one in eng2sp True >>> six in eng2sp False >>> tres in eng2sp False

# note that in tests keys, not values.

This method can be very useful, since looking up a non-existent key in a dictionary causes a runtime error:

20.2. Dictionary methods

269

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

>>> eng2esp[dog] Traceback (most recent call last): ... KeyError: dog >>>

20.3 Aliasing and copying


As in the case of lists, because dictionaries are mutable, you need to be aware of aliasing. Whenever two variables refer to the same object, changes to one affect the other. If you want to modify a dictionary and keep a copy of the original, use the copy method. For example, opposites is a dictionary that contains pairs of opposites:
>>> opposites = {up: down, right: wrong, yes: no} >>> alias = opposites >>> copy = opposites.copy()

alias and opposites refer to the same object; copy refers to a fresh copy of the same dictionary. If we modify alias, opposites is also changed:
>>> alias[right] = left >>> opposites[right] left

If we modify copy, opposites is unchanged:


>>> copy[right] = privilege >>> opposites[right] left

20.4 Sparse matrices


We previously used a list of lists to represent a matrix. That is a good choice for a matrix with mostly nonzero values, but consider a sparse matrix like this one:

The list representation contains a lot of zeroes:


matrix = [[0, [0, [0, [0, [0, 0, 0, 2, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 3, 0], 0], 0], 0], 0]]

270

Chapter 20. Dictionaries

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

An alternative is to use a dictionary. For the keys, we can use tuples that contain the row and column numbers. Here is the dictionary representation of the same matrix:
>>> matrix = {(0, 3): 1, (2, 1): 2, (4, 3): 3}

We only need three key-value pairs, one for each nonzero element of the matrix. Each key is a tuple, and each value is an integer. To access an element of the matrix, we could use the [] operator:
>>> matrix[(0, 3)] 1

Notice that the syntax for the dictionary representation is not the same as the syntax for the nested list representation. Instead of two integer indices, we use one index, which is a tuple of integers. There is one problem. If we specify an element that is zero, we get an error, because there is no entry in the dictionary with that key:
>>> matrix[(1, 3)] KeyError: (1, 3)

The get method solves this problem:


>>> matrix.get((0, 3), 0) 1

The rst argument is the key; the second argument is the value get should return if the key is not in the dictionary:
>>> matrix.get((1, 3), 0) 0

get denitely improves the semantics of accessing a sparse matrix. Shame about the syntax.

20.5 Memos
If you played around with the fibo function from the chapter on recursion, you might have noticed that the bigger the argument you provide, the longer the function takes to run. Furthermore, the run time increases very quickly. On one of our machines, fib(20) nishes instantly, fib(30) takes about a second, and fib(40) takes roughly forever. To understand why, consider this call graph for fib with n = 4:

20.5. Memos

271

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

A call graph shows some function frames (instances when the function has been invoked), with lines connecting each frame to the frames of the functions it calls. At the top of the graph, fib with n = 4 calls fib with n = 3 and n = 2. In turn, fib with n = 3 calls fib with n = 2 and n = 1. And so on. Count how many times fib(0) and fib(1) are called. This is an inefcient solution to the problem, and it gets far worse as the argument gets bigger. A good solution is to keep track of values that have already been computed by storing them in a dictionary. A previously computed value that is stored for later use is called a memo. Here is an implementation of fib using memos:
1 2 3 4 5 6 7 8 9

alreadyknown = {0: 0, 1: 1} def fib(n): if n in alreadyknown: return alreadyknown[n] else: new_value = fib(n-1) + fib(n-2) alreadyknown[n] = new_value return new_value

The dictionary named alreadyknown keeps track of the Fibonacci numbers we already know. We start with only two pairs: 0 maps to 1; and 1 maps to 1. Whenever fib is called, it checks the dictionary to determine if it contains the result. If its there, the function can return immediately without making any more recursive calls. If not, it has to compute the new value. The new value is added to the dictionary before the function returns. Using this version of fib, our machines can compute fib(100) in an eyeblink.
>>> fib(100) 354224848179261915075

272

Chapter 20. Dictionaries

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

20.6 Counting letters


In the exercises in Chapter 8 we wrote a function that counted the number of occurrences of a letter in a string. A more general version of this problem is to form a frequency table of the letters in the string, that is, how many times each letter appears. Such a frequency table might be useful for compressing a text le. Because different letters appear with different frequencies, we can compress a le by using shorter codes for common letters and longer codes for letters that appear less frequently. Dictionaries provide an elegant way to generate a frequency table:
>>> letter_counts = {} >>> for letter in "Mississippi": ... letter_counts[letter] = letter_counts.get (letter, 0) + 1 ... >>> letter_counts {M: 1, s: 4, p: 2, i: 4}

We start with an empty dictionary. For each letter in the string, we nd the current count (possibly zero) and increment it. At the end, the dictionary contains pairs of letters and their frequencies. It might be more appealing to display the frequency table in alphabetical order. We can do that with the items and sort methods:
>>> letter_items = list(letter_counts.items()) >>> letter_items.sort() >>> print(letter_items) [(M, 1), (i, 4), (p, 2), (s, 4)]

Notice in the rst line we had to call the type conversion function list. That turns the promise we get from items into a list, a step that is needed before we can use the lists sort method.

20.7 Glossary
call graph A graph consisting of nodes which represent function frames (or invocations), and directed edges (lines with arrows) showing which frames gave rise to other frames. dictionary A collection of key-value pairs that maps from keys to values. The keys can be any immutable type, and the values can be any type. key A data item that is mapped to a value in a dictionary. Keys are used to look up values in a dictionary. key-value pair One of the pairs of items in a dictionary. Values are looked up in a dictionary by key. mapping type A mapping type is a data type comprised of a collection of keys and associated values. Pythons only built-in mapping type is the dictionary. Dictionaries implement the associative array abstract data type. 20.6. Counting letters 273

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

memo Temporary storage of precomputed values to avoid duplicating the same computation.

20.8 Exercises
1. Write a program that reads a string and returns a table of the letters of the alphabet in alphabetical order which occur in the string together with the number of times each letter occurs. Case should be ignored. A sample output of the program when the user enters the data ThiS is String with Upper and lower case Letters would look this this:
a c d e g h i l n o p r s t u w 2 1 1 5 1 2 4 2 2 1 2 4 5 5 1 2

2. Give the Python interpreters response to each of the following from a continuous interpreter session: (a) >>> d = {apples: 15, bananas: 35, grapes: 12}
>>> d[bananas]

(b) >>> d[oranges] = 20


>>> len(d)

(c) >>> grapes in d (d) >>> d[pears] (e) >>> d.get(pears, 0) (f) >>> fruits = list(d.keys())
>>> fruits.sort() >>> print(fruits)

274

Chapter 20. Dictionaries

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

(g) >>> del d[apples]


>>> apples in d

Be sure you understand why you get each result. Then apply what you have learned to ll in the body of the function below:
1 2 3 4 5 6 7 8 9 10

def add_fruit(inventory, fruit, quantity=0): return # make these tests work... new_inventory = {} add_fruit(new_inventory, strawberries, 10) test(strawberries in new_inventory, True) test(new_inventory[strawberries], 10) add_fruit(new_inventory, strawberries, 25) test(new_inventory[strawberries] , 35)

3. Write a program called alice_words.py that creates a text le named alice_words.txt containing an alphabetical listing of all the words, and the number of times each occurs, in the text version of Alices Adventures in Wonderland. (You can obtain a free plain text version of the book, along with many others, from http://www.gutenberg.org.) The rst 10 lines of your output le should look something like this:
Word Count ======================= a 631 a-piece 1 abide 1 able 1 about 94 above 3 absence 1 absurd 2

How many times does the word, alice, occur in the book? 4. What is the longest word in Alice in Wonderland? How many characters does it have?

20.8. Exercises

275

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

276

Chapter 20. Dictionaries

CHAPTER

TWENTYONE

EVEN MORE OOP


21.1 MyTime
As another example of a user-dened type, well dene a class called MyTime that records the time of day. Well provide an __init__ method to ensure that every instance is created with appropriate attributes and initialization. The class denition looks like this:
1 2 3 4 5 6 7

class MyTime: def __init__(self, hrs=0, mins=0, secs=0): """ Create a MyTime object initialized to hrs, mins, secs """ self.hours = hrs self.minutes = mins self.seconds = secs

We can instantiate a new MyTime object:


1

tim1 = MyTime(11, 59, 30)

The state diagram for the object looks like this:

Well leave it as an exercise for the readers to add a __str__ method so that MyTime objects can print themselves decently.

21.2 Pure functions


In the next few sections, well write two versions of a function called add_time, which calculates the sum of two MyTime objects. They will demonstrate two kinds of functions: pure functions and modiers. The following is a rough version of add_time:

277

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2 3 4 5 6

def add_time(t1, t2): h = t1.hours + t2.hours m = t1.minutes + t2.minutes s = t1.seconds + t2.seconds sum_t = MyTime(h, m, s) return sum_t

The function creates a new MyTime object and returns a reference to the new object. This is called a pure function because it does not modify any of the objects passed to it as parameters and it has no side effects, such as updating global variables, displaying a value, or getting user input. Here is an example of how to use this function. Well create two MyTime objects: current_time, which contains the current time; and bread_time, which contains the amount of time it takes for a breadmaker to make bread. Then well use add_time to gure out when the bread will be done.
>>> current_time = MyTime(9, 14, 30) >>> bread_time = MyTime(3, 35, 0) >>> done_time = add_time(current_time, bread_time) >>> print(done_time) 12:49:30

The output of this program is 12:49:30, which is correct. On the other hand, there are cases where the result is not correct. Can you think of one? The problem is that this function does not deal with cases where the number of seconds or minutes adds up to more than sixty. When that happens, we have to carry the extra seconds into the minutes column or the extra minutes into the hours column. Heres a better version of the function:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16

def add_time(t1, t2): h = t1.hours + t2.hours m = t1.minutes + t2.minutes s = t1.seconds + t2.seconds if s >= 60: s -= 60 m += 1 if m >= 60: m -= 60 h += 1 sum_t = MyTime(h, m, s) return sum_t

This function is starting to get bigger, and still doesnt work for all possible cases. Later we will suggest an alternative approach that yields better code.

278

Chapter 21. Even more OOP

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

21.3 Modiers
There are times when it is useful for a function to modify one or more of the objects it gets as parameters. Usually, the caller keeps a reference to the objects it passes, so any changes the function makes are visible to the caller. Functions that work this way are called modiers. increment, which adds a given number of seconds to a MyTime object, would be written most naturally as a modier. A rough draft of the function looks like this:
1 2 3 4 5 6 7 8 9 10

def increment(t, secs): t.seconds += secs if t.seconds >= 60: t.seconds -= 60 t.minutes += 1 if t.minutes >= 60: t.minutes -= 60 t.hours += 1

The rst line performs the basic operation; the remainder deals with the special cases we saw before. Is this function correct? What happens if the parameter seconds is much greater than sixty? In that case, it is not enough to carry once; we have to keep doing it until seconds is less than sixty. One solution is to replace the if statements with while statements:
1 2 3 4 5 6 7 8 9 10

def increment(t, seconds): t.seconds += seconds while t.seconds >= 60: t.seconds -= 60 t.minutes += 1 while t.minutes >= 60: t.minutes -= 60 t.hours += 1

This function is now correct when seconds is not negative, and when hours does not exceed 23, but it is not a particularly good solution.

21.4 Converting increment to a method


Once again, OOP programmers would prefer to put functions that work with MyTime objects into the MyTime class, so lets convert increment to a method. To save space, we will leave out previously dened methods, but you should keep them in your version:
1 2 3

class MyTime: #previous method definitions here...

21.3. Modiers

279

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

4 5 6 7 8 9 10 11 12 13

def increment(self, seconds): self.seconds += seconds while self.seconds >= 60: self.seconds -= 60 self.minutes += 1 while self.minutes >= 60: self.minutes -= 60 self.hours += 1

The transformation is purely mechanical - we move the denition into the class denition and (optionally) change the name of the rst parameter to self, to t with Python style conventions. Now we can invoke increment using the syntax for invoking a method.
1

current_time.increment(500)

Again, the object on which the method is invoked gets assigned to the rst parameter, self. The second parameter, seconds gets the value 500.

21.5 An Aha! insight


Often a high-level insight into the problem can make the programming much easier. In this case, the insight is that a MyTime object is really a three-digit number in base 60! The second component is the ones column, the minute component is the sixties column, and the hour component is the thirty-six hundreds column. When we wrote add_time and increment, we were effectively doing addition in base 60, which is why we had to carry from one column to the next. This observation suggests another approach to the whole problem we can convert a MyTime object into a single number and take advantage of the fact that the computer knows how to do arithmetic with numbers. The following method is added to the MyTime class to convert any instance into a corresponding number of seconds:
1 2 3 4 5 6 7 8

class MyTime: # ... def to_seconds(self): """ Return the number of seconds represented by this instance """ return self.hours * 3600 + self.minutes * 60 + self.seconds

Now, all we need is a way to convert from an integer back to a MyTime object. Supposing we have tsecs seconds, some integer division and mod operators can do this for us:

280

Chapter 21. Even more OOP

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2 3 4

hrs = tsecs // 3600 leftoversecs = tsecs % 3600 mins = leftoversecs // 60 secs = leftoversecs % 60

You might have to think a bit to convince yourself that this technique to convert from one base to another is correct. In OOP were really trying to wrap together the data and the operations that apply to it. So wed like to have this logic inside the MyTime class. A good solution is to rewrite the class initializer so that it can cope with initial values of seconds or minutes that are outside the normalized values. (A normalized time would be something like 3 hours 12 minutes and 20 seconds. The same time, but unnormalized could be 2 hours 70 minutes and 140 seconds.) Lets rewrite a more powerful initializer for MyTime:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15

class MyTime: # ... def __init__(self, hrs=0, mins=0, secs=0): """ Create a new MyTime object initialized to hrs, mins, secs. The values of mins and secs may be outside the range 0-59, but the resulting MyTime object will be normalized. """ # calculate total seconds to represent totalsecs = hrs*3600 + mins*60 + secs self.hours = totalsecs // 3600 # split in h, m, s leftoversecs = totalsecs % 3600 self.minutes = leftoversecs // 60 self.seconds = leftoversecs % 60

Now we can rewrite add_time like this:


1 2 3

def add_time(t1, t2): secs = t1.to_seconds() + t2.to_seconds() return MyTime(0, 0, secs)

This version is much shorter than the original, and it is much easier to demonstrate or reason that it is correct.

21.6 Generalization
In some ways, converting from base 60 to base 10 and back is harder than just dealing with times. Base conversion is more abstract; our intuition for dealing with times is better. But if we have the insight to treat times as base 60 numbers and make the investment of writing the conversions, we get a program that is shorter, easier to read and debug, and more reliable. It is also easier to add features later. For example, imagine subtracting two MyTime objects to nd the duration between them. The naive approach would be to implement subtraction with 21.6. Generalization 281

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

borrowing. Using the conversion functions would be easier and more likely to be correct. Ironically, sometimes making a problem harder (or more general) makes the programming easier, because there are fewer special cases and fewer opportunities for error. Specialization versus Generalization Computer Scientists are generally fond of specializing their types, while mathematicians often take the opposite approach, and generalize everything. What do we mean by this? If you ask a mathematician to solve a problem involving weekdays, days of the century, playing cards, time, or dominoes, their most likely response is to observe that all these objects can be represented by integers. Playing cards, for example, can be numbered from 0 to 51. Days within the century can be numbered. Mathematicians will say These things are enumerable the elements can be uniquely numbered (and we can reverse this numbering to get back to the original concept). So lets number them, and conne our thinking to integers. Luckily, we have powerful techniques and a good understanding of integers, and so our abstractions the way we tackle and simplify these problems is to try to reduce them to problems about integers. Computer Scientists tend to do the opposite. We will argue that there are many integer operations that are simply not meaningful for dominoes, or for days of the century. So well often dene new specialized types, like MyTime, because we can restrict, control, and specialize the operations that are possible. Object-oriented programming is particularly popular because it gives us a good way to bundle methods and specialized data into a new type. Both approaches are powerful problem-solving techniques. Often it may help to try to think about the problem from both points of view What would happen if I tried to reduce everything to very few primitive types?, versus What would happen if this thing had its own specialized type?

21.7 Another example


The after function should compare two times, and tell us whether the rst time is strictly after the second, e.g.
>>> t1 = MyTime(10, 55, 12) >>> t2 = MyTime(10, 48, 22) >>> after(t1, t2) True

# is t1 after t2?

This is slightly more complicated because it operates on two MyTime objects, not just one. But wed prefer to write it as a method anyway in this case, a method on the rst argument:
1 2 3 4

class MyTime: #previous method definitions here... def after(self, time2):

282

Chapter 21. Even more OOP

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

5 6 7 8 9 10 11 12 13 14 15 16 17 18

""" Return True if I am strictly greater than time2 """ if self.hours > time2.hours: return True if self.hours < time2.hours: return False if self.minutes > time2.minutes: return True if self.minutes < time2.minutes: return False if self.seconds > time2.seconds: return True return False

We invoke this method on one object and pass the other as an argument:
1 2

if current_time.after(done_time): print("The bread will be done before it starts!")

You can almost read the invocation like English: If the current time is after the done time, then... The logic of the if statements deserve special attention here. Lines 11-18 will only be reached if the two hour elds are the same. Similarly, the test at line 16 is only executed if both times have the same hours and the same minutes. Could we make this easier by using our Aha! insight and extra work from earlier, and reducing both times to integers? Yes, with spectacular results!
1 2 3 4 5 6

class MyTime: #previous method definitions here... def after(self, time2): """ Return True if I am strictly greater than time2 """ return self.to_seconds() > time2.to_seconds()

This is a great way to code this: if you want to tell if the rst time is after the second time, turn them both into integers and compare the integers.

21.8 Operator overloading


Some languages, including Python, make it possible to have different meanings for the same operator when applied to different types. For example, + in Python means quite different things for integers and for strings. This feature is called operator overloading. It is especially useful when programmers can also overload the operators for their own userdened types. For example, to override the addition operator +, we can provide a method named __add__:

21.8. Operator overloading

283

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2 3 4 5

class MyTime: # previously defined methods here... def __add__(self, other): return MyTime(0, 0, self.to_seconds() + other.to_seconds())

As usual, the rst parameter is the object on which the method is invoked. The second parameter is conveniently named other to distinguish it from self. To add two MyTime objects, we create and return a new MyTime object that contains their sum. Now, when we apply the + operator to MyTime objects, Python invokes the __add__ method that we have written:
>>> t1 = MyTime(1, 15, 42) >>> t2 = MyTime(3, 50, 30) >>> t3 = t1 + t2 >>> print(t3) 05:06:12

The expression t1 + t2 is equivalent to t1.__add__(t2), but obviously more elegant. As an exercise, add a method __sub__(self, other) that overloads the subtraction operator, and try it out. For the next couple of exercises well go back to the Point class dened in our rst chapter about objects, and overload some of its operators. Firstly, adding two points adds their respective (x, y) coordinates:
1 2 3 4 5

class Point: # previously defined methods here... def __add__(self, other): return Point(self.x + other.x,

self.y + other.y)

There are several ways to override the behavior of the multiplication operator: by dening a method named __mul__, or __rmul__, or both. If the left operand of * is a Point, Python invokes __mul__, which assumes that the other operand is also a Point. It computes the dot product of the two Points, dened according to the rules of linear algebra:
1 2

def __mul__(self, other): return self.x * other.x + self.y * other.y

If the left operand of * is a primitive type and the right operand is a Point, Python invokes __rmul__, which performs scalar multiplication:
1 2

def __rmul__(self, other): return Point(other * self.x,

other * self.y)

The result is a new Point whose coordinates are a multiple of the original coordinates. If other is a type that cannot be multiplied by a oating-point number, then __rmul__ will yield an error. This example demonstrates both kinds of multiplication: 284 Chapter 21. Even more OOP

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

>>> p1 = Point(3, 4) >>> p2 = Point(5, 7) >>> print(p1 * p2) 43 >>> print(2 * p2) (10, 14)

What happens if we try to evaluate p2 * 2? Since the rst parameter is a Point, Python invokes __mul__ with 2 as the second argument. Inside __mul__, the program tries to access the x coordinate of other, which fails because an integer has no attributes:
>>> print(p2 * 2) AttributeError: int object has no attribute x

Unfortunately, the error message is a bit opaque. This example demonstrates some of the difculties of object-oriented programming. Sometimes it is hard enough just to gure out what code is running.

21.9 Polymorphism
Most of the methods we have written only work for a specic type. When you create a new object, you write methods that operate on that type. But there are certain operations that you will want to apply to many types, such as the arithmetic operations in the previous sections. If many types support the same set of operations, you can write functions that work on any of those types. For example, the multadd operation (which is common in linear algebra) takes three parameters; it multiplies the rst two and then adds the third. We can write it in Python like this:
1 2

def multadd (x, y, z): return x * y + z

This function will work for any values of x and y that can be multiplied and for any value of z that can be added to the product. We can invoke it with numeric values:
>>> multadd (3, 2, 1) 7

Or with Points:
>>> p1 = Point(3, >>> p2 = Point(5, >>> print(multadd (11, 15) >>> print(multadd 44 4) 7) (2, p1, p2)) (p1, p2, 1))

21.9. Polymorphism

285

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

In the rst case, the Point is multiplied by a scalar and then added to another Point. In the second case, the dot product yields a numeric value, so the third parameter also has to be a numeric value. A function like this that can take arguments with different types is called polymorphic. As another example, consider the functionfront_and_back, which prints a list twice, forward and backward:
1 2 3 4 5

def front_and_back(front): import copy back = copy.copy(front) back.reverse() print(str(front) + str(back))

Because the reverse method is a modier, we make a copy of the list before reversing it. That way, this function doesnt modify the list it gets as a parameter. Heres an example that applies front_and_back to a list:
>>> myList = [1, 2, 3, 4] >>> front_and_back(myList) [1, 2, 3, 4][4, 3, 2, 1]

Of course, we intended to apply this function to lists, so it is not surprising that it works. What would be surprising is if we could apply it to a Point. To determine whether a function can be applied to a new type, we apply Pythons fundamental rule of polymorphism, called the duck typing rule: If all of the operations inside the function can be applied to the type, the function can be applied to the type. The operations in the front_and_back function include copy, reverse, and print. Not all programming languages dene polymorphism in this way. Look up duck typing, and see if you can gure out why it has this name. copy works on any object, and we have already written a __str__ method for Point objects, so all we need is a reverse method in the Point class:
1 2

def reverse(self): (self.x , self.y) = (self.y, self.x)

Then we can pass Points to front_and_back:


>>> p = Point(3, 4) >>> front_and_back(p) (3, 4)(4, 3)

The most interesting polymorphism is the unintentional kind, where you discover that a function you have already written can be applied to a type for which you never planned.

286

Chapter 21. Even more OOP

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

21.10 Glossary
dot product An operation dened in linear algebra that multiplies two Points and yields a numeric value. functional programming style A style of program design in which the majority of functions are pure. modier A function or method that changes one or more of the objects it receives as parameters. Most modiers are void. normalized Data is said to be normalized if it ts into some reduced range or set of rules. We usually normalize our angles to values in the range [0..360). We normalize minutes and seconds to be values in the range [0..60). And wed be surprised if the local store advertised its cold drinks at One dollar, two hundred and fty cents. operator overloading Extending built-in operators ( +, -, *, >, <, etc.) so that they do different things for different types of arguments. Weve seen early in the book how + is overloaded for numbers and strings, and here weve shown how to further overload it for user-dened types. polymorphic A function that can operate on more than one type. Notice the subtle distinction: overloading has different functions (all with the same name) for different types, whereas a polymorphic function is a single function that can work for a range of types. pure function A function that does not modify any of the objects it receives as parameters. Most pure functions are fruitful. scalar multiplication An operation dened in linear algebra that multiplies each of the coordinates of a Point by a numeric value.

21.11 Exercises
1. Write a boolean function between that takes two MyTime objects, t1 and t2, as arguments, and returns True if the invoking object falls between the two times. Assume t1 <= t2, and make the test closed at the lower bound and open at the upper bound, i.e. return True if t1 <= obj < t2 2. Turn the above function into a method in the MyTime class. 3. Overload the necessary operator(s) so that instead of having to write
if t1.after(t2): ...

we can use the more convenient


if t1 > t2: ...

4. Rewrite increment as a method that uses our Aha insight.

21.10. Glossary

287

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

5. Create some test cases for the increment method. Consider specically the case where the number of seconds to add to the time is negative. Fix up increment so that it handles this case if it does not do so already. (You may assume that you will never subtract more seconds than are in the time object.) 6. Can physical time be negative, or must time always move in the forward direction? Some serious physicists think this is not such a dumb question. See what you can nd on the Internet about this.

288

Chapter 21. Even more OOP

CHAPTER

TWENTYTWO

COLLECTIONS OF OBJECTS
22.1 Composition
By now, you have seen several examples of composition. One of the rst examples was using a method invocation as part of an expression. Another example is the nested structure of statements; you can put an if statement within a while loop, within another if statement, and so on. Having seen this pattern, and having learned about lists and objects, you should not be surprised to learn that you can create lists of objects. You can also create objects that contain lists (as attributes); you can create lists that contain lists; you can create objects that contain objects; and so on. In this chapter and the next, we will look at some examples of these combinations, using Card objects as an example.

22.2 Card objects


If you are not familiar with common playing cards, now would be a good time to get a deck, or else this chapter might not make much sense. There are fty-two cards in a deck, each of which belongs to one of four suits and one of thirteen ranks. The suits are Spades, Hearts, Diamonds, and Clubs (in descending order in bridge). The ranks are Ace, 2, 3, 4, 5, 6, 7, 8, 9, 10, Jack, Queen, and King. Depending on the game that you are playing, the rank of Ace may be higher than King or lower than 2. The rank is sometimes called the face-value of the card. If we want to dene a new object to represent a playing card, it is obvious what the attributes should be: rank and suit. It is not as obvious what type the attributes should be. One possibility is to use strings containing words like "Spade" for suits and "Queen" for ranks. One problem with this implementation is that it would not be easy to compare cards to see which had a higher rank or suit. An alternative is to use integers to encode the ranks and suits. By encode, we do not mean what some people think, which is to encrypt or translate into a secret code. What a computer scientist means by encode is to dene a mapping between a sequence of numbers and the items I want to represent. For example:

289

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Spades Hearts Diamonds Clubs

--> --> --> -->

3 2 1 0

An obvious feature of this mapping is that the suits map to integers in order, so we can compare suits by comparing integers. The mapping for ranks is fairly obvious; each of the numerical ranks maps to the corresponding integer, and for face cards:
Jack Queen King --> --> --> 11 12 13

The reason we are using mathematical notation for these mappings is that they are not part of the Python program. They are part of the program design, but they never appear explicitly in the code. The class denition for the Card type looks like this:
1 2 3 4

class Card: def __init__(self, suit=0, rank=0): self.suit = suit self.rank = rank

As usual, we provide an initialization method that takes an optional parameter for each attribute. To create some objects, representing say the 3 of Clubs and the Jack of Diamonds, use these commands:
1 2

three_of_clubs = Card(0, 3) card1 = Card(1, 11)

In the rst case above, for example, the rst argument, 0, represents the suit Clubs.

22.3 Class attributes and the __str__ method


In order to print Card objects in a way that people can easily read, we want to map the integer codes onto words. A natural way to do that is with lists of strings. We assign these lists to class attributes at the top of the class denition:
1 2 3 4 5 6 7 8 9 10 11

class Card: suits = ["Clubs", "Diamonds", "Hearts", "Spades"] ranks = ["narf", "Ace", "2", "3", "4", "5", "6", "7", "8", "9", "10", "Jack", "Queen", "King"] def __init__(self, suit=0, rank=0): self.suit = suit self.rank = rank def __str__(self): return (self.ranks[self.rank] + " of " + self.suits[self.suit])

290

Chapter 22. Collections of objects

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

A class attribute is dened outside of any method, and it can be accessed from any of the methods in the class. Inside __str__, we can use suits and ranks to map the numerical values of suit and rank to strings. For example, the expression self.suits[self.suit] means use the attribute suit from the object self as an index into the class attribute named suits, and select the appropriate string. The reason for the "narf" in the rst element in ranks is to act as a place keeper for the zeroeth element of the list, which will never be used. The only valid ranks are 1 to 13. This wasted item is not entirely necessary. We could have started at 0, as usual, but it is less confusing to encode the rank 2 as integer 2, 3 as 3, and so on. With the methods we have so far, we can create and print cards:
>>> card1 = Card(1, 11) >>> print(card1) Jack of Diamonds

Class attributes like suits are shared by all Card objects. The advantage of this is that we can use any Card object to access the class attributes:
>>> card2 = Card(1, 3) >>> print(card2) 3 of Diamonds >>> print(card2.suits[1]) Diamonds

Because every Card instance references the same class attribute, we have an aliasing situation. The disadvantage is that if we modify a class attribute, it affects every instance of the class. For example, if we decide that Jack of Diamonds should really be called Jack of Swirly Whales, we could do this:
>>> card1.suits[1] = "Swirly Whales" >>> print(card1) Jack of Swirly Whales

The problem is that all of the Diamonds just became Swirly Whales:
>>> print(card2) 3 of Swirly Whales

It is usually not a good idea to modify class attributes.

22.4 Comparing cards


For primitive types, there are six relational operators ( <, >, ==, etc.) that compare values and determine when one is greater than, less than, or equal to another. If you want your own types to be comparable using the syntax of these relational operators, you need to dene six corresponding special methods in your class.

22.4. Comparing cards

291

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Wed like to start with a single method named cmp that houses the logic of ordering. By convention, a comparison method takes two parameters, self and other, and returns 1 if the rst object is greater, -1 if the second object is greater, and 0 if they are equal to each other. Some types are completely ordered, which means that you can compare any two elements and tell which is bigger. For example, the integers and the oating-point numbers are completely ordered. Some types are unordered, which means that there is no meaningful way to say that one element is bigger than another. For example, the fruits are unordered, which is why you cannot compare apples and oranges, and you cannot meaningfully order a collection of images, or a collection of cellphones. Playing cards are partially ordered, which means that sometimes you can compare cards and sometimes not. For example, you know that the 3 of Clubs is higher than the 2 of Clubs, and the 3 of Diamonds is higher than the 3 of Clubs. But which is better, the 3 of Clubs or the 2 of Diamonds? One has a higher rank, but the other has a higher suit. In order to make cards comparable, you have to decide which is more important, rank or suit. To be honest, the choice is arbitrary. For the sake of choosing, we will say that suit is more important, because a new deck of cards comes sorted with all the Clubs together, followed by all the Diamonds, and so on. With that decided, we can write cmp:
1 2 3 4 5 6 7 8 9

def cmp(self, other): # check the suits if self.suit > other.suit: return 1 if self.suit < other.suit: return -1 # suits are the same... check ranks if self.rank > other.rank: return 1 if self.rank < other.rank: return -1 # ranks are the same... its a tie return 0

In this ordering, Aces appear lower than Deuces (2s). Now, we can dene the six special methods that do the overloading of each of the relational operators for us:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15

def __eq__(self, other): return self.cmp(other) == 0 def __le__(self, other): return self.cmp(other) <= 0 def __ge__(self, other): return self.cmp(other) >= 0 def __gt__(self, other): return self.cmp(other) > 0 def __lt__(self, other): return self.cmp(other) < 0

292

Chapter 22. Collections of objects

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

16 17

def __ne__(self, other): return self.cmp(other) != 0

With this machinery in place, the relational operators now work as wed like them to:
>>> card1 >>> card2 >>> card3 >>> card1 False >>> card1 True = = = < Card(1, 11) Card(1, 3) Card(1, 11) card2

== card3

22.5 Decks
Now that we have objects to represent Cards, the next logical step is to dene a class to represent a Deck. Of course, a deck is made up of cards, so each Deck object will contain a list of cards as an attribute. Many card games will need at least two different decks a red deck and a blue deck. The following is a class denition for the Deck class. The initialization method creates the attribute cards and generates the standard pack of fty-two cards:
1 2 3 4 5 6

class Deck: def __init__(self): self.cards = [] for suit in range(4): for rank in range(1, 14): self.cards.append(Card(suit, rank))

The easiest way to populate the deck is with a nested loop. The outer loop enumerates the suits from 0 to 3. The inner loop enumerates the ranks from 1 to 13. Since the outer loop iterates four times, and the inner loop iterates thirteen times, the total number of times the body is executed is fty-two (thirteen times four). Each iteration creates a new instance of Card with the current suit and rank, and appends that card to the cards list. With this in place, we can instantiate some decks:
1 2

redDeck = Deck() blueDeck = Deck()

22.6 Printing the deck


As usual, when we dene a new type we want a method that prints the contents of an instance. To print a Deck, we traverse the list and print each Card:
1 2

class Deck: ...

22.5. Decks

293

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

3 4 5

def print_deck(self): for card in self.cards: print(card)

Here, and from now on, the ellipsis ( ...) indicates that we have omitted the other methods in the class. As an alternative to print_deck, we could write a __str__ method for the Deck class. The advantage of __str__ is that it is more exible. Rather than just printing the contents of the object, it generates a string representation that other parts of the program can manipulate before printing, or store for later use. Here is a version of __str__ that returns a string representation of a Deck. To add a bit of pizzazz, it arranges the cards in a cascade where each card is indented one space more than the previous card:
1 2 3 4 5 6 7

class Deck: ... def __str__(self): s = "" for i in range(len(self.cards)): s = s + " " * i + str(self.cards[i]) + "\n" return s

This example demonstrates several features. First, instead of traversing self.cards and assigning each card to a variable, we are using i as a loop variable and an index into the list of cards. Second, we are using the string multiplication operator to indent each card by one more space than the last. The expression " " * i yields a number of spaces equal to the current value of i. Third, instead of using the print command to print the cards, we use the str function. Passing an object as an argument to str is equivalent to invoking the __str__ method on the object. Finally, we are using the variable s as an accumulator. Initially, s is the empty string. Each time through the loop, a new string is generated and concatenated with the old value of s to get the new value. When the loop ends, s contains the complete string representation of the Deck, which looks like this:
>>> redDeck = Deck() >>> print(redDeck) Ace of Clubs 2 of Clubs 3 of Clubs 4 of Clubs 5 of Clubs 6 of Clubs 7 of Clubs 8 of Clubs 9 of Clubs 10 of Clubs

294

Chapter 22. Collections of objects

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Jack of Clubs Queen of Clubs King of Clubs Ace of Diamonds 2 of Diamonds ...

And so on. Even though the result appears on 52 lines, it is one long string that contains newlines.

22.7 Shufing the deck


If a deck is perfectly shufed, then any card is equally likely to appear anywhere in the deck, and any location in the deck is equally likely to contain any card. To shufe the deck, we will use the randrange function from the random module. With two integer arguments, a and b, randrange chooses a random integer in the range a <= x < b. Since the upper bound is strictly less than b, we can use the length of a list as the second parameter, and we are guaranteed to get a legal index. For example, if rng has already been instantiated as a random number source, this expression chooses the index of a random card in a deck:
1

rng.randrange(0, len(self.cards))

An easy way to shufe the deck is by traversing the cards and swapping each card with a randomly chosen one. It is possible that the card will be swapped with itself, but that is ne. In fact, if we precluded that possibility, the order of the cards would be less than entirely random:
1 2 3 4 5 6 7 8 9

class Deck: ... def shuffle(self): import random rng = random.Random() # create a random generator num_cards = len(self.cards) for i in range(num_cards): j = rng.randrange(i, num_cards) (self.cards[i], self.cards[j]) = (self.cards[j], self.cards[i])

Rather than assume that there are fty-two cards in the deck, we get the actual length of the list and store it in num_cards. For each card in the deck, we choose a random card from among the cards that havent been shufed yet. Then we swap the current card ( i) with the selected card ( j). To swap the cards we use a tuple assignment:
1

(self.cards[i], self.cards[j]) = (self.cards[j], self.cards[i])

While this is a good shufing method, a random number generator object also has a shuffle method that can shufe elements in a list, in place. So we could rewrite this function to use the one provided for us: 22.7. Shufing the deck 295

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2 3 4 5 6

class Deck: ... def shuffle(self): import random rng = random.Random() rng.shuffle(self.cards)

# create a random generator # use its shuffle method

22.8 Removing and dealing cards


Another method that would be useful for the Deck class is remove, which takes a card as a parameter, removes it, and returns True if the card was in the deck and False otherwise:
1 2 3 4 5 6 7 8

class Deck: ... def remove(self, card): if card in self.cards: self.cards.remove(card) return True else: return False

The in operator returns True if the rst operand is in the second. If the rst operand is an object, Python uses the objects __eq__ method to determine equality with items in the list. Since the __eq__ we provided in the Card class checks for deep equality, the remove method checks for deep equality. To deal cards, we want to remove and return the top card. The list method pop provides a convenient way to do that:
1 2 3 4

class Deck: ... def pop(self): return self.cards.pop()

Actually, pop removes the last card in the list, so we are in effect dealing from the bottom of the deck. One more operation that we are likely to want is the boolean function is_empty, which returns True if the deck contains no cards:
1 2 3 4

class Deck: ... def is_empty(self): return len(self.cards) == 0

296

Chapter 22. Collections of objects

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

22.9 Glossary
encode To represent one type of value using another type of value by constructing a mapping between them. class attribute A variable that is dened inside a class denition but outside any method. Class attributes are accessible from any method in the class and are shared by all instances of the class. accumulator A variable used in a loop to accumulate a series of values, such as by concatenating them onto a string or adding them to a running sum.

22.10 Exercises
1. Modify cmp so that Aces are ranked higher than Kings.

22.9. Glossary

297

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

298

Chapter 22. Collections of objects

CHAPTER

TWENTYTHREE

INHERITANCE
23.1 Inheritance
The language feature most often associated with object-oriented programming is inheritance. Inheritance is the ability to dene a new class that is a modied version of an existing class. The primary advantage of this feature is that you can add new methods to a class without modifying the existing class. It is called inheritance because the new class inherits all of the methods of the existing class. Extending this metaphor, the existing class is sometimes called the parent class. The new class may be called the child class or sometimes subclass. Inheritance is a powerful feature. Some programs that would be complicated without inheritance can be written concisely and simply with it. Also, inheritance can facilitate code reuse, since you can customize the behavior of parent classes without having to modify them. In some cases, the inheritance structure reects the natural structure of the problem, which makes the program easier to understand. On the other hand, inheritance can make programs difcult to read. When a method is invoked, it is sometimes not clear where to nd its denition. The relevant code may be scattered among several modules. Also, many of the things that can be done using inheritance can be done as elegantly (or more so) without it. If the natural structure of the problem does not lend itself to inheritance, this style of programming can do more harm than good. In this chapter we will demonstrate the use of inheritance as part of a program that plays the card game Old Maid. One of our goals is to write code that could be reused to implement other card games.

23.2 A hand of cards


For almost any card game, we need to represent a hand of cards. A hand is similar to a deck, of course. Both are made up of a set of cards, and both require operations like adding and removing cards. Also, we might like the ability to shufe both decks and hands. A hand is also different from a deck. Depending on the game being played, we might want to perform some operations on hands that dont make sense for a deck. For example, in poker

299

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

we might classify a hand (straight, ush, etc.) or compare it with another hand. In bridge, we might want to compute a score for a hand in order to make a bid. This situation suggests the use of inheritance. If Hand is a subclass of Deck, it will have all the methods of Deck, and new methods can be added. In the class denition, the name of the parent class appears in parentheses:
1 2

class Hand(Deck): pass

This statement indicates that the new Hand class inherits from the existing Deck class. The Hand constructor initializes the attributes for the hand, which are name and cards. The string name identies this hand, probably by the name of the player that holds it. The name is an optional parameter with the empty string as a default value. cards is the list of cards in the hand, initialized to the empty list:
1 2 3 4

class Hand(Deck): def __init__(self, name=""): self.cards = [] self.name = name

For just about any card game, it is necessary to add and remove cards from the deck. Removing cards is already taken care of, since Hand inherits remove from Deck. But we have to write add:
1 2 3 4

class Hand(Deck): ... def add(self,card): self.cards.append(card)

Again, the ellipsis indicates that we have omitted other methods. The list append method adds the new card to the end of the list of cards.

23.3 Dealing cards


Now that we have a Hand class, we want to deal cards from the Deck into hands. It is not immediately obvious whether this method should go in the Hand class or in the Deck class, but since it operates on a single deck and (possibly) several hands, it is more natural to put it in Deck. deal should be fairly general, since different games will have different requirements. We may want to deal out the entire deck at once or add one card to each hand. deal takes two parameters, a list (or tuple) of hands and the total number of cards to deal. If there are not enough cards in the deck, the method deals out all of the cards and stops:
1 2 3 4

class Deck : ... def deal(self, hands, num_cards=999): num_hands = len(hands)

300

Chapter 23. Inheritance

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

5 6 7 8 9

for i in range(num_cards): if self.is_empty(): break card = self.pop() hand = hands[i % num_hands] hand.add(card)

# # # #

break if out of cards take the top card whose turn is next? add the card to the hand

The second parameter, num_cards, is optional; the default is a large number, which effectively means that all of the cards in the deck will get dealt. The loop variable i goes from 0 to num_cards-1. Each time through the loop, a card is removed from the deck using the list method pop, which removes and returns the last item in the list. The modulus operator ( %) allows us to deal cards in a round robin (one card at a time to each hand). When i is equal to the number of hands in the list, the expression i % num_hands wraps around to the beginning of the list (index 0).

23.4 Printing a Hand


To print the contents of a hand, we can take advantage of the printDeck and __str__ methods inherited from Deck. For example:
>>> deck = Deck() >>> deck.shuffle() >>> hand = Hand("frank") >>> deck.deal([hand], 5) >>> print(hand) Hand frank contains 2 of Spades 3 of Spades 4 of Spades Ace of Hearts 9 of Clubs

Its not a great hand, but it has the makings of a straight ush. Although it is convenient to inherit the existing methods, there is additional information in a Hand object we might want to include when we print one. To do that, we can provide a __str__ method in the Hand class that overrides the one in the Deck class:
1 2 3 4 5 6 7 8 9

class Hand(Deck) ... def __str__(self): s = "Hand " + self.name if self.is_empty(): s = s + " is empty\n" else: s = s + " contains\n" return s + Deck.__str__(self)

23.4. Printing a Hand

301

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Initially, s is a string that identies the hand. If the hand is empty, the program appends the words is empty and returns s. Otherwise, the program appends the word contains and the string representation of the Deck, computed by invoking the __str__ method in the Deck class on self. It may seem odd to send self, which refers to the current Hand, to a Deck method, until you remember that a Hand is a kind of Deck. Hand objects can do everything Deck objects can, so it is legal to send a Hand to a Deck method. In general, it is always legal to use an instance of a subclass in place of an instance of a parent class.

23.5 The CardGame class


The CardGame class takes care of some basic chores common to all games, such as creating the deck and shufing it:
1 2 3 4

class CardGame: def __init__(self): self.deck = Deck() self.deck.shuffle()

This is the rst case we have seen where the initialization method performs a signicant computation, beyond initializing attributes. To implement specic games, we can inherit from CardGame and add features for the new game. As an example, well write a simulation of Old Maid. The object of Old Maid is to get rid of cards in your hand. You do this by matching cards by rank and color. For example, the 4 of Clubs matches the 4 of Spades since both suits are black. The Jack of Hearts matches the Jack of Diamonds since both are red. To begin the game, the Queen of Clubs is removed from the deck so that the Queen of Spades has no match. The fty-one remaining cards are dealt to the players in a round robin. After the deal, all players match and discard as many cards as possible. When no more matches can be made, play begins. In turn, each player picks a card (without looking) from the closest neighbor to the left who still has cards. If the chosen card matches a card in the players hand, the pair is removed. Otherwise, the card is added to the players hand. Eventually all possible matches are made, leaving only the Queen of Spades in the losers hand. In our computer simulation of the game, the computer plays all hands. Unfortunately, some nuances of the real game are lost. In a real game, the player with the Old Maid goes to some effort to get their neighbor to pick that card, by displaying it a little more prominently, or perhaps failing to display it more prominently, or even failing to fail to display that card more prominently. The computer simply picks a neighbors card at random.

302

Chapter 23. Inheritance

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

23.6 OldMaidHand class


A hand for playing Old Maid requires some abilities beyond the general abilities of a Hand. We will dene a new class, OldMaidHand, that inherits from Hand and provides an additional method called remove_matches:
1 2 3 4 5 6 7 8 9 10 11 12 13

class OldMaidHand(Hand): def remove_matches(self): count = 0 original_cards = self.cards[:] for card in original_cards: match = Card(3 - card.suit, card.rank) if match in self.cards: self.cards.remove(card) self.cards.remove(match) print("Hand {0}: {1} matches {2}" .format(self.name, card, match)) count = count + 1 return count

We start by making a copy of the list of cards, so that we can traverse the copy while removing cards from the original. Since self.cards is modied in the loop, we dont want to use it to control the traversal. Python can get quite confused if it is traversing a list that is changing! For each card in the hand, we gure out what the matching card is and go looking for it. The match card has the same rank and the other suit of the same color. The expression 3 card.suit turns a Club (suit 0) into a Spade (suit 3) and a Diamond (suit 1) into a Heart (suit 2). You should satisfy yourself that the opposite operations also work. If the match card is also in the hand, both cards are removed. The following example demonstrates how to use remove_matches:
>>> game = CardGame() >>> hand = OldMaidHand("frank") >>> game.deck.deal([hand], 13) >>> print(hand) Hand frank contains Ace of Spades 2 of Diamonds 7 of Spades 8 of Clubs 6 of Hearts 8 of Spades 7 of Clubs Queen of Clubs 7 of Diamonds 5 of Clubs Jack of Diamonds 10 of Diamonds 10 of Hearts >>> hand.remove_matches()

23.6. OldMaidHand class

303

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Hand frank: 7 of Spades matches 7 of Clubs Hand frank: 8 of Spades matches 8 of Clubs Hand frank: 10 of Diamonds matches 10 of Hearts >>> print(hand) Hand frank contains Ace of Spades 2 of Diamonds 6 of Hearts Queen of Clubs 7 of Diamonds 5 of Clubs Jack of Diamonds

Notice that there is no __init__ method for the OldMaidHand class. We inherit it from Hand.

23.7 OldMaidGame class


Now we can turn our attention to the game itself. OldMaidGame is a subclass of CardGame with a new method called play that takes a list of players as a parameter. Since __init__ is inherited from CardGame, a new OldMaidGame object contains a new shufed deck:
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

class OldMaidGame(CardGame): def play(self, names): # remove Queen of Clubs self.deck.remove(Card(0,12)) # make a hand for each player self.hands = [] for name in names: self.hands.append(OldMaidHand(name)) # deal the cards self.deck.deal(self.hands) print("---------- Cards have been dealt") self.printHands() # remove initial matches matches = self.remove_all_matches() print("---------- Matches discarded, play begins") self.printHands() # play until all 50 cards are matched turn = 0 numHands = len(self.hands) while matches < 25: matches = matches + self.play_one_turn(turn)

304

Chapter 23. Inheritance

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

26 27 28 29

turn = (turn + 1) % numHands print("---------- Game is Over") self.printHands()

The writing of printHands() is left as an exercise. Some of the steps of the game have been separated into methods. remove_all_matches traverses the list of hands and invokes remove_matches on each:
1 2 3 4 5 6 7

class OldMaidGame(CardGame): ... def remove_all_matches(self): count = 0 for hand in self.hands: count = count + hand.remove_matches() return count

count is an accumulator that adds up the number of matches in each hand and returns the total. When the total number of matches reaches twenty-ve, fty cards have been removed from the hands, which means that only one card is left and the game is over. The variable turn keeps track of which players turn it is. It starts at 0 and increases by one each time; when it reaches numHands, the modulus operator wraps it back around to 0. The method playOneTurn takes a parameter that indicates whose turn it is. The return value is the number of matches made during this turn:
1 2 3 4 5 6 7 8 9 10 11 12

class OldMaidGame(CardGame): ... def play_one_turn(self, i): if self.hands[i].is_empty(): return 0 neighbor = self.find_neighbor(i) pickedCard = self.hands[neighbor].pop() self.hands[i].add(pickedCard) print("Hand", self.hands[i].name, "picked", pickedCard) count = self.hands[i].remove_matches() self.hands[i].shuffle() return count

If a players hand is empty, that player is out of the game, so he or she does nothing and returns 0. Otherwise, a turn consists of nding the rst player on the left that has cards, taking one card from the neighbor, and checking for matches. Before returning, the cards in the hand are shufed so that the next players choice is random. The method find_neighbor starts with the player to the immediate left and continues around the circle until it nds a player that still has cards:

23.7. OldMaidGame class

305

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2 3 4 5 6 7 8

class OldMaidGame(CardGame): ... def find_neighbor(self, i): numHands = len(self.hands) for next in range(1,numHands): neighbor = (i + next) % numHands if not self.hands[neighbor].is_empty(): return neighbor

If find_neighbor ever went all the way around the circle without nding cards, it would return None and cause an error elsewhere in the program. Fortunately, we can prove that that will never happen (as long as the end of the game is detected correctly). We have omitted the print_hands method. You can write that one yourself. The following output is from a truncated form of the game where only the top fteen cards (tens and higher) were dealt to three players. With this small deck, play stops after seven matches instead of twenty-ve.
>>> import cards >>> game = cards.OldMaidGame() >>> game.play(["Allen","Jeff","Chris"]) ---------- Cards have been dealt Hand Allen contains King of Hearts Jack of Clubs Queen of Spades King of Spades 10 of Diamonds Hand Jeff contains Queen of Hearts Jack of Spades Jack of Hearts King of Diamonds Queen of Diamonds Hand Chris contains Jack of Diamonds King of Clubs 10 of Spades 10 of Hearts 10 of Clubs Hand Jeff: Queen of Hearts matches Queen of Diamonds Hand Chris: 10 of Spades matches 10 of Clubs ---------- Matches discarded, play begins Hand Allen contains King of Hearts Jack of Clubs Queen of Spades King of Spades

306

Chapter 23. Inheritance

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

10 of Diamonds Hand Jeff contains Jack of Spades Jack of Hearts King of Diamonds Hand Chris contains Jack of Diamonds King of Clubs 10 of Hearts Hand Allen picked King of Diamonds Hand Allen: King of Hearts matches King of Diamonds Hand Jeff picked 10 of Hearts Hand Chris picked Jack of Clubs Hand Allen picked Jack of Hearts Hand Jeff picked Jack of Diamonds Hand Chris picked Queen of Spades Hand Allen picked Jack of Diamonds Hand Allen: Jack of Hearts matches Jack of Diamonds Hand Jeff picked King of Clubs Hand Chris picked King of Spades Hand Allen picked 10 of Hearts Hand Allen: 10 of Diamonds matches 10 of Hearts Hand Jeff picked Queen of Spades Hand Chris picked Jack of Spades Hand Chris: Jack of Clubs matches Jack of Spades Hand Jeff picked King of Spades Hand Jeff: King of Clubs matches King of Spades ---------- Game is Over Hand Allen is empty Hand Jeff contains Queen of Spades Hand Chris is empty

So Jeff loses.

23.8 Glossary
inheritance The ability to dene a new class that is a modied version of a previously dened class. parent class The class from which a child class inherits. child class A new class created by inheriting from an existing class; also called a subclass.

23.8. Glossary

307

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

23.9 Exercises
1. Add a method, print_hands, to the OldMaidGame class which traverses self.hands and prints each hand. 2. Dene a new kind of Turtle, TurtleGTX, that comes with some extra features: it can jump forward a given distance, and it has an odometer that keeps track of how far the turtle has travelled since it came off the production line. (The parent class has a number of synonyms like fd, forward, back, backward, and bk: for this exercise, just focus on putting this functionality on the forward method.) Think carefully about how to count the distance if the turtle is asked to move forward by a negative amount. (We would not want to buy a second-hand turtle whose odometer reading was faked because its previous owner drove it backwards around the block too often. Try this in a car near you, and see if the cars odometer counts up or down when you reverse.) 3. After travelling some random distance, your turtle should break down with a at tyre. After this happens, raise an exception whenever forward is called. Also provide a changeTyre method that can x the at.

308

Chapter 23. Inheritance

CHAPTER

TWENTYFOUR

LINKED LISTS
24.1 Embedded references
We have seen examples of attributes that refer to other objects, which we called embedded references. A common data structure, the linked list, takes advantage of this feature. Linked lists are made up of nodes, where each node contains a reference to the next node in the list. In addition, each node contains a unit of data called the cargo. A linked list is considered a recursive data structure because it has a recursive denition. A linked list is either: 1. the empty list, represented by None, or 2. a node that contains a cargo object and a reference to a linked list. Recursive data structures lend themselves to recursive methods.

24.2 The Node class


As usual when writing a new class, well start with the initialization and __str__ methods so that we can test the basic mechanism of creating and displaying the new type:
1 2 3 4 5 6 7

class Node: def __init__(self, cargo=None, next=None): self.cargo = cargo self.next = next def __str__(self): return str(self.cargo)

As usual, the parameters for the initialization method are optional. By default, both the cargo and the link, next, are set to None. The string representation of a node is just the string representation of the cargo. Since any value can be passed to the str function, we can store any value in a list. To test the implementation so far, we can create a Node and print it: 309

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

>>> node = Node("test") >>> print(node) test

To make it interesting, we need a list with more than one node:


>>> node1 = Node(1) >>> node2 = Node(2) >>> node3 = Node(3)

This code creates three nodes, but we dont have a list yet because the nodes are not linked. The state diagram looks like this:

To link the nodes, we have to make the rst node refer to the second and the second node refer to the third:
>>> node1.next = node2 >>> node2.next = node3

The reference of the third node is None, which indicates that it is the end of the list. Now the state diagram looks like this:

Now you know how to create nodes and link them into lists. What might be less clear at this point is why.

24.3 Lists as collections


Lists are useful because they provide a way to assemble multiple objects into a single entity, sometimes called a collection. In the example, the rst node of the list serves as a reference to the entire list. To pass the list as a parameter, we only have to pass a reference to the rst node. For example, the function print_list takes a single node as an argument. Starting with the head of the list, it prints each node until it gets to the end:
1 2 3

def print_list(node): while node != None: print(node, end= )

310

Chapter 24. Linked lists

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

4 5

node = node.next print()

To invoke this method, we pass a reference to the rst node:


>>> print_list(node1) 1 2 3

Inside print_list we have a reference to the rst node of the list, but there is no variable that refers to the other nodes. We have to use the next value from each node to get to the next node. To traverse a linked list, it is common to use a loop variable like node to refer to each of the nodes in succession. This diagram shows the value of list and the values that node takes on:

24.4 Lists and recursion


It is natural to express many list operations using recursive methods. For example, the following is a recursive algorithm for printing a list backwards: 1. Separate the list into two pieces: the rst node (called the head); and the rest (called the tail). 2. Print the tail backward. 3. Print the head. Of course, Step 2, the recursive call, assumes that we have a way of printing a list backward. But if we assume that the recursive call works the leap of faith then we can convince ourselves that this algorithm works. All we need are a base case and a way of proving that for any list, we will eventually get to the base case. Given the recursive denition of a list, a natural base case is the empty list, represented by None:
1 2 3 4 5 6

def print_backward(list): if list == None: return head = list tail = list.next print_backward(tail) print(head,end= )

24.4. Lists and recursion

311

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

The rst line handles the base case by doing nothing. The next two lines split the list into head and tail. The last two lines print the list. The end argument of the print statement keeps Python from printing a newline after each node. We invoke this method as we invoked print_list:
>>> print_backward(node1) 3 2 1

The result is a backward list. You might wonder why print_list and print_backward are functions and not methods in the Node class. The reason is that we want to use None to represent the empty list and it is not legal to invoke a method on None. This limitation makes it awkward to write list manipulating code in a clean object-oriented style. Can we prove that print_backward will always terminate? In other words, will it always reach the base case? In fact, the answer is no. Some lists will make this method crash.

24.5 Innite lists


There is nothing to prevent a node from referring back to an earlier node in the list, including itself. For example, this gure shows a list with two nodes, one of which refers to itself:

If we invoke print_list on this list, it will loop forever. If we invoke print_backward, it will recurse innitely. This sort of behavior makes innite lists difcult to work with. Nevertheless, they are occasionally useful. For example, we might represent a number as a list of digits and use an innite list to represent a repeating fraction. Regardless, it is problematic that we cannot prove that print_list and print_backward terminate. The best we can do is the hypothetical statement, If the list contains no loops, then these methods will terminate. This sort of claim is called a precondition. It imposes a constraint on one of the parameters and describes the behavior of the method if the constraint is satised. You will see more examples soon.

24.6 The fundamental ambiguity theorem


One part of print_backward might have raised an eyebrow:

312

Chapter 24. Linked lists

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2

head = list tail = list.next

After the rst assignment, head and list have the same type and the same value. So why did we create a new variable? The reason is that the two variables play different roles. We think of head as a reference to a single node, and we think of list as a reference to the rst node of a list. These roles are not part of the program; they are in the mind of the programmer. In general we cant tell by looking at a program what role a variable plays. This ambiguity can be useful, but it can also make programs difcult to read. We often use variable names like node and list to document how we intend to use a variable and sometimes create additional variables to disambiguate. We could have written print_backward without head and tail, which makes it more concise but possibly less clear:
1 2 3 4

def print_backward(list) : if list == None : return print_backward(list.next) print(list,end= )

Looking at the two function calls, we have to remember that print_backward treats its argument as a collection and print treats its argument as a single object. The fundamental ambiguity theorem describes the ambiguity that is inherent in a reference to a node: A variable that refers to a node might treat the node as a single object or as the rst in a list of nodes.

24.7 Modifying lists


There are two ways to modify a linked list. Obviously, we can change the cargo of one of the nodes, but the more interesting operations are the ones that add, remove, or reorder the nodes. As an example, lets write a method that removes the second node in the list and returns a reference to the removed node:
1 2 3 4 5 6 7 8 9

def removeSecond(list): if list == None: return first = list second = list.next # make the first node refer to the third first.next = second.next # separate the second node from the rest of the list second.next = None return second

Again, we are using temporary variables to make the code more readable. Here is how to use this method:

24.7. Modifying lists

313

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

>>> 1 2 >>> >>> 2 >>> 1 3

print_list(node1) 3 removed = removeSecond(node1) print_list(removed) print_list(node1)

This state diagram shows the effect of the operation:

What happens if you invoke this method and pass a list with only one element (a singleton)? What happens if you pass the empty list as an argument? Is there a precondition for this method? If so, x the method to handle a violation of the precondition in a reasonable way.

24.8 Wrappers and helpers


It is often useful to divide a list operation into two methods. For example, to print a list backward in the conventional list format [3, 2, 1] we can use the print_backward method to print 3, 2, but we need a separate method to print the brackets and the rst node. Lets call it print_backward_nicely:
1 2 3 4 5 6 7 8

def print_backward_nicely(list) : print("[", end= ) if list != None : head = list tail = list.next print_backward(tail) print(head, end= ) print("]", end= )

Again, it is a good idea to check methods like this to see if they work with special cases like an empty list or a singleton. When we use this method elsewhere in the program, we invoke print_backward_nicely directly, and it invokes print_backward on our behalf. In that sense, print_backward_nicely acts as a wrapper, and it uses print_backward as a helper.

314

Chapter 24. Linked lists

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

24.9 The LinkedList class


There are some subtle problems with the way we have been implementing lists. In a reversal of cause and effect, well propose an alternative implementation rst and then explain what problems it solves. First, well create a new class called LinkedList. Its attributes are an integer that contains the length of the list and a reference to the rst node. LinkedList objects serve as handles for manipulating lists of Node objects:
1 2 3 4

class LinkedList: def __init__(self): self.length = 0 self.head = None

One nice thing about the LinkedList class is that it provides a natural place to put wrapper functions like print_backward_nicely, which we can make a method of the LinkedList class:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15

class LinkedList: ... def print_backward(self): print("[", end= ) if self.head != None: self.head.print_backward() print("]", end= ) class Node: ... def print_backward(self): if self.next != None: tail = self.next tail.print_backward() print(self.cargo, end= )

Just to make things confusing, we renamed print_backward_nicely. Now there are two methods named print_backward: one in the Node class (the helper); and one in the LinkedList class (the wrapper). When the wrapper invokes self.head.print_backward, it is invoking the helper, because self.head is a Node object. Another benet of the LinkedList class is that it makes it easier to add or remove the rst element of a list. For example, addFirst is a method for LinkedLists; it takes an item of cargo as an argument and puts it at the beginning of the list:
1 2 3 4 5 6 7

class LinkedList: ... def addFirst(self, cargo): node = Node(cargo) node.next = self.head self.head = node self.length = self.length + 1

24.9. The LinkedList class

315

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

As usual, you should check code like this to see if it handles the special cases. For example, what happens if the list is initially empty?

24.10 Invariants
Some lists are well formed ; others are not. For example, if a list contains a loop, it will cause many of our methods to crash, so we might want to require that lists contain no loops. Another requirement is that the length value in the LinkedList object should be equal to the actual number of nodes in the list. Requirements like these are called invariants because, ideally, they should be true of every object all the time. Specifying invariants for objects is a useful programming practice because it makes it easier to prove the correctness of code, check the integrity of data structures, and detect errors. One thing that is sometimes confusing about invariants is that there are times when they are violated. For example, in the middle of addFirst, after we have added the node but before we have incremented length, the invariant is violated. This kind of violation is acceptable; in fact, it is often impossible to modify an object without violating an invariant for at least a little while. Normally, we require that every method that violates an invariant must restore the invariant. If there is any signicant stretch of code in which the invariant is violated, it is important for the comments to make that clear, so that no operations are performed that depend on the invariant.

24.11 Glossary
embedded reference A reference stored in an attribute of an object. linked list A data structure that implements a collection using a sequence of linked nodes. node An element of a list, usually implemented as an object that contains a reference to another object of the same type. cargo An item of data contained in a node. link An embedded reference used to link one object to another. precondition An assertion that must be true in order for a method to work correctly. fundamental ambiguity theorem A reference to a list node can be treated as a single object or as the rst in a list of nodes. singleton A linked list with a single node. wrapper A method that acts as a middleman between a caller and a helper method, often making the method easier or less error-prone to invoke.

316

Chapter 24. Linked lists

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

helper A method that is not invoked directly by a caller but is used by another method to perform part of an operation. invariant An assertion that should be true of an object at all times (except perhaps while the object is being modied).

24.12 Exercises
1. By convention, lists are often printed in brackets with commas between the elements, as in [1, 2, 3]. Modify print_list so that it generates output in this format.

24.12. Exercises

317

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

318

Chapter 24. Linked lists

CHAPTER

TWENTYFIVE

STACKS
25.1 Abstract data types
The data types you have seen so far are all concrete, in the sense that we have completely specied how they are implemented. For example, the Card class represents a card using two integers. As we discussed at the time, that is not the only way to represent a card; there are many alternative implementations. An abstract data type, or ADT, species a set of operations (or methods) and the semantics of the operations (what they do), but it does not not specify the implementation of the operations. Thats what makes it abstract. Why is that useful? 1. It simplies the task of specifying an algorithm if you can denote the operations you need without having to think at the same time about how the operations are performed. 2. Since there are usually many ways to implement an ADT, it might be useful to write an algorithm that can be used with any of the possible implementations. 3. Well-known ADTs, such as the Stack ADT in this chapter, are often implemented in standard libraries so they can be written once and used by many programmers. 4. The operations on ADTs provide a common high-level language for specifying and talking about algorithms. When we talk about ADTs, we often distinguish the code that uses the ADT, called the client code, from the code that implements the ADT, called the provider code.

25.2 The Stack ADT


In this chapter, we will look at one common ADT, the stack. A stack is a collection, meaning that it is a data structure that contains multiple elements. Other collections we have seen include dictionaries and lists. An ADT is dened by the operations that can be performed on it, which is called an interface. The interface for a stack consists of these operations:

319

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

__init__ Initialize a new empty stack. push Add a new item to the stack. pop Remove and return an item from the stack. The item that is returned is always the last one that was added. is_empty Check whether the stack is empty. A stack is sometimes called a last in, rst out or LIFO data structure, because the last item added is the rst to be removed.

25.3 Implementing stacks with Python lists


The list operations that Python provides are similar to the operations that dene a stack. The interface isnt exactly what it is supposed to be, but we can write code to translate from the Stack ADT to the built-in operations. This code is called an implementation of the Stack ADT. In general, an implementation is a set of methods that satisfy the syntactic and semantic requirements of an interface. Here is an implementation of the Stack ADT that uses a Python list:
1 2 3 4 5 6 7 8 9 10 11 12

class Stack : def __init__(self): self.items = [] def push(self, item): self.items.append(item) def pop(self): return self.items.pop() def is_empty(self): return (self.items == [])

A Stack object contains an attribute named items that is a list of items in the stack. The initialization method sets items to the empty list. To push a new item onto the stack, push appends it onto items. To pop an item off the stack, pop uses the homonymous ( same-named) list method to remove and return the last item on the list. Finally, to check if the stack is empty, is_empty compares items to the empty list. An implementation like this, in which the methods consist of simple invocations of existing methods, is called a veneer. In real life, veneer is a thin coating of good quality wood used in furniture-making to hide lower quality wood underneath. Computer scientists use this metaphor to describe a small piece of code that hides the details of an implementation and provides a simpler, or more standard, interface.

320

Chapter 25. Stacks

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

25.4 Pushing and popping


A stack is a generic data structure, which means that we can add any type of item to it. The following example pushes two integers and a string onto the stack:
>>> >>> >>> >>> s = Stack() s.push(54) s.push(45) s.push("+")

We can use is_empty and pop to remove and print all of the items on the stack:
1 2

while not s.is_empty(): print(s.pop(), end= )

The output is + 45 54. In other words, we just used a stack to print the items backward! Granted, its not the standard format for printing a list, but by using a stack, it was remarkably easy to do. You should compare this bit of code to the implementation of print_backward in the last chapter. There is a natural parallel between the recursive version of print_backward and the stack algorithm here. The difference is that print_backward uses the runtime stack to keep track of the nodes while it traverses the list, and then prints them on the way back from the recursion. The stack algorithm does the same thing, except that is use a Stack object instead of the runtime stack.

25.5 Using a stack to evaluate postx


In most programming languages, mathematical expressions are written with the operator between the two operands, as in 1 + 2. This format is called inx. An alternative used by some calculators is called postx. In postx, the operator follows the operands, as in 1 2 +. The reason postx is sometimes useful is that there is a natural way to evaluate a postx expression using a stack: 1. Starting at the beginning of the expression, get one term (operator or operand) at a time. If the term is an operand, push it on the stack. If the term is an operator, pop two operands off the stack, perform the operation on them, and push the result back on the stack. 2. When you get to the end of the expression, there should be exactly one operand left on the stack. That operand is the result.

25.6 Parsing
To implement the previous algorithm, we need to be able to traverse a string and break it into operands and operators. This process is an example of parsing, and the resultsthe individual 25.4. Pushing and popping 321

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

chunks of the string are called tokens. You might remember these words from Chapter 1. Python provides a split method in both string objects and the re (regular expression) module. A strings split method splits it into a list using a single character as a delimiter. For example:
>>> import string >>> "Now is the time".split(" ") [Now, is, the, time]

In this case, the delimiter is the space character, so the string is split at each space. The function re.split is more powerful, allowing us to provide a regular expression instead of a delimiter. A regular expression is a way of specifying a set of strings. For example, [A-z] is the set of all letters and [0-9] is the set of all numbers. The ^ operator negates a set, so [^0-9] is the set of everything that is not a number, which is exactly the set we want to use to split up postx expressions:
>>> import re >>> re.split("([^0-9])", "123+456*/") [123, +, 456, *, , /, ]

The resulting list includes the operands 123 and 456 and the operators * and /. It also includes two empty strings that are inserted after the operands.

25.7 Evaluating postx


To evaluate a postx expression, we will use the parser from the previous section and the algorithm from the section before that. To keep things simple, well start with an evaluator that only implements the operators + and *:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16

def eval_postfix(expr): import re token_list = re.split("([^0-9])", expr) stack = Stack() for token in token_list: if token == or token == : continue if token == +: sum = stack.pop() + stack.pop() stack.push(sum) elif token == *: product = stack.pop() * stack.pop() stack.push(product) else: stack.push(int(token)) return stack.pop()

The rst condition takes care of spaces and empty strings. The next two conditions handle operators. We assume, for now, that anything else must be an operand. Of course, it would be better to check for erroneous input and report an error message, but well get to that later. 322 Chapter 25. Stacks

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Lets test it by evaluating the postx form of (56+47)*2:


>>> print(eval_postfix ("56 47 + 2 *")) 206

Thats close enough.

25.8 Clients and providers


One of the fundamental goals of an ADT is to separate the interests of the provider, who writes the code that implements the ADT, and the client, who uses the ADT. The provider only has to worry about whether the implementation is correct in accord with the specication of the ADT and not how it will be used. Conversely, the client assumes that the implementation of the ADT is correct and doesnt worry about the details. When you are using one of Pythons built-in types, you have the luxury of thinking exclusively as a client. Of course, when you implement an ADT, you also have to write client code to test it. In that case, you play both roles, which can be confusing. You should make some effort to keep track of which role you are playing at any moment.

25.9 Glossary
abstract data type (ADT) A data type (usually a collection of objects) that is dened by a set of operations but that can be implemented in a variety of ways. client A program (or the person who wrote it) that uses an ADT. delimiter A character that is used to separate tokens, such as punctuation in a natural language. generic data structure A kind of data structure that can contain data of any type. implementation Code that satises the syntactic and semantic requirements of an interface. interface The set of operations that dene an ADT. inx A way of writing mathematical expressions with the operators between the operands. parse To read a string of characters or tokens and analyze its grammatical structure. postx A way of writing mathematical expressions with the operators after the operands. provider The code (or the person who wrote it) that implements an ADT. token A set of characters that are treated as a unit for purposes of parsing, such as the words in a natural language. veneer A class denition that implements an ADT with method denitions that are invocations of other methods, sometimes with simple transformations. The veneer does no signicant work, but it improves or standardizes the interface seen by the client.

25.8. Clients and providers

323

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

25.10 Exercises
1. Apply the postx algorithm to the expression 1 2 + 3 *.This example demonstrates one of the advantages of postxthere is no need to use parentheses to control the order of operations. To get the same result in inx, we would have to write (1 + 2) * 3. 2. Write a postx expression that is equivalent to 1 + 2 * 3.

324

Chapter 25. Stacks

CHAPTER

TWENTYSIX

QUEUES
This chapter presents two ADTs: the Queue and the Priority Queue. In real life, a queue is a line of customers waiting for service of some kind. In most cases, the rst customer in line is the next customer to be served. There are exceptions, though. At airports, customers whose ights are leaving soon are sometimes taken from the middle of the queue. At supermarkets, a polite customer might let someone with only a few items go rst. The rule that determines who goes next is called the queueing policy. The simplest queueing policy is called FIFO, for rst- in-rst-out. The most general queueing policy is priority queueing, in which each customer is assigned a priority and the customer with the highest priority goes rst, regardless of the order of arrival. We say this is the most general policy because the priority can be based on anything: what time a ight leaves; how many groceries the customer has; or how important the customer is. Of course, not all queueing policies are fair, but fairness is in the eye of the beholder. The Queue ADT and the Priority Queue ADT have the same set of operations. The difference is in the semantics of the operations: a queue uses the FIFO policy; and a priority queue (as the name suggests) uses the priority queueing policy.

26.1 The Queue ADT


The Queue ADT is dened by the following operations: __init__ Initialize a new empty queue. insert Add a new item to the queue. remove Remove and return an item from the queue. The item that is returned is the rst one that was added. is_empty Check whether the queue is empty.

26.2 Linked Queue


The rst implementation of the Queue ADT we will look at is called a linked queue because it is made up of linked Node objects. Here is the class denition: 325

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

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

class Queue: def __init__(self): self.length = 0 self.head = None def is_empty(self): return (self.length == 0) def insert(self, cargo): node = Node(cargo) node.next = None if self.head == None: # if list is empty the new node goes first self.head = node else: # find the last node in the list last = self.head while last.next: last = last.next # append the new node last.next = node self.length = self.length + 1 def remove(self): cargo = self.head.cargo self.head = self.head.next self.length = self.length - 1 return cargo

The methods is_empty and remove are identical to the LinkedList methods is_empty and remove_first. The insert method is new and a bit more complicated. We want to insert new items at the end of the list. If the queue is empty, we just set head to refer to the new node. Otherwise, we traverse the list to the last node and tack the new node on the end. We can identify the last node because its next attribute is None. There are two invariants for a properly formed Queue object. The value of length should be the number of nodes in the queue, and the last node should have next equal to None. Convince yourself that this method preserves both invariants.

26.3 Performance characteristics


Normally when we invoke a method, we are not concerned with the details of its implementation. But there is one detail we might want to knowthe performance characteristics of the method. How long does it take, and how does the run time change as the number of items in the collection increases? First look at remove. There are no loops or function calls here, suggesting that the runtime of this method is the same every time. Such a method is called a constant-time operation. In 326 Chapter 26. Queues

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

reality, the method might be slightly faster when the list is empty since it skips the body of the conditional, but that difference is not signicant. The performance of insert is very different. In the general case, we have to traverse the list to nd the last element. This traversal takes time proportional to the length of the list. Since the runtime is a linear function of the length, this method is called linear time. Compared to constant time, thats very bad.

26.4 Improved Linked Queue


We would like an implementation of the Queue ADT that can perform all operations in constant time. One way to do that is to modify the Queue class so that it maintains a reference to both the rst and the last node, as shown in the gure: The ImprovedQueue implementation looks like this:
1 2 3 4 5 6 7 8

class ImprovedQueue: def __init__(self): self.length = 0 self.head = None self.last = None def is_empty(self): return (self.length == 0)

So far, the only change is the attribute last. It is used in insert and remove methods:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15

class ImprovedQueue: ... def insert(self, cargo): node = Node(cargo) node.next = None if self.length == 0: # if list is empty, the new node is head and last self.head = self.last = node else: # find the last node last = self.last # append the new node last.next = node self.last = node self.length = self.length + 1

Since last keeps track of the last node, we dont have to search for it. As a result, this method is constant time. There is a price to pay for that speed. We have to add a special case to remove to set last to None when the last node is removed:

26.4. Improved Linked Queue

327

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2 3 4 5 6 7 8 9

class ImprovedQueue: ... def remove(self): cargo = self.head.cargo self.head = self.head.next self.length = self.length - 1 if self.length == 0: self.last = None return cargo

This implementation is more complicated than the Linked Queue implementation, and it is more difcult to demonstrate that it is correct. The advantage is that we have achieved the goal both insert and remove are constant-time operations.

26.5 Priority queue


The Priority Queue ADT has the same interface as the Queue ADT, but different semantics. Again, the interface is: __init__ Initialize a new empty queue. insert Add a new item to the queue. remove Remove and return an item from the queue. The item that is returned is the one with the highest priority. is_empty Check whether the queue is empty. The semantic difference is that the item that is removed from the queue is not necessarily the rst one that was added. Rather, it is the item in the queue that has the highest priority. What the priorities are and how they compare to each other are not specied by the Priority Queue implementation. It depends on which items are in the queue. For example, if the items in the queue have names, we might choose them in alphabetical order. If they are bowling scores, we might go from highest to lowest, but if they are golf scores, we would go from lowest to highest. As long as we can compare the items in the queue, we can nd and remove the one with the highest priority. This implementation of Priority Queue has as an attribute a Python list that contains the items in the queue.
1 2 3 4 5 6 7 8 9

class PriorityQueue: def __init__(self): self.items = [] def is_empty(self): return self.items == [] def insert(self, item): self.items.append(item)

328

Chapter 26. Queues

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

The initialization method, is_empty, and insert are all veneers on list operations. The only interesting method is remove:
1 2 3 4 5 6 7 8 9

class PriorityQueue: ... def remove(self): maxi = 0 for i in range(1, len(self.items)): if self.items[i] > self.items[maxi]: maxi = i item = self.items[maxi] self.items[maxi:maxi+1] = [] return item

At the beginning of each iteration, maxi holds the index of the biggest item (highest priority) we have seen so far. Each time through the loop, the program compares the i-eth item to the champion. If the new item is bigger, the value of maxi if set to i. When the for statement completes, maxi is the index of the biggest item. This item is removed from the list and returned. Lets test the implementation:
>>> >>> >>> >>> >>> >>> 14 13 12 11 q = PriorityQueue() q.insert(11) q.insert(12) q.insert(14) q.insert(13) while not q.is_empty(): print(q.remove())

If the queue contains simple numbers or strings, they are removed in numerical or alphabetical order, from highest to lowest. Python can nd the biggest integer or string because it can compare them using the built-in comparison operators. If the queue contains an object type, it has to provide a __gt__ method. When remove uses the > operator to compare items, it invokes the __gt__ for one of the items and passes the other as a parameter. As long as the __gt__ method works correctly, the Priority Queue will work.

26.6 The Golfer class


As an example of an object with an unusual denition of priority, lets implement a class called Golfer that keeps track of the names and scores of golfers. As usual, we start by dening __init__ and __str__:
1 2 3

class Golfer: def __init__(self, name, score): self.name = name

26.6. The Golfer class

329

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

4 5 6 7

self.score= score def __str__(self): return "{0:16}: {1}".format(self.name, self.score)

__str__ uses the format method to put the names and scores in neat columns. Next we dene a version of __gt__ where the lowest score gets highest priority. As always, __gt__ returns True if self is greater than other, and False otherwise.
1 2 3 4

class Golfer: ... def __gt__(self, other): return self.score < other.score

# less is more

Now we are ready to test the priority queue with the Golfer class:
>>> tiger = Golfer("Tiger Woods", 61) >>> phil = Golfer("Phil Mickelson", 72) >>> hal = Golfer("Hal Sutton", 69) >>> >>> pq = PriorityQueue() >>> pq.insert(tiger) >>> pq.insert(phil) >>> pq.insert(hal) >>> while not pq.is_empty(): print(pq.remove()) Tiger Woods : 61 Hal Sutton : 69 Phil Mickelson : 72

26.7 Glossary
constant time An operation whose runtime does not depend on the size of the data structure. FIFO First In, First Out, a queueing policy in which the rst member to arrive is the rst to be removed. linear time An operation whose runtime is a linear function of the size of the data structure. linked queue An implementation of a queue using a linked list. priority queue A queueing policy in which each member has a priority determined by external factors. The member with the highest priority is the rst to be removed. Priority Queue An ADT that denes the operations one might perform on a priority queue. queue An ordered set of objects waiting for a service of some kind. Queue An ADT that performs the operations one might perform on a queue. queueing policy The rules that determine which member of a queue is removed next.

330

Chapter 26. Queues

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

26.8 Exercises
1. Write an implementation of the Queue ADT using a Python list. Compare the performance of this implementation to the ImprovedQueue for a range of queue lengths. 2. Write an implementation of the Priority Queue ADT using a linked list. You should keep the list sorted so that removal is a constant time operation. Compare the performance of this implementation with the Python list implementation.

26.8. Exercises

331

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

332

Chapter 26. Queues

CHAPTER

TWENTYSEVEN

TREES
Like linked lists, trees are made up of nodes. A common kind of tree is a binary tree, in which each node contains a reference to two other nodes (possibly None). These references are referred to as the left and right subtrees. Like list nodes, tree nodes also contain cargo. A state diagram for a tree looks like this: To avoid cluttering up the picture, we often omit the Nones. The top of the tree (the node tree refers to) is called the root. In keeping with the tree metaphor, the other nodes are called branches and the nodes at the tips with null references are called leaves. It may seem odd that we draw the picture with the root at the top and the leaves at the bottom, but that is not the strangest thing. To make things worse, computer scientists mix in another metaphor the family tree. The top node is sometimes called a parent and the nodes it refers to are its children. Nodes with the same parent are called siblings. Finally, there is a geometric vocabulary for talking about trees. We already mentioned left and right, but there is also up (toward the parent/root) and down (toward the children/leaves). Also, all of the nodes that are the same distance from the root comprise a level of the tree. We probably dont need three metaphors for talking about trees, but there they are. Like linked lists, trees are recursive data structures because they are dened recursively. A tree is either: 1. the empty tree, represented by None, or 2. a node that contains an object reference (cargo) and two tree references.

27.1 Building trees


The process of assembling a tree is similar to the process of assembling a linked list. Each constructor invocation builds a single node.
1 2 3 4

class Tree: def __init__(self, cargo, left=None, right=None): self.cargo = cargo self.left = left

333

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

5 6 7 8

self.right = right def __str__(self): return str(self.cargo)

The cargo can be any type, but the left and right parameters should be tree nodes. left and right are optional; the default value is None. To print a node, we just print the cargo. One way to build a tree is from the bottom up. Allocate the child nodes rst:
1 2

left = Tree(2) right = Tree(3)

Then create the parent node and link it to the children:


1

tree = Tree(1, left, right)

We can write this code more concisely by nesting constructor invocations:


>>> tree = Tree(1, Tree(2), Tree(3))

Either way, the result is the tree at the beginning of the chapter.

27.2 Traversing trees


Any time you see a new data structure, your rst question should be, How do I traverse it? The most natural way to traverse a tree is recursively. For example, if the tree contains integers as cargo, this function returns their sum:
1 2 3

def total(tree): if tree == None: return 0 return total(tree.left) + total(tree.right) + tree.cargo

The base case is the empty tree, which contains no cargo, so the sum is 0. The recursive step makes two recursive calls to nd the sum of the child trees. When the recursive calls complete, we add the cargo of the parent and return the total.

27.3 Expression trees


A tree is a natural way to represent the structure of an expression. Unlike other notations, it can represent the computation unambiguously. For example, the inx expression 1 + 2 * 3 is ambiguous unless we know that the multiplication happens before the addition. This expression tree represents the same computation: The nodes of an expression tree can be operands like 1 and 2 or operators like + and *. Operands are leaf nodes; operator nodes contain references to their operands. (All of these operators are binary, meaning they have exactly two operands.) 334 Chapter 27. Trees

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

We can build this tree like this:


>>> tree = Tree(+, Tree(1), Tree(*, Tree(2), Tree(3)))

Looking at the gure, there is no question what the order of operations is; the multiplication happens rst in order to compute the second operand of the addition. Expression trees have many uses. The example in this chapter uses trees to translate expressions to postx, prex, and inx. Similar trees are used inside compilers to parse, optimize, and translate programs.

27.4 Tree traversal


We can traverse an expression tree and print the contents like this:
1 2 3 4 5

def print_tree(tree): if tree == None: return print(tree.cargo, end= ) print_tree(tree.left) print_tree(tree.right)

In other words, to print a tree, rst print the contents of the root, then print the entire left subtree, and then print the entire right subtree. This way of traversing a tree is called a preorder, because the contents of the root appear before the contents of the children. For the previous example, the output is:
>>> tree = Tree(+, Tree(1), Tree(*, Tree(2), Tree(3))) >>> print_tree(tree) + 1 * 2 3

This format is different from both postx and inx; it is another notation called prex, in which the operators appear before their operands. You might suspect that if you traverse the tree in a different order, you will get the expression in a different notation. For example, if you print the subtrees rst and then the root node, you get:
1 2 3 4 5

def print_tree_postorder(tree): if tree == None: return print_tree_postorder(tree.left) print_tree_postorder(tree.right) print(tree.cargo, end= )

The result, 1 2 3 * +, is in postx! This order of traversal is called postorder. Finally, to traverse a tree inorder, you print the left tree, then the root, and then the right tree:
1 2 3 4 5

def print_tree_inorder(tree): if tree == None: return print_tree_inorder(tree.left) print(tree.cargo, end= ) print_tree_inorder(tree.right)

27.4. Tree traversal

335

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

The result is 1 + 2 * 3, which is the expression in inx. To be fair, we should point out that we have omitted an important complication. Sometimes when we write an expression in inx, we have to use parentheses to preserve the order of operations. So an inorder traversal is not quite sufcient to generate an inx expression. Nevertheless, with a few improvements, the expression tree and the three recursive traversals provide a general way to translate expressions from one format to another. If we do an inorder traversal and keep track of what level in the tree we are on, we can generate a graphical representation of a tree:
1 2 3 4 5

def print_tree_indented(tree, level=0): if tree == None: return print_tree_indented(tree.right, level+1) print( * level + str(tree.cargo)) print_tree_indented(tree.left, level+1)

The parameter level keeps track of where we are in the tree. By default, it is initially 0. Each time we make a recursive call, we pass level+1 because the childs level is always one greater than the parents. Each item is indented by two spaces per level. The result for the example tree is:
>>> print_tree_indented(tree) 3 * 2 + 1

If you look at the output sideways, you see a simplied version of the original gure.

27.5 Building an expression tree


In this section, we parse inx expressions and build the corresponding expression trees. For example, the expression (3 + 7) * 9 yields the following tree: Notice that we have simplied the diagram by leaving out the names of the attributes. The parser we will write handles expressions that include numbers, parentheses, and the operators + and *. We assume that the input string has already been tokenized into a Python list (producing this list is left as an exercise). The token list for (3 + 7) * 9 is:
[(, 3, +, 7, ), *, 9, end]

The end token is useful for preventing the parser from reading past the end of the list. The rst function well write is get_token, which takes a token list and an expected token as parameters. It compares the expected token to the rst token on the list: if they match, it removes the token from the list and returns True; otherwise, it returns False:

336

Chapter 27. Trees

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2 3 4 5 6

def get_token(token_list, expected): if token_list[0] == expected: del token_list[0] return True else: return False

Since token_list refers to a mutable object, the changes made here are visible to any other variable that refers to the same object. The next function, get_number, handles operands. If the next token in token_list is a number, get_number removes it and returns a leaf node containing the number; otherwise, it returns None.
1 2 3 4 5

def get_number(token_list): x = token_list[0] if type(x) != type(0): return None del token_list[0] return Tree (x, None, None)

Before continuing, we should test get_number in isolation. We assign a list of numbers to token_list, extract the rst, print the result, and print what remains of the token list:
>>> token_list = [9, 11, end] >>> x = get_number(token_list) >>> print_tree_postorder(x) 9 >>> print(token_list) [11, end]

The next method we need is get_product, which builds an expression tree for products. A simple product has two numbers as operands, like 3 * 7. Here is a version of get_product that handles simple products.
1 2 3 4 5 6 7

def get_product(token_list): a = get_number(token_list) if get_token(token_list, *): b = get_number(token_list) return Tree (*, a, b) else: return a

Assuming that get_number succeeds and returns a singleton tree, we assign the rst operand to a. If the next character is *, we get the second number and build an expression tree with a, b, and the operator. If the next character is anything else, then we just return the leaf node with a. Here are two examples:
>>> token_list = [9, *, 11, end] >>> tree = get_product(token_list) >>> print_tree_postorder(tree)

27.5. Building an expression tree

337

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

9 11 * >>> token_list = [9, +, 11, end] >>> tree = get_product(token_list) >>> print_tree_postorder(tree) 9

The second example implies that we consider a single operand to be a kind of product. This denition of product is counterintuitive, but it turns out to be useful. Now we have to deal with compound products, like like 3 * 5 * 13. We treat this expression as a product of products, namely 3 * (5 * 13). The resulting tree is: With a small change in get_product, we can handle an arbitrarily long product:
1 2 3 4 5 6 7

def get_product(token_list): a = get_number(token_list) if get_token(token_list, *): b = get_product(token_list) return Tree (*, a, b) else: return a

# this line changed

In other words, a product can be either a singleton or a tree with * at the root, a number on the left, and a product on the right. This kind of recursive denition should be starting to feel familiar. Lets test the new version with a compound product:
>>> >>> >>> 2 3 token_list = [2, *, 3, *, 5 , *, 7, end] tree = get_product(token_list) print_tree_postorder(tree) 5 7 * * *

Next we will add the ability to parse sums. Again, we use a slightly counterintuitive denition of sum. For us, a sum can be a tree with + at the root, a product on the left, and a sum on the right. Or, a sum can be just a product. If you are willing to play along with this denition, it has a nice property: we can represent any expression (without parentheses) as a sum of products. This property is the basis of our parsing algorithm. get_sum tries to build a tree with a product on the left and a sum on the right. But if it doesnt nd a +, it just builds a product.
1 2 3 4 5 6 7

def get_sum(token_list): a = get_product(token_list) if get_token(token_list, +): b = get_sum(token_list) return Tree (+, a, b) else: return a

Lets test it with 9 * 11 + 5 * 7: 338 Chapter 27. Trees

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

>>> token_list = [9, *, 11, +, 5, *, 7, end] >>> tree = get_sum(token_list) >>> print_tree_postorder(tree) 9 11 * 5 7 * +

We are almost done, but we still have to handle parentheses. Anywhere in an expression where there can be a number, there can also be an entire sum enclosed in parentheses. We just need to modify get_number to handle subexpressions:
1 2 3 4 5 6 7 8 9 10

def get_number(token_list): if get_token(token_list, (): x = get_sum(token_list) # get the subexpression get_token(token_list, )) # remove the closing parenthesis return x else: x = token_list[0] if type(x) != type(0): return None token_list[0:1] = [] return Tree (x, None, None)

Lets test this code with 9 * (11 + 5) * 7:


>>> token_list = [9, *, (, 11, +, 5, ), *, 7, end] >>> tree = get_sum(token_list) >>> print_tree_postorder(tree) 9 11 5 + 7 * *

The parser handled the parentheses correctly; the addition happens before the multiplication. In the nal version of the program, it would be a good idea to give get_number a name more descriptive of its new role.

27.6 Handling errors


Throughout the parser, weve been assuming that expressions are well-formed. For example, when we reach the end of a subexpression, we assume that the next character is a close parenthesis. If there is an error and the next character is something else, we should deal with it.
1 2 3 4 5 6 7 8

def get_number(token_list): if get_token(token_list, (): x = get_sum(token_list) if not get_token(token_list, )): raise BadExpressionError, missing parenthesis return x else: # the rest of the function omitted

The raise statement creates an exception; in this case we create a new kind of exception, called a BadExpressionError. If the function that called get_number, or one of the other functions in the traceback, handles the exception, then the program can continue. Otherwise, Python will print an error message and quit. 27.6. Handling errors 339

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

27.7 The animal tree


In this section, we develop a small program that uses a tree to represent a knowledge base. The program interacts with the user to create a tree of questions and animal names. Here is a sample run:
Are you thinking of an animal? y Is it a bird? n What is the animals name? dog What question would distinguish a dog from a bird? Can it fly If the animal were dog the answer would be? n Are you thinking of an animal? y Can it fly? n Is it a dog? n What is the animals name? cat What question would distinguish a cat from a dog? Does it bark If the animal were cat the answer would be? n Are you thinking of an animal? y Can it fly? n Does it bark? y Is it a dog? y I rule! Are you thinking of an animal? n

Here is the tree this dialog builds: At the beginning of each round, the program starts at the top of the tree and asks the rst question. Depending on the answer, it moves to the left or right child and continues until it gets to a leaf node. At that point, it makes a guess. If the guess is not correct, it asks the user for the name of the new animal and a question that distinguishes the (bad) guess from the new animal. Then it adds a node to the tree with the new question and the new animal. Here is the code:
1 2 3 4 5 6 7 8 9 10 11 12 13 14

def yes(ques): ans = input(ques).lower() return ans[0] == y def animal(): # start with a singleton root = Tree("bird") # loop until the user quits while True: print if not yes("Are you thinking of an animal? "): break # walk the tree

340

Chapter 27. Trees

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

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

tree = root while tree.left != None: prompt = tree.cargo + "? " if yes(prompt): tree = tree.right else: tree = tree.left # make a guess guess = tree.cargo prompt = "Is it a " + guess + "? " if yes(prompt): print("I rule!") continue # get new information prompt = "What is the animals name? " animal = input(prompt) prompt = "What question would distinguish a {0} from a {1}? " question = input(prompt.format(animal, guess)) # add new information to the tree tree.cargo = question prompt = "If the animal were {0} the answer would be? " if yes(prompt.format(animal)): tree.left = Tree(guess) tree.right = Tree(animal) else: tree.left = Tree(animal) tree.right = Tree(guess)

The function yes is a helper; it prints a prompt and then takes input from the user. If the response begins with y or Y, the function returns True. The condition of the outer loop of animal is True, which means it will continue until the break statement executes, if the user is not thinking of an animal. The inner while loop walks the tree from top to bottom, guided by the users responses. When a new node is added to the tree, the new question replaces the cargo, and the two children are the new animal and the original cargo. One shortcoming of the program is that when it exits, it forgets everything you carefully taught it! Fixing this problem is left as an exercise.

27.8 Glossary
binary operator An operator that takes two operands. binary tree A tree in which each node refers to zero, one, or two dependent nodes.

27.8. Glossary

341

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

child One of the nodes referred to by a node. leaf A bottom-most node in a tree, with no children. level The set of nodes equidistant from the root. parent The node that refers to a given node. postorder A way to traverse a tree, visiting the children of each node before the node itself. prex notation A way of writing a mathematical expression with each operator appearing before its operands. preorder A way to traverse a tree, visiting each node before its children. root The topmost node in a tree, with no parent. siblings Nodes that share a common parent. subexpression An expression in parentheses that acts as a single operand in a larger expression.

27.9 Exercises
1. Modify print_tree_inorder so that it puts parentheses around every operator and pair of operands. Is the output correct and unambiguous? Are the parentheses always necessary? 2. Write a function that takes an expression string and returns a token list. 3. Find other places in the expression tree functions where errors can occur and add appropriate raise statements. Test your code with improperly formed expressions. 4. Think of various ways you might save the animal knowledge tree in a le. Implement the one you think is easiest.

342

Chapter 27. Trees

APPENDIX

DEBUGGING
Different kinds of errors can occur in a program, and it is useful to distinguish among them in order to track them down more quickly: 1. Syntax errors are produced by Python when it is translating the source code into byte code. They usually indicate that there is something wrong with the syntax of the program. Example: Omitting the colon at the end of a def statement yields the somewhat redundant message SyntaxError: invalid syntax. 2. Runtime errors are produced by the runtime system if something goes wrong while the program is running. Most runtime error messages include information about where the error occurred and what functions were executing. Example: An innite recursion eventually causes a runtime error of maximum recursion depth exceeded. 3. Semantic errors are problems with a program that compiles and runs but doesnt do the right thing. Example: An expression may not be evaluated in the order you expect, yielding an unexpected result. The rst step in debugging is to gure out which kind of error you are dealing with. Although the following sections are organized by error type, some techniques are applicable in more than one situation.

A.1 Syntax errors


Syntax errors are usually easy to x once you gure out what they are. Unfortunately, the error messages are often not helpful. The most common messages are SyntaxError: invalid syntax and SyntaxError: invalid token, neither of which is very informative. On the other hand, the message does tell you where in the program the problem occurred. Actually, it tells you where Python noticed a problem, which is not necessarily where the error is. Sometimes the error is prior to the location of the error message, often on the preceding line. If you are building the program incrementally, you should have a good idea about where the error is. It will be in the last line you added. If you are copying code from a book, start by comparing your code to the books code very carefully. Check every character. At the same time, remember that the book might be wrong, 343

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

so if you see something that looks like a syntax error, it might be. Here are some ways to avoid the most common syntax errors: 1. Make sure you are not using a Python keyword for a variable name. 2. Check that you have a colon at the end of the header of every compound statement, including for, while, if, and def statements. 3. Check that indentation is consistent. You may indent with either spaces or tabs but its best not to mix them. Each level should be nested the same amount. 4. Make sure that any strings in the code have matching quotation marks. 5. If you have multiline strings with triple quotes (single or double), make sure you have terminated the string properly. An unterminated string may cause an invalid token error at the end of your program, or it may treat the following part of the program as a string until it comes to the next string. In the second case, it might not produce an error message at all! 6. An unclosed bracket (, {, or [ makes Python continue with the next line as part of the current statement. Generally, an error occurs almost immediately in the next line. 7. Check for the classic = instead of == inside a conditional. If nothing works, move on to the next section...

A.2 I cant get my program to run no matter what I do.


If the compiler says there is an error and you dont see it, that might be because you and the compiler are not looking at the same code. Check your programming environment to make sure that the program you are editing is the one Python is trying to run. If you are not sure, try putting an obvious and deliberate syntax error at the beginning of the program. Now run (or import) it again. If the compiler doesnt nd the new error, there is probably something wrong with the way your environment is set up. If this happens, one approach is to start again with a new program like Hello, World!, and make sure you can get a known program to run. Then gradually add the pieces of the new program to the working one.

A.3 Runtime errors


Once your program is syntactically correct, Python can import it and at least start running it. What could possibly go wrong?

344

Appendix A. Debugging

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

A.4 My program does absolutely nothing.


This problem is most common when your le consists of functions and classes but does not actually invoke anything to start execution. This may be intentional if you only plan to import this module to supply classes and functions. If it is not intentional, make sure that you are invoking a function to start execution, or execute one from the interactive prompt. Also see the Flow of Execution section below.

A.5 My program hangs.


If a program stops and seems to be doing nothing, we say it is hanging. Often that means that it is caught in an innite loop or an innite recursion. 1. If there is a particular loop that you suspect is the problem, add a print statement immediately before the loop that says entering the loop and another immediately after that says exiting the loop. 2. Run the program. If you get the rst message and not the second, youve got an innite loop. Go to the Innite Loop section below. 3. Most of the time, an innite recursion will cause the program to run for a while and then produce a RuntimeError: Maximum recursion depth exceeded error. If that happens, go to the Innite Recursion section below. 4. If you are not getting this error but you suspect there is a problem with a recursive method or function, you can still use the techniques in the Innite Recursion section. 5. If neither of those steps works, start testing other loops and other recursive functions and methods. 6. If that doesnt work, then it is possible that you dont understand the ow of execution in your program. Go to the Flow of Execution section below.

A.6 Innite Loop


If you think you have an innite loop and you think you know what loop is causing the problem, add a print statement at the end of the loop that prints the values of the variables in the condition and the value of the condition. For example:
1 2 3 4 5 6 7

while x > 0 and y < 0: # do something to x # do something to y print("x: ", x) print("y: ", y) print("condition: ", (x > 0 and y < 0))

A.4. My program does absolutely nothing.

345

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Now when you run the program, you will see three lines of output for each time through the loop. The last time through the loop, the condition should be false. If the loop keeps going, you will be able to see the values of x and y, and you might gure out why they are not being updated correctly. In a development environment like PyScripter, one can also set a breakpoint at the start of the loop, and single-step through the loop. While you do this, inspect the values of x and y by hovering your cursor over them. Of course, all programming and debugging require that you have a good mental model of what the algorithm ought to be doing: if you dont understand what ought to happen to x and y, printing or inspecting its value is of little use. Probably the best place to debug the code is away from your computer, working on your understanding of what should be happening.

A.7 Innite Recursion


Most of the time, an innite recursion will cause the program to run for a while and then produce a Maximum recursion depth exceeded error. If you suspect that a function or method is causing an innite recursion, start by checking to make sure that there is a base case. In other words, there should be some condition that will cause the function or method to return without making a recursive invocation. If not, then you need to rethink the algorithm and identify a base case. If there is a base case but the program doesnt seem to be reaching it, add a print statement at the beginning of the function or method that prints the parameters. Now when you run the program, you will see a few lines of output every time the function or method is invoked, and you will see the parameters. If the parameters are not moving toward the base case, you will get some ideas about why not. Once again, if you have an environment that supports easy single-stepping, breakpoints, and inspection, learn to use them well. It is our opinion that walking through code step-by-step builds the best and most accurate mental model of how computation happens. Use it if you have it!

A.8 Flow of Execution


If you are not sure how the ow of execution is moving through your program, add print statements to the beginning of each function with a message like entering function foo, where foo is the name of the function. Now when you run the program, it will print a trace of each function as it is invoked. If youre not sure, step through the program with your debugger.

346

Appendix A. Debugging

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

A.9 When I run the program I get an exception.


If something goes wrong during runtime, Python prints a message that includes the name of the exception, the line of the program where the problem occurred, and a traceback. Put a breakpoint on the line causing the exception, and look around! The traceback identies the function that is currently running, and then the function that invoked it, and then the function that invoked that, and so on. In other words, it traces the path of function invocations that got you to where you are. It also includes the line number in your le where each of these calls occurs. The rst step is to examine the place in the program where the error occurred and see if you can gure out what happened. These are some of the most common runtime errors: NameError You are trying to use a variable that doesnt exist in the current environment. Remember that local variables are local. You cannot refer to them from outside the function where they are dened. TypeError There are several possible causes: 1. You are trying to use a value improperly. Example: indexing a string, list, or tuple with something other than an integer. 2. There is a mismatch between the items in a format string and the items passed for conversion. This can happen if either the number of items does not match or an invalid conversion is called for. 3. You are passing the wrong number of arguments to a function or method. For methods, look at the method denition and check that the rst parameter is self. Then look at the method invocation; make sure you are invoking the method on an object with the right type and providing the other arguments correctly. KeyError You are trying to access an element of a dictionary using a key value that the dictionary does not contain. AttributeError You are trying to access an attribute or method that does not exist. IndexError The index you are using to access a list, string, or tuple is greater than its length minus one. Immediately before the site of the error, add a print statement to display the value of the index and the length of the array. Is the array the right size? Is the index the right value?

A.10 I added so many print statements I get inundated with output.


One of the problems with using print statements for debugging is that you can end up buried in output. There are two ways to proceed: simplify the output or simplify the program. To simplify the output, you can remove or comment out print statements that arent helping, or combine them, or format the output so it is easier to understand. A.9. When I run the program I get an exception. 347

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

To simplify the program, there are several things you can do. First, scale down the problem the program is working on. For example, if you are sorting an array, sort a small array. If the program takes input from the user, give it the simplest input that causes the problem. Second, clean up the program. Remove dead code and reorganize the program to make it as easy to read as possible. For example, if you suspect that the problem is in a deeply nested part of the program, try rewriting that part with simpler structure. If you suspect a large function, try splitting it into smaller functions and testing them separately. Often the process of nding the minimal test case leads you to the bug. If you nd that a program works in one situation but not in another, that gives you a clue about what is going on. Similarly, rewriting a piece of code can help you nd subtle bugs. If you make a change that you think doesnt affect the program, and it does, that can tip you off. You can also wrap your debugging print statements in some condition, so that you suppress much of the output. For example, if you are trying to nd an element using a binary search, and it is not working, you might code up a debugging print statement inside a conditional: if the range of candidate elements is less that 6, then print debugging information, otherwise dont print. Similarly, breakpoints can be made conditional: you can set a breakpoint on a statement, then edit the breakpoint to say only break if this expression becomes true.

A.11 Semantic errors


In some ways, semantic errors are the hardest to debug, because the compiler and the runtime system provide no information about what is wrong. Only you know what the program is supposed to do, and only you know that it isnt doing it. The rst step is to make a connection between the program text and the behavior you are seeing. You need a hypothesis about what the program is actually doing. One of the things that makes that hard is that computers run so fast. You will often wish that you could slow the program down to human speed, and with some debuggers you can. But the time it takes to insert a few well-placed print statements is often short compared to setting up the debugger, inserting and removing breakpoints, and walking the program to where the error is occurring.

A.12 My program doesnt work.


You should ask yourself these questions: 1. Is there something the program was supposed to do but which doesnt seem to be happening? Find the section of the code that performs that function and make sure it is executing when you think it should. 2. Is something happening that shouldnt? Find code in your program that performs that function and see if it is executing when it shouldnt. 348 Appendix A. Debugging

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

3. Is a section of code producing an effect that is not what you expected? Make sure that you understand the code in question, especially if it involves invocations to functions or methods in other Python modules. Read the documentation for the functions you invoke. Try them out by writing simple test cases and checking the results. In order to program, you need to have a mental model of how programs work. If you write a program that doesnt do what you expect, very often the problem is not in the program; its in your mental model. The best way to correct your mental model is to break the program into its components (usually the functions and methods) and test each component independently. Once you nd the discrepancy between your model and reality, you can solve the problem. Of course, you should be building and testing components as you develop the program. If you encounter a problem, there should be only a small amount of new code that is not known to be correct.

A.13 Ive got a big hairy expression and it doesnt do what I expect.
Writing complex expressions is ne as long as they are readable, but they can be hard to debug. It is often a good idea to break a complex expression into a series of assignments to temporary variables. For example:
1

self.hands[i].addCard (self.hands[self.findNeighbor(i)].popCard())

This can be rewritten as:


1 2 3

neighbor = self.findNeighbor (i) pickedCard = self.hands[neighbor].popCard() self.hands[i].addCard (pickedCard)

The explicit version is easier to read because the variable names provide additional documentation, and it is easier to debug because you can check the types of the intermediate variables and display or inspect their values. Another problem that can occur with big expressions is that the order of evaluation may not be what you expect. For example, if you are translating the expression x/2pi into Python, you might write:
y = x / 2 * math.pi

That is not correct because multiplication and division have the same precedence and are evaluated from left to right. So this expression computes (x/2)pi. A good way to debug expressions is to add parentheses to make the order of evaluation explicit:
y = x / (2 * math.pi)

A.13. Ive got a big hairy expression and it doesnt do what I expect.

349

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Whenever you are not sure of the order of evaluation, use parentheses. Not only will the program be correct (in the sense of doing what you intended), it will also be more readable for other people who havent memorized the rules of precedence.

A.14 Ive got a function or method that doesnt return what I expect.
If you have a return statement with a complex expression, you dont have a chance to print the return value before returning. Again, you can use a temporary variable. For example, instead of:
return self.hands[i].removeMatches()

you could write:


count = self.hands[i].removeMatches() return count

Now you have the opportunity to display or inspect the value of count before returning.

A.15 Im really, really stuck and I need help.


First, try getting away from the computer for a few minutes. Computers emit waves that affect the brain, causing these effects: 1. Frustration and/or rage. 2. Superstitious beliefs ( the computer hates me ) and magical thinking ( the program only works when I wear my hat backward ). 3. Random-walk programming (the attempt to program by writing every possible program and choosing the one that does the right thing). If you nd yourself suffering from any of these symptoms, get up and go for a walk. When you are calm, think about the program. What is it doing? What are some possible causes of that behavior? When was the last time you had a working program, and what did you do next? Sometimes it just takes time to nd a bug. We often nd bugs when we are away from the computer and let our minds wander. Some of the best places to nd bugs are trains, showers, and in bed, just before you fall asleep.

A.16 No, I really need help.


It happens. Even the best programmers occasionally get stuck. Sometimes you work on a program so long that you cant see the error. A fresh pair of eyes is just the thing.

350

Appendix A. Debugging

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

Before you bring someone else in, make sure you have exhausted the techniques described here. Your program should be as simple as possible, and you should be working on the smallest input that causes the error. You should have print statements in the appropriate places (and the output they produce should be comprehensible). You should understand the problem well enough to describe it concisely. When you bring someone in to help, be sure to give them the information they need: 1. If there is an error message, what is it and what part of the program does it indicate? 2. What was the last thing you did before this error occurred? What were the last lines of code that you wrote, or what is the new test case that fails? 3. What have you tried so far, and what have you learned? Good instructors and helpers will also do something that should not offend you: they wont believe when you tell them Im sure all the input routines are working just ne, and that Ive set up the data correctly!. They will want to validate and check things for themselves. After all, your program has a bug. Your understanding and inspection of the code have not found it yet. So you should expect to have your assumptions challenged. And as you gain skills and help others, youll need to do the same for them. When you nd the bug, take a second to think about what you could have done to nd it faster. Next time you see something similar, you will be able to nd the bug more quickly. Remember, the goal is not just to make the program work. The goal is to learn how to make the program work.

A.16. No, I really need help.

351

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

352

Appendix A. Debugging

APPENDIX

AN ODDS-AND-ENDS WORKBOOK
This workbook / cookbook of recipes is still very much under construction.

B.1 The Five Strands of Prociency


This was an important study commissioned by the President in the USA. It looked at what was needed for students to become procient in maths. But it is also an amazingly accurate t for what we need for prociency in Computer Science, or even for prociency in playing Jazz!

1. Procedural Fluency: Learn the syntax. Learn to type. Learn your way around your tools. Learn and practice your scales. Learn to rearrange formulae. 2. Conceptual Understanding: Understand why the bits t together like they do.

353

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

3. Strategic Competence: Can you see what to do next? Can you formulate this word problem into your notation? Can you take the music where you want it to go? 4. Adaptive Reasoning: Can you see how to change what youve learnt for this new problem? 5. A Productive Disposition: We need that Can Do! attitude! (a) You habitually think it is worthwhile studying this stuff. (b) You are diligent and disciplined enough to grind through the tough stuff, and to put in your practice hours. (c) You develop a sense of efcacy that you can make things happen! Check out http://mason.gmu.edu/~jsuh4/teaching/strands.htm, or Kilpatricks book at http://www.nap.edu/openbook.php?isbn=0309069955

B.2 Sending Email


Sometimes it is fun to do powerful things with Python remember that part of the productive disposition we saw under the ve threads of prociency included efcacy the sense of being able to accomplish something useful. Here is a Python example of how you can send email to someone.
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

import smtplib, email.mime.text me = joe@my.org.com # put your own email here fred = fred@his.org.com # and freds email address here your_mail_server = mail.my.org.com # Ask your system administrator # Create a text message containing the body of the email. # You could read this from a file, of course. msg = email.mime.text.MIMEText("""Hey Fred, Im having a party, please come at 8pm. Bring a plate of snacks and your own drinks. Joe""" ) msg[From] = me # add headers to the message object msg[To] = fred msg[Subject] = Party on Saturday 23rd # create a connection to your mail server svr = smtplib.SMTP(your_mail_server) response = svr.sendmail(me, fred, msg.as_string()) # send message if response != {}: print("Sending failed for ", response) else: print("Message sent.")

354

Appendix B. An odds-and-ends Workbook

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

27 28

svr.quit()

# close the connection

In the context of the course, notice how we use the two objects in this program: we create a message object on line 9, and set some attributes at lines 16-18. We then create a connection object at line 21, and ask it to send our message.

B.2. Sending Email

355

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

356

Appendix B. An odds-and-ends Workbook

APPENDIX

CONFIGURING UBUNTU FOR PYTHON DEVELOPMENT


Note: the following instructions assume that you are connected to the Internet and that you have both the main and universe package repositories enabled. All unix shell commands are assumed to be running from your home directory ($HOME). Finally, any command that begins with sudo assums that you have administrative rights on your machine. If you do not please ask your system administrator about installing the software you need. What follows are instructions for setting up an Ubuntu 9.10 (Karmic) home environment for use with this book. I use Ubuntu GNU/Linux for both development and testing of the book, so it is the only system about which I can personally answer setup and conguration questions. In the spirit of software freedom and open collaboration, please contact me if you would like to maintain a similar appendix for your own favorite system. Id be more than happy to link to it or put it on the Open Book Project site, provided you agree to answer user feedback concerning it. Thanks!

Jeffrey Elkner Governors Career and Technical Academy in Arlington Arlington, Virginia

C.1 Vim
Vim can be used very effectively for Python development, but Ubuntu only comes with the vim-tiny package installed by default, so it doesnt support color syntax highlighting or autoindenting. To use Vim, do the following: 1. From the unix command prompt, run:

357

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

$ sudo apt-get install vim-gnome

2. Create a le in your home directory named .vimrc that contains the following:
syntax enable filetype indent on set et set sw=4 set smarttab map <f2> :w\|!python %

When you edit a le with a .py extension, you should now have color systax highlighting and auto indenting. Pressing the key should run your program, and bring you back to the editor when the program completes. To learn to use vim, run the following command at a unix command prompt:
$ vimtutor

C.2 GASP
Several of the case studies use GASP (Graphics API for Students for Python), which is the only additional library needed to use this book. To install GASP on Ubuntu 9.04 (Jaunty) or later, run the following command at a unix command prompt:
$ sudo apt-get install python-gasp

or use the synaptic package manager.

C.2.1 Getting GASP from Launchpad


To install the latest version of GASP into your home directory, run the following commands at a unix command prompt:
$ sudo apt-get install bzr $ bzr branch lp:gasp-code

C.3 $HOME environment


The following creates a useful environment in your home directory for adding your own Python libraries and executable scripts: 1. From the command prompt in your home directory, create bin and lib/python subdirectories by running the following commands:

358

Appendix C. Conguring Ubuntu for Python Development

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

$ mkdir bin lib $ mkdir lib/python

2. Add the following lines to the bottom of your .bashrc in your home directory:
PYTHONPATH=$HOME/lib/python EDITOR=vim export PYTHONPATH EDITOR

This will set your prefered editor to Vim, add your own lib/python subdirectory for your Python libraries to your Python path, and add your own bin directory as a place to put executable scripts. You need to logout and log back in before your local bin directory will be in your search path.

C.4 Making a python script executable and runnable from anywhere


On unix systems, Python scripts can be made executable using the following process: 1. Add this line as the rst line in the script:
#!/usr/bin/env python

2. At the unix command prompt, type the following to make myscript.py executable:
$ chmod +x myscript.py

3. Move myscript.py into your bin directory, and it will be runnable from anywhere.

C.4. Making a python script executable and runnable from anywhere

359

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

360

Appendix C. Conguring Ubuntu for Python Development

APPENDIX

CUSTOMIZING AND CONTRIBUTING TO THE BOOK


Note: the following instructions assume that you are connected to the Internet and that you have both the main and universe package repositories enabled. All unix shell commands are assumed to be running from your home directory ($HOME). Finally, any command that begins with sudo assums that you have administrative rights on your machine. If you do not please ask your system administrator about installing the software you need. This book is free as in freedom, which means you have the right to modify it to suite your needs, and to redistribute your modications so that our whole community can benet. That freedom lacks meaning, however, if you the tools needed to make a custom version or to contribute corrections and additions are not within your reach. This appendix attempts to put those tools in your hands. Thanks!

Jeffrey Elkner Governors Career and Technical Academy in Arlington Arlington, Virginia

D.1 Getting the Source


This book is marked up in ReStructuredText using a document generation system called Sphinx. The source code is located on the Launchpad http://bazaar.launchpad.net/~thinkcspy/thinkcspy/english2e/les. The easiest way to get the source code on an Ubuntu 9.10 computer is: 1. run sudo apt-get install bzr on your system to install bzr. 2. run bzr branch lp:thinkcspy. website at

361

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

The last command above will download the book source from Launchpad into a directory named thinkcspy which contains the Sphinx source and conguration information needed to build the book.

D.2 Making the HTML Version


To generate the html version of the book: 1. run sudo apt-get install python-sphinx to install the Sphinx documentation system. 2. cd thinkcspy - change into the thinkcspy directory containing the book source. 3. make html. The last command will run sphinx and create a directory named build containing the html verion of the text. Note: Sphinx supports building other output types as well, such as PDF. This requires that LaTeX be present on your system. Since I only personally use the html version, I will not attempt to document that process here.

362

Appendix D. Customizing and Contributing to the Book

APPENDIX

SOME TIPS, TRICKS, AND COMMON ERRORS


These are small summaries of ideas, tips, and commonly seen errors that might be helpful to those beginning Python.

E.1 Functions
Functions help us with our mental chunking: they allow us to group together statements for a high-level purpose, e.g. a function to sort a list of items, a function to make the turtle draw a spiral, or a function to compute the mean and standard deviation of some measurements. There are two kinds of functions: fruitful functions calculate and return a value, and we use them because were primarily interested in the value theyll return. Non-fruitful functions are used because they perform actions that we want done e.g. make a turtle draw a rectangle, or print the rst thousand prime numbers. They always return None a special dummy value. Tip: None is not a string Values like None, True and False are not strings: they are special values in Python, and are in the list of keywords we gave in chapter 2. Keywords are special in the language: they are part of the syntax. So we cannot create our own variable or function with a name True well get a syntax error. (Built-in functions are not priviledged like keywords: we can dene our own variable or function called len, but wed be silly to do so!) Along with the fruitful/non-fruitful division, there are two avours of the return statement in Python - one that returns a useful value, and the other that returns nothing, or None. And if we get to the end of any function and we have not explicitly executed any return statement, Python automatically returns the value None. Tip: Understand what the function needs to return Perhaps nothing some functions exists purely to perform actions rather than to calculate and return a result.

363

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

To make functions more useful, they are given parameters. So a function to make a turtle draw a square might have two parameters one for the turtle that needs to do the drawing, and another for the size of the square. See the rst example in Chapter 4 that function can be used with any turtle, and for any size square. So it is much more general than a function that always uses a specic turtle, say tess to draw a square of a specic size, say 30. Tip: Use parameters to generalize functions Understand which parts of the function will be hard-coded and unchangeable, and which parts should become parameters so that they can be customized by the caller of the function.

Tip: Try to relate Python functions to ideas we already know In math, were familiar with functions like f(x) = 3x + 5. We already understand that when we call the function f(3) we make some association between the parameter x and the argument 3. Try to draw parallels to argument passing in Python. Quiz: Is the function f(z) = 3z + 5 the same as function f above?

E.1.1 Problems with logic and ow of control


We often want to know if some condition holds for any item in a list, e.g. does the list have any odd numbers? This is a common mistake:
1 2 3 4 5 6 7

def anyOdd(xs): Return True if there is an odd number in xs, a list of integers. for v in xs: if v % 2 == 1: return True else: return False

Can we spot whats wrong here? As soon as we execute a return, well leave the function. So the logic of saying If I nd an odd number I can return True is ne. However, we cannot return False after only looking at one item we can only return False if weve been through all the items, and none of them are odd. So line 6 should not be there, and line 7 has to be outside the loop. Here is a corrected version:
1 2 3 4 5 6

def anyOdd(xs): Return True if there is an odd number in xs, a list of integers. for v in xs: if v % 2 == 1: return True return False

This eureka, or short-circuit style of returning from a function as soon as we are certain what the outcome will be was rst seen in Section 8.10, in the chapter on strings. It is preferred over this one, which also works correctly: 364 Appendix E. Some Tips, Tricks, and Common Errors

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

1 2 3 4 5 6 7 8 9 10

def anyOdd(xs): Return True if there is an odd number in xs, a list of integers. count = 0 for v in xs: if v % 2 == 1: count += 1 # count the odd numbers if count > 0: return True else: return False

The performance disadvantage of this one is that it traverses the whole list, even if it knows the outcome very early on. Tip: Think about the return conditions of the function Do I need to look at all elements in all cases? Can I shortcut and take an early exit? Under what conditions? When will I have to examine all the items in the list? The code in lines 7-10 can also be tightened up. The expression count > 0 evaluates to a boolean value, either True or False. The value can be used directly in the return statement. So we could cut out that code and simply have the following:
1 2 3 4 5 6 7 8

def anyOdd(xs): Return True if there is an odd number in xs, a list of integers. count = 0 for v in xs: if v % 2 == 1: count += 1 # count the odd numbers return count > 0 # Aha! a programmer who understands that boolean # expressions are not just used in if statements!

Although this code is tighter, it is not as nice as the one that did the short-circuit return as soon as the rst odd number was found. Tip: Generalize your use of booleans Mature programmers wont write if isPrime(n) == True: when they could say instead if isPrime(n): Think more generally about boolean values, not just in the context of if or while statements. Like arithmetic expressions, they have their own set of operators (and, or, not) and values (True, False) and can be assigned to variables, put into lists, etc. A good resource for improving your use of booleans is http://en.wikibooks.org/wiki/NonProgrammer%27s_Tutorial_for_Python_3/Boolean_Expressions Exercise time: How would we adapt this to make another function which returns True if all the numbers are odd? Can you still use a short-circuit style? How would we adapt it to return True if at least three of the numbers are odd? Short-

E.1. Functions

365

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

circuit the traversal when the third odd number is found dont traverse the whole list unless we have to.

E.1.2 Local variables


Functions are called, or activated, and while theyre busy they create their own stack frame which holds local variables. A local variable is one that belongs to the current activation. As soon as the function returns (whether from an explicit return statement or because Python reached the last statement), the stack frame and its local variables are all destroyed. The important consquence of this is that a function cannot use its own variables to remember any kind of state between different activations. It cannot count how many times it has been called, or remember to switch colours between red and blue UNLESS it makes use of variables that are global. Global variables will survive even after our function has exited, so they are the correct way maintain information between calls.
1 2 3 4 5 6 7

sz = 2 def h2(): Draw the next step of a spiral on each call. global sz tess.turn(42) tess.forward(sz) sz = sz + 1

This fragment assumes our turtle is tess. Each time we call h2() it turns, draws, and increases the global variable sz. Python always assumes that an assignment to a variable (as in line 7) means that we want a new local variable, unless weve provided a global declaration (on line 4). So leaving out the global declaration means this does not work. Tip: Local variables do not survive when you exit the function Use a Python visualizer like the one at http://netserv.ict.ru.ac.za/python3_viz to build a strong understanding of function calls, stack frames, local variables, and function returns.

Tip: Assigment in a function creates a local variable Any assignment to a variable within a function means Python will make a local variable, unless we override with global.

E.1.3 Event handler functions


Our chapter on event handling showed three different kinds of events that we could handle. They each have their own subtle points that can trip us up. Event handlers are non-fruitful functions they dont return any values. Theyre automatically called by the Python interpreter in response to an event, so we dont get to see the code that calls them. 366 Appendix E. Some Tips, Tricks, and Common Errors

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

A mouse-click event passes two coordinate arguments to its handler, so when we write this handler we have to provide for two parameters (usually named x and y). This is how the handler knows where the mouse click occurred. A keypress event handler has to be bound to the key it responds to. There is a messy extra step when using keypresses - we have to remember to issue a wn.listen() before our program will receive any keypresses. But if the user presses the key 10 times, the handler will be called ten times. Using a timer to create a future-dated event only causes one call to the handler. If we want repeated periodic handler activations, then from within the handler we call wn.ontimer(....) to set up the next event.

E.2 String handling


There are only four really important operations on strings, and well be able to do just about anything. There are many more nice-to-have methods (well call them sugar coating) that can make life easier, but if we can work with the basic four operations smoothly, well have a great grounding. len(str) nds the length of a string. str[i] the subscript operation extracts the ith character of the string, as a new string. str[i:j] the slice operation extracts a substring out of a string. str.nd(target) returns the index where target occurs within the string, or -1 if it is not found. So if we need to know if snake occurs as a substring within s, we could write
1 2

if s.find("snake") >= 0: if "snake" in s: ...

... # also works, nice-to-know sugar coating!

It would be wrong to split the string into words unless we were asked whether the word snake occurred in the string. Suppose were asked to read some lines of data and nd function denitions, e.g.: def someFunctionName(x, y):, and we are further asked to isolate and work with the name of the function. (Lets say, print it.)
1 2 3 4 5 6

s = "..." defPos = s.find("def ") if defPos == 0: opIndex = s.find(() fnname = s[4:opIndex] print(fnname)

# # # # # #

somehow get the next line to work with look for "def " in the line if it occurs at the left margin find the index of the open parenthesis slice out the function name and work with it.

One can extend these ideas: What if the function def was indented, and didnt start at column 0? The code would need a bit of adjustment, and wed probably want to be sure that all the characters in front of

E.2. String handling

367

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

the defpos were spaces. We would not want to do the wrong thing on data like this: # I def initely like Python! Weve assumed on line 3 that we will nd an open parenthesis. It may need to be checked that we did! We have also assumed that there was exactly one space between the keyword def and the start of the function name. It will not work nicely for def f(x) As weve already mentioned, there are many more sugar-coated methods that let us work more easily with strings. There is an rfind method, like find, that searches from the end of the string backwards. It is useful if we want to nd the last occurrence of something. The lower and upper methods can do case conversion. And the split method is great for breaking a string into a list of words, or into a list of lines. Weve also made extensive use in this book of the format method. In fact, if we want to practice reading the python documentation and learning some new methods on our own, the string methods are an excellent resource. Exercises: Suppose any line of text can contain at most one url that starts with http:// and ends at the next space in the line. Write a fragment of code to extract and print the full url if it is present. (Hint: read the documentation for find. It takes some extra arguments, so you can set a starting point from which it will search.) Suppose a string contains at most one substring < ... >. Write a fragment of code to extract and print the portion of the string between the angle brackets.

E.3 Looping and lists


Computers are useful because they can repeat computation, accurately and fast. So loops are going to be a central feature of almost all programs you encounter. Tip: Dont create unnecessary lists Lists are useful if you need to keep data for later computation. But if you dont need lists, it is probably better not to generate them. Here are two functions that both generate ten million random numbers, and return the sum of the numbers. They both work.
1 2 3 4 5 6 7 8 9 10

import random joe = random.Random() def sum1(): Build a list of random numbers, then sum them xs = [] for i in range(10000000): num = joe.randrange(1000) # generate one random number xs.append(num) # save it in our list

368

Appendix E. Some Tips, Tricks, and Common Errors

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

11 12 13 14 15 16 17 18 19 20 21 22 23

tot = sum(xs) return tot def sum2(): Sum the random numbers as we generate them tot = 0 for i in range(10000000): num = joe.randrange(1000) tot += num return tot print(sum1()) print(sum2())

What reasons are there for preferring the second version here? (Hint: open a tool like the Performance Monitor on your computer, and watch the memory usage. How big can you make the list before you get a fatal memory error in sum1?) In a similar way, when working with les, we often have an option to read the whole le contents into a single string, or we can read one line at a time and process each line as we read it. Line-at-a-time is the more traditional and perhaps safer way to do things youll be able to work comfortably no matter how large the le is. But you may nd whole-le-at-once is sometimes more convenient!

E.3. Looping and lists

369

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

370

Appendix E. Some Tips, Tricks, and Common Errors

APPENDIX

GNU FREE DOCUMENTATION LICENSE


Version 1.3, 3 November 2008 Copyright 2000, 2001, 2002, 2007, 2008 Free Software Foundation, Inc. Everyone is permitted to copy and distribute verbatim copies of this license document, but changing it is not allowed.

F.1 0. PREAMBLE
The purpose of this License is to make a manual, textbook, or other functional and useful document free in the sense of freedom: to assure everyone the effective freedom to copy and redistribute it, with or without modifying it, either commercially or noncommercially. Secondarily, this License preserves for the author and publisher a way to get credit for their work, while not being considered responsible for modications made by others. This License is a kind of copyleft, which means that derivative works of the document must themselves be free in the same sense. It complements the GNU General Public License, which is a copyleft license designed for free software. We have designed this License in order to use it for manuals for free software, because free software needs free documentation: a free program should come with manuals providing the same freedoms that the software does. But this License is not limited to software manuals; it can be used for any textual work, regardless of subject matter or whether it is published as a printed book. We recommend this License principally for works whose purpose is instruction or reference.

F.2 1. APPLICABILITY AND DEFINITIONS


This License applies to any manual or other work, in any medium, that contains a notice placed by the copyright holder saying it can be distributed under the terms of this License. Such a notice grants a world-wide, royalty-free license, unlimited in duration, to use that work under the conditions stated herein. The Document, below, refers to any such manual or work. Any 371

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

member of the public is a licensee, and is addressed as you. You accept the license if you copy, modify or distribute the work in a way requiring permission under copyright law. A Modied Version of the Document means any work containing the Document or a portion of it, either copied verbatim, or with modications and/or translated into another language. A Secondary Section is a named appendix or a front-matter section of the Document that deals exclusively with the relationship of the publishers or authors of the Document to the Documents overall subject (or to related matters) and contains nothing that could fall directly within that overall subject. (Thus, if the Document is in part a textbook of mathematics, a Secondary Section may not explain any mathematics.) The relationship could be a matter of historical connection with the subject or with related matters, or of legal, commercial, philosophical, ethical or political position regarding them. The Invariant Sections are certain Secondary Sections whose titles are designated, as being those of Invariant Sections, in the notice that says that the Document is released under this License. If a section does not t the above denition of Secondary then it is not allowed to be designated as Invariant. The Document may contain zero Invariant Sections. If the Document does not identify any Invariant Sections then there are none. The Cover Texts are certain short passages of text that are listed, as Front-Cover Texts or Back-Cover Texts, in the notice that says that the Document is released under this License. A Front-Cover Text may be at most 5 words, and a Back-Cover Text may be at most 25 words. A Transparent copy of the Document means a machine-readable copy, represented in a format whose specication is available to the general public, that is suitable for revising the document straightforwardly with generic text editors or (for images composed of pixels) generic paint programs or (for drawings) some widely available drawing editor, and that is suitable for input to text formatters or for automatic translation to a variety of formats suitable for input to text formatters. A copy made in an otherwise Transparent le format whose markup, or absence of markup, has been arranged to thwart or discourage subsequent modication by readers is not Transparent. An image format is not Transparent if used for any substantial amount of text. A copy that is not Transparent is called Opaque. Examples of suitable formats for Transparent copies include plain ASCII without markup, Texinfo input format, LaTeX input format, SGML or XML using a publicly available DTD, and standard-conforming simple HTML, PostScript or PDF designed for human modication. Examples of transparent image formats include PNG, XCF and JPG. Opaque formats include proprietary formats that can be read and edited only by proprietary word processors, SGML or XML for which the DTD and/or processing tools are not generally available, and the machinegenerated HTML, PostScript or PDF produced by some word processors for output purposes only. The Title Page means, for a printed book, the title page itself, plus such following pages as are needed to hold, legibly, the material this License requires to appear in the title page. For works in formats which do not have any title page as such, Title Page means the text near the most prominent appearance of the works title, preceding the beginning of the body of the text. The publisher means any person or entity that distributes copies of the Document to the public. A section Entitled XYZ means a named subunit of the Document whose title either is pre-

372

Appendix F. GNU Free Documentation License

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

cisely XYZ or contains XYZ in parentheses following text that translates XYZ in another language. (Here XYZ stands for a specic section name mentioned below, such as Acknowledgements, Dedications, Endorsements, or History.) To Preserve the Title of such a section when you modify the Document means that it remains a section Entitled XYZ according to this denition. The Document may include Warranty Disclaimers next to the notice which states that this License applies to the Document. These Warranty Disclaimers are considered to be included by reference in this License, but only as regards disclaiming warranties: any other implication that these Warranty Disclaimers may have is void and has no effect on the meaning of this License.

F.3 2. VERBATIM COPYING


You may copy and distribute the Document in any medium, either commercially or noncommercially, provided that this License, the copyright notices, and the license notice saying this License applies to the Document are reproduced in all copies, and that you add no other conditions whatsoever to those of this License. You may not use technical measures to obstruct or control the reading or further copying of the copies you make or distribute. However, you may accept compensation in exchange for copies. If you distribute a large enough number of copies you must also follow the conditions in section 3. You may also lend copies, under the same conditions stated above, and you may publicly display copies.

F.4 3. COPYING IN QUANTITY


If you publish printed copies (or copies in media that commonly have printed covers) of the Document, numbering more than 100, and the Documents license notice requires Cover Texts, you must enclose the copies in covers that carry, clearly and legibly, all these Cover Texts: Front-Cover Texts on the front cover, and Back-Cover Texts on the back cover. Both covers must also clearly and legibly identify you as the publisher of these copies. The front cover must present the full title with all words of the title equally prominent and visible. You may add other material on the covers in addition. Copying with changes limited to the covers, as long as they preserve the title of the Document and satisfy these conditions, can be treated as verbatim copying in other respects. If the required texts for either cover are too voluminous to t legibly, you should put the rst ones listed (as many as t reasonably) on the actual cover, and continue the rest onto adjacent pages. If you publish or distribute Opaque copies of the Document numbering more than 100, you must either include a machine-readable Transparent copy along with each Opaque copy, or state in or with each Opaque copy a computer-network location from which the general networkusing public has access to download using public-standard network protocols a complete Transparent copy of the Document, free of added material. If you use the latter option, you must take

F.3. 2. VERBATIM COPYING

373

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

reasonably prudent steps, when you begin distribution of Opaque copies in quantity, to ensure that this Transparent copy will remain thus accessible at the stated location until at least one year after the last time you distribute an Opaque copy (directly or through your agents or retailers) of that edition to the public. It is requested, but not required, that you contact the authors of the Document well before redistributing any large number of copies, to give them a chance to provide you with an updated version of the Document.

F.5 4. MODIFICATIONS
You may copy and distribute a Modied Version of the Document under the conditions of sections 2 and 3 above, provided that you release the Modied Version under precisely this License, with the Modied Version lling the role of the Document, thus licensing distribution and modication of the Modied Version to whoever possesses a copy of it. In addition, you must do these things in the Modied Version: A. Use in the Title Page (and on the covers, if any) a title distinct from that of the Document, and from those of previous versions (which should, if there were any, be listed in the History section of the Document). You may use the same title as a previous version if the original publisher of that version gives permission. B. List on the Title Page, as authors, one or more persons or entities responsible for authorship of the modications in the Modied Version, together with at least ve of the principal authors of the Document (all of its principal authors, if it has fewer than ve), unless they release you from this requirement. C. State on the Title page the name of the publisher of the Modied Version, as the publisher. D. Preserve all the copyright notices of the Document. E. Add an appropriate copyright notice for your modications adjacent to the other copyright notices. F. Include, immediately after the copyright notices, a license notice giving the public permission to use the Modied Version under the terms of this License, in the form shown in the Addendum below. G. Preserve in that license notice the full lists of Invariant Sections and required Cover Texts given in the Documents license notice. H. Include an unaltered copy of this License. I. Preserve the section Entitled History, Preserve its Title, and add to it an item stating at least the title, year, new authors, and publisher of the Modied Version as given on the Title Page. If there is no section Entitled History in the Document, create one stating the title, year, authors, and publisher of the Document as given on its Title Page, then add an item describing the Modied Version as stated in the previous sentence. J. Preserve the network location, if any, given in the Document for public access to a Transparent copy of the Document, and likewise the network locations given in the Doc374 Appendix F. GNU Free Documentation License

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

ument for previous versions it was based on. These may be placed in the History section. You may omit a network location for a work that was published at least four years before the Document itself, or if the original publisher of the version it refers to gives permission. K. For any section Entitled Acknowledgements or Dedications, Preserve the Title of the section, and preserve in the section all the substance and tone of each of the contributor acknowledgements and/or dedications given therein. L. Preserve all the Invariant Sections of the Document, unaltered in their text and in their titles. Section numbers or the equivalent are not considered part of the section titles. M. Delete any section Entitled Endorsements. Such a section may not be included in the Modied Version. N. Do not retitle any existing section to be Entitled Endorsements or to conict in title with any Invariant Section. O. Preserve any Warranty Disclaimers. If the Modied Version includes new front-matter sections or appendices that qualify as Secondary Sections and contain no material copied from the Document, you may at your option designate some or all of these sections as invariant. To do this, add their titles to the list of Invariant Sections in the Modied Versions license notice. These titles must be distinct from any other section titles. You may add a section Entitled Endorsements, provided it contains nothing but endorsements of your Modied Version by various partiesfor example, statements of peer review or that the text has been approved by an organization as the authoritative denition of a standard. You may add a passage of up to ve words as a Front-Cover Text, and a passage of up to 25 words as a Back-Cover Text, to the end of the list of Cover Texts in the Modied Version. Only one passage of Front-Cover Text and one of Back-Cover Text may be added by (or through arrangements made by) any one entity. If the Document already includes a cover text for the same cover, previously added by you or by arrangement made by the same entity you are acting on behalf of, you may not add another; but you may replace the old one, on explicit permission from the previous publisher that added the old one. The author(s) and publisher(s) of the Document do not by this License give permission to use their names for publicity for or to assert or imply endorsement of any Modied Version.

F.6 5. COMBINING DOCUMENTS


You may combine the Document with other documents released under this License, under the terms dened in section 4 above for modied versions, provided that you include in the combination all of the Invariant Sections of all of the original documents, unmodied, and list them all as Invariant Sections of your combined work in its license notice, and that you preserve all their Warranty Disclaimers. The combined work need only contain one copy of this License, and multiple identical Invariant Sections may be replaced with a single copy. If there are multiple Invariant Sections with the

F.6. 5. COMBINING DOCUMENTS

375

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

same name but different contents, make the title of each such section unique by adding at the end of it, in parentheses, the name of the original author or publisher of that section if known, or else a unique number. Make the same adjustment to the section titles in the list of Invariant Sections in the license notice of the combined work. In the combination, you must combine any sections Entitled History in the various original documents, forming one section Entitled History; likewise combine any sections Entitled Acknowledgements, and any sections Entitled Dedications. You must delete all sections Entitled Endorsements.

F.7 6. COLLECTIONS OF DOCUMENTS


You may make a collection consisting of the Document and other documents released under this License, and replace the individual copies of this License in the various documents with a single copy that is included in the collection, provided that you follow the rules of this License for verbatim copying of each of the documents in all other respects. You may extract a single document from such a collection, and distribute it individually under this License, provided you insert a copy of this License into the extracted document, and follow this License in all other respects regarding verbatim copying of that document.

F.8 7. AGGREGATION WITH INDEPENDENT WORKS


A compilation of the Document or its derivatives with other separate and independent documents or works, in or on a volume of a storage or distribution medium, is called an aggregate if the copyright resulting from the compilation is not used to limit the legal rights of the compilations users beyond what the individual works permit. When the Document is included in an aggregate, this License does not apply to the other works in the aggregate which are not themselves derivative works of the Document. If the Cover Text requirement of section 3 is applicable to these copies of the Document, then if the Document is less than one half of the entire aggregate, the Documents Cover Texts may be placed on covers that bracket the Document within the aggregate, or the electronic equivalent of covers if the Document is in electronic form. Otherwise they must appear on printed covers that bracket the whole aggregate.

F.9 8. TRANSLATION
Translation is considered a kind of modication, so you may distribute translations of the Document under the terms of section 4. Replacing Invariant Sections with translations requires special permission from their copyright holders, but you may include translations of some or all Invariant Sections in addition to the original versions of these Invariant Sections. You may include a translation of this License, and all the license notices in the Document, and any Warranty Disclaimers, provided that you also include the original English version of this License and the original versions of those notices and disclaimers. In case of a disagreement between 376 Appendix F. GNU Free Documentation License

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

the translation and the original version of this License or a notice or disclaimer, the original version will prevail. If a section in the Document is Entitled Acknowledgements, Dedications, or History, the requirement (section 4) to Preserve its Title (section 1) will typically require changing the actual title.

F.10 9. TERMINATION
You may not copy, modify, sublicense, or distribute the Document except as expressly provided under this License. Any attempt otherwise to copy, modify, sublicense, or distribute it is void, and will automatically terminate your rights under this License. However, if you cease all violation of this License, then your license from a particular copyright holder is reinstated (a) provisionally, unless and until the copyright holder explicitly and nally terminates your license, and (b) permanently, if the copyright holder fails to notify you of the violation by some reasonable means prior to 60 days after the cessation. Moreover, your license from a particular copyright holder is reinstated permanently if the copyright holder noties you of the violation by some reasonable means, this is the rst time you have received notice of violation of this License (for any work) from that copyright holder, and you cure the violation prior to 30 days after your receipt of the notice. Termination of your rights under this section does not terminate the licenses of parties who have received copies or rights from you under this License. If your rights have been terminated and not permanently reinstated, receipt of a copy of some or all of the same material does not give you any rights to use it.

F.11 10. FUTURE REVISIONS OF THIS LICENSE


The Free Software Foundation may publish new, revised versions of the GNU Free Documentation License from time to time. Such new versions will be similar in spirit to the present version, but may differ in detail to address new problems or concerns. See http://www.gnu.org/copyleft/. Each version of the License is given a distinguishing version number. If the Document species that a particular numbered version of this License or any later version applies to it, you have the option of following the terms and conditions either of that specied version or of any later version that has been published (not as a draft) by the Free Software Foundation. If the Document does not specify a version number of this License, you may choose any version ever published (not as a draft) by the Free Software Foundation. If the Document species that a proxy can decide which future versions of this License can be used, that proxys public statement of acceptance of a version permanently authorizes you to choose that version for the Document.

F.10. 9. TERMINATION

377

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

F.12 11. RELICENSING


Massive Multiauthor Collaboration Site (or MMC Site) means any World Wide Web server that publishes copyrightable works and also provides prominent facilities for anybody to edit those works. A public wiki that anybody can edit is an example of such a server. A Massive Multiauthor Collaboration (or MMC) contained in the site means any set of copyrightable works thus published on the MMC site. CC-BY-SA means the Creative Commons Attribution-Share Alike 3.0 license published by Creative Commons Corporation, a not-for-prot corporation with a principal place of business in San Francisco, California, as well as future copyleft versions of that license published by that same organization. Incorporate means to publish or republish a Document, in whole or in part, as part of another Document. An MMC is eligible for relicensing if it is licensed under this License, and if all works that were rst published under this License somewhere other than this MMC, and subsequently incorporated in whole or in part into the MMC, (1) had no cover texts or invariant sections, and (2) were thus incorporated prior to November 1, 2008. The operator of an MMC Site may republish an MMC contained in the site under CC-BY-SA on the same site at any time before August 1, 2009, provided the MMC is eligible for relicensing.

F.13 ADDENDUM: How to use this License for your documents


To use this License in a document you have written, include a copy of the License in the document and put the following copyright and license notices just after the title page:
Copyright (C) YEAR YOUR NAME. Permission is granted to copy, distribute and/or modify this document under the terms of the GNU Free Documentation License, Version 1.3 or any later version published by the Free Software Foundation; with no Invariant Sections, no Front-Cover Texts, and no Back-Cover Texts. A copy of the license is included in the section entitled "GNU Free Documentation License".

If you have Invariant Sections, Front-Cover Texts and Back-Cover Texts, replace the with . . . Texts. line with this:
with the Invariant Sections being LIST THEIR TITLES, with the Front-Cover Texts being LIST, and with the Back-Cover Texts being LIST.

If you have Invariant Sections without Cover Texts, or some other combination of the three, merge those two alternatives to suit the situation. If your document contains nontrivial examples of program code, we recommend releasing these examples in parallel under your choice of free software license, such as the GNU General Public License, to permit their use in free software. 378 Appendix F. GNU Free Documentation License

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

search

F.13. ADDENDUM: How to use this License for your documents

379

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

380

Appendix F. GNU Free Documentation License

INDEX

Symbols
3n + 1 sequence, 88

A
abbreviated assignment, 92 abecedarian series, 116 abstract data type (ADT), 323 accumulator, 297 algorithm, 2, 7, 107, 108 deterministic, 164 aliases, 149, 157, 270 alternative execution, 57 ambiguity, 5 animation rate, 246 argument, 44, 48 argv, 173 assignment, 13, 85 tuple, 132 assignment statement, 13, 22, 85 assignment token, 22 attribute, 25, 34, 171, 173, 211, 217

break statement, 98 breakpoint, 108 bug, 3, 7 builtin scope, 170 bump, 108

C
call graph, 273 canvas, 25, 34 cargo, 316 chained conditional, 58, 66 character, 113 chatterbox function, 79 child, 341 child class, 307 chunking, 31, 47 class, 11, 217 class attribute, 297 client, 323 clone, 149, 157 collection, 142, 186 command line, 173 command line argument, 173 command prompt, 173 comment, 7 comments, 6 comparison of strings, 118 comparison operator, 53, 66 compile, 1 composition, 22, 44, 75 composition (of functions), 79 composition of functions, 20 compound data type, 113, 128, 209 compound statement, 48, 55 body, 55 header, 55 computation pattern, 120 concatenate, 22 381

B
baked animation, 246 bar chart, 63 base case, 252, 256 binary operator, 341 binary search, 204 binary tree, 341 bind, 141 blit, 246 block, 55, 66 body, 48, 55, 66, 108 boolean algebra, 66 boolean expression, 53, 66 boolean function, 76, 79 boolean value, 53, 66 branch, 57, 66

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

concatenation, 19, 116 condition, 66, 87 conditional chained, 58 conditional branching, 55 conditional execution, 55 conditional statement, 55, 66 conditionals nested, 59 constant time, 330 constructor, 217 continue statement, 102, 173 control ow, 30, 34 copy, 224 deep, 224 shallow, 224 counter, 108 counting pattern, 120 cursor, 94, 108

elif, 55 else, 55 embedded reference, 316 encapsulate, 108 encapsulation, 96 encode, 297 enumerate, 150 equality, 222 deep, 222 shallow, 222 escape sequence, 94, 108 eureka traversal, 120 evaluate, 22 event, 134, 141 exception, 4, 7, 261, 265 handling, 261 expression, 16, 22 boolean, 53

D
data structure, 134, 252 recursive, 252 data type, 11, 22 dead code, 71, 79 debugging, 3, 7, 75 decrement, 108 deep copy, 225 deep equality, 222, 225 default value, 121, 128 denite iteration, 108 denition function, 25, 39 recursive, 252 del statement, 147, 268 delimiter, 157, 185, 323 deterministic algorithm, 164 development plan, 97, 108 dictionary, 267, 273 dir function, 122 directory, 183, 185 docstring, 49, 122, 128 dot notation, 122, 128 dot operator, 171, 173 dot product, 287 Doyle, Arthur Conan, 4

F
bonacci numbers, 254 eld width, 124 FIFO, 330 le, 179, 185 text, 182 le handle, 179 le system, 185 oat, 11, 17, 22 ow of execution, 30, 42, 49 for loop, 29, 34, 86, 116, 150 for loop traversal (for), 127 formal language, 5, 7 formatting strings, 124 fractal Cesaro torn square, 256 Sierpinski triangle, 257 frame, 49 frame rate, 246 fruitful function, 49, 79 fully qualied name, 185 function, 25, 39, 49, 104 argument, 44 composition, 44 len, 115 parameter, 44 pure, 277 function call, 49 Index

E
element, 142, 157, 186 382

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

function composition, 49, 75 function denition, 25, 39, 49 function type, 122 functional programming style, 287 fundamental ambiguity theorem, 316

G
game loop, 226, 246 generalization, 96, 281 generalize, 108 generic data structure, 323 global scope, 170

H
hand trace, 90 handle, 179, 185 handle an exception, 265 handler, 134, 141 handling an exception, 261 header line, 49 help, 93 helper, 316 high-level language, 1, 7 Holmes, Sherlock, 4

input dialog, 19 instance, 27, 34, 217 instantiate, 217 int, 11, 17, 22 integer, 11 integer division, 16, 22 Intel, 94 interface, 323 interpret, 1 interpreter, 7 invariant, 317 invoke, 25, 35 is operator, 148 item, 142, 157, 186 item assignment, 146 iteration, 85, 87, 108

J
join, 154 justication, 124

K
key, 267, 273 key-value pair, 267, 273 keyword, 14, 22

I
if, 55 if statement, 55 immediate mode, 7 immutable, 118, 128, 131, 146 immutable data type, 134, 265 implementation, 323 import statement, 44, 49, 168, 171, 173 in and not in operator (in, not in), 127 in operator, 119 increment, 108 incremental development, 72, 79 indenite iteration, 108 index, 113, 128, 143, 157 negative, 115 indexing ([]), 127 innite loop, 87, 108 innite recursion, 252, 256 inx, 323 inheritance, 307 initialization (of a variable), 108 initializer method, 217 input, 19 Index

L
leaf, 342 len function, 115 length function (len), 127 level, 342 lifetime, 47, 49 linear, 204 linear search, 204 linear time, 330 link, 316 linked list, 316 linked queue, 330 Linux, 4 list, 142, 157, 186 append, 152 nested, 143, 156 list index, 143 list traversal, 143, 157 literalness, 6 local scope, 170 local variable, 47, 49, 97 logarithm, 94 383

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

logical operator, 53, 54, 66 loop, 87, 109 loop body, 34, 87 loop variable, 34, 109 low-level language, 1, 7

M
Make Way for Ducklings, 116 mapping type, 267, 273 matrix, 156, 270 McCloskey, Robert, 116 memo, 271, 273 Merge algorithm, 204 meta-notation, 93, 109 method, 25, 34, 173, 217 middle-test loop, 109 mode, 185 modier, 153, 157, 278, 287 module, 25, 35, 122, 173 modulus operator, 21, 22 mutable, 118, 131, 146 mutable data type, 134, 157, 222

comparison, 53 in, 119 logical, 53, 54 modulus, 21 operator overloading, 287 optional parameter, 121, 128 order of operations, 18

P
parameter, 44, 49, 151 optional, 121 parent, 342 parent class, 307 parse, 5, 8, 323 pass statement, 55 path, 185 pattern, 158 pattern of computation, 120 Pentium, 94 pixel, 246 poetry, 6 poll, 226, 246 polymorphic, 287 portability, 8 portable, 1 post-test loop, 109 postx, 323 postorder, 342 pre-test loop, 109 precondition, 316 prex notation, 342 preorder, 342 print function, 8 Priority Queue, 330 priority queue, 330 probe, 204 problem solving, 8 program, 2, 6, 8 program development, 96 program tracing, 90 programming language, 1 promise, 155, 158, 163 prompt, 67 prose, 6 provider, 323 pure function, 158, 287 PyGame, 226 PyScripter, 1 Index

N
namespace, 168, 173 naming collision, 173 natural language, 5, 8 negative index, 115 nested conditionals, 59 nested list, 143, 156, 158 nested loop, 109 nesting, 66 newline, 94, 109 Newtons method, 106 node, 316 non-volatile memory, 185 None, 71, 79 normalized, 287

O
object, 25, 35, 158, 217 object code, 8 object-oriented language, 218 object-oriented programming, 209, 217 objects and values, 148 operand, 16, 22 operations on strings, 124 operator, 16, 22 384

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

single stepping, 43 Python shell, 8

Q
Queue, 330 queue, 330 queueing policy, 330

R
raise, 265 random numbers, 163 range, 35 range function, 31, 155 reassignment, 85, 109 rectangle, 221 recursion, 252, 256 innite, 252 recursive call, 252, 256 recursive data structure, 252 recursive denition, 252, 256 redundancy, 6 refactor, 49 refactoring code, 47 return statement, 60, 71 return value, 71, 80 root, 342 rules of precedence, 18, 22 runtime error, 4, 8, 115, 118

S
safe language, 4 scaffolding, 72, 80, 187 scalar multiplication, 287 scope, 170 builtin, 170 global, 170 local, 170 script, 8 semantic error, 4, 8 semantics, 4, 8 sequence, 142, 158, 186 shallow copy, 225 shallow equality, 222, 225 short-circuit evaluation, 120, 128 shufe, 163 siblings, 342 side effect, 153, 158 single-step, 109 Index

singleton, 316 slice, 117, 128, 146 slicing ([:]), 127 socket, 186 source code, 8 split, 154 sprite, 246 stack diagram, 49 state, 25 state snapshot, 13, 23 statement, 15, 23 assignment, 85 continue, 102 del, 147, 268 if, 55 import, 44, 168 pass, 55 return, 60 statement block, 55 statement: break, 98 step size, 158 str, 17, 23 string, 11 string comparison, 118 string comparison (>, <, >=, <=, ==, =), 127 string formatting, 124 string module, 122 string operations, 19 string slice, 117 strings and lists, 154 style, 77 subexpression, 342 sublist, 117, 146 subscript operator, 113 substring, 117 surface, 226, 246 syntax, 3, 8 syntax error, 3, 8

T
tab, 94, 109 table, 94 temporary variable, 71, 80 terminating condition, 35 test suite, 80 Test-driven development, 187 test-driven development (TDD), 204 385

How to Think Like a Computer Scientist: Learning with Python 3 Documentation, Release 3rd Edition

text le, 182, 185 token, 5, 8, 323 trace, 109 traceback, 49 tracing a program, 90 traversal, 116, 120 traverse, 128 trichotomy, 109 triple quoted string, 11 truncation, 17 truth table, 67 try ... except, 261 try ... except ... nally, 264 tuple, 131, 134 assignment, 132 return value, 132 tuple assignment, 134 turtle module, 25 two-dimensional table, 96 type conversion, 62 type conversion, 62, 67 type converter functions, 17

U
underscore character, 14 unit testing, 80 unit tests, 164 unreachable code, 71

V
value, 11, 23, 71, 267 boolean, 53 variable, 13, 23 local, 47, 97 temporary, 71 variable name, 23 veneer, 323 volatile memory, 186

W
while loop, 87 while statement, 87 whitespace, 128 wrapper, 316 wrapping code in a function, 57, 67

386

Index

You might also like