You can order print and ebook versions of Think Python 3e from Bookshop.org and Amazon.
7. Iteration and Search#
In 1939 Ernest Vincent Wright published a 50,000 word novel called Gadsby that does not contain the letter “e”. Since “e” is the most common letter in English, writing even a few words without using it is difficult. To get a sense of how difficult, in this chapter we’ll compute the fraction of English words have at least one “e”.
For that, we’ll use for
statements to loop through the letters in a string and the words in a file, and we’ll update variables in a loop to count the number of words that contain an “e”.
We’ll use the in
operator to check whether a letter appears in a word, and you’ll learn a programming pattern called a “linear search”.
As an exercise, you’ll use these tools to solve a word puzzle called “Spelling Bee”.
7.1. Loops and strings#
In Chapter 3 we saw a for
loop that uses the range
function to display a sequence of numbers.
for i in range(3):
print(i, end=' ')
0 1 2
This version uses the keyword argument end
so the print
function puts a space after each number rather than a newline.
We can also use a for
loop to display the letters in a string.
for letter in 'Gadsby':
print(letter, end=' ')
G a d s b y
Notice that I changed the name of the variable from i
to letter
, which provides more information about the value it refers to.
The variable defined in a for
loop is called the loop variable.
Now that we can loop through the letters in a word, we can check whether it contains the letter “e”.
for letter in "Gadsby":
if letter == 'E' or letter == 'e':
print('This word has an "e"')
Before we go on, let’s encapsulate that loop in a function.
def has_e():
for letter in "Gadsby":
if letter == 'E' or letter == 'e':
print('This word has an "e"')
And let’s make it a pure function that return True
if the word contains an “e” and False
otherwise.
def has_e():
for letter in "Gadsby":
if letter == 'E' or letter == 'e':
return True
return False
We can generalize it to take the word as a parameter.
def has_e(word):
for letter in word:
if letter == 'E' or letter == 'e':
return True
return False
Now we can test it like this:
has_e('Gadsby')
False
has_e('Emma')
True
7.2. Reading the word list#
To see how many words contain an “e”, we’ll need a word list. The one we’ll use is a list of about 114,000 official crosswords; that is, words that are considered valid in crossword puzzles and other word games.
The word list is in a file called words.txt
, which is downloaded in the notebook for this chapter.
To read it, we’ll use the built-in function open
, which takes the name of the file as a parameter and returns a file object we can use to read the file.
file_object = open('words.txt')
The file object provides a function called readline
, which reads characters from the file until it gets to a newline and returns the result as a string:
file_object.readline()
'aa\n'
Notice that the syntax for calling readline
is different from functions we’ve seen so far. That’s because it is a method, which is a function associated with an object.
In this case readline
is associated with the file object, so we call it using the name of the object, the dot operator, and the name of the method.
The first word in the list is “aa”, which is a kind of lava.
The sequence \n
represents the newline character that separates this word from the next.
The file object keeps track of where it is in the file, so if you call
readline
again, you get the next word:
line = file_object.readline()
line
'aah\n'
To remove the newline from the end of the word, we can use strip
, which is a method associated with strings, so we can call it like this.
word = line.strip()
word
'aah'
strip
removes whitespace characters – including spaces, tabs, and newlines – from the beginning and end of the string.
You can also use a file object as part of a for
loop.
This program reads words.txt
and prints each word, one per line:
for line in open('words.txt'):
word = line.strip()
print(word)
Now that we can read the word list, the next step is to count them. For that, we will need the ability to update variables.
7.3. Updating variables#
As you may have discovered, 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).
For example, here is an initial assignment that creates a variable.
x = 5
x
5
And here is an assignment that changes the value of a variable.
x = 7
x
7
The following figure shows what these assignments looks like in a state diagram.
The dotted arrow indicates that x
no longer refers to 5
.
The solid arrow indicates that it now refers to 7
.
A common kind of assignment is an update, where the new value of the variable depends on the old.
x = x + 1
x
8
This statement means “get the current value of x
, add one, and assign the result back to x
.”
If you try to update a variable that doesn’t exist, you get an error, because Python evaluates the expression on the right before it assigns a value to the variable on the left.
z = z + 1
NameError: name 'z' is not defined
Before you can update a variable, you have to initialize it, usually with a simple assignment:
z = 0
z = z + 1
z
1
Increasing the value of a variable is called an increment; decreasing the value is called a decrement.
Because these operations are so common, Python provides augmented assignment operators that update a variable more concisely.
For example, the +=
operator increments a variable by the given amount.
z += 2
z
3
There are augmented assignment operators for the other arithmetic operators, including -=
and *=
.
7.4. Looping and counting#
The following program counts the number of words in the word list.
total = 0
for line in open('words.txt'):
word = line.strip()
total += 1
It starts by initializing total
to 0
.
Each time through the loop, it increments total
by 1
.
So when the loop exits, total
refers to the total number of words.
total
113783
A variable like this, used to count the number of times something happens, is called a counter.
We can add a second counter to the program to keep track of the number of words that contain an “e”.
total = 0
count = 0
for line in open('words.txt'):
word = line.strip()
total = total + 1
if has_e(word):
count += 1
Let’s see how many words contain an “e”.
count
76162
As a percentage of total
, about two-thirds of the words use the letter “e”.
count / total * 100
66.93618554617122
So you can understand why it’s difficult to craft a book without using any such words.
7.5. The in operator#
The version of has_e
we wrote in this chapter is more complicated than it needs to be.
Python provides an operator, in
, that checks whether a character appears in a string.
word = 'Gadsby'
'e' in word
False
So we can rewrite has_e
like this.
def has_e(word):
if 'E' in word or 'e' in word:
return True
else:
return False
And because the conditional of the if
statement has a boolean value, we can eliminate the if
statement and return the boolean directly.
def has_e(word):
return 'E' in word or 'e' in word
We can simplify this function even more using the method lower
, which converts the letters in a string to lowercase.
Here’s an example.
word.lower()
'gadsby'
lower
makes a new string – it does not modify the existing string – so the value of word
is unchanged.
word
'Gadsby'
Here’s how we can use lower
in has_e
.
def has_e(word):
return 'e' in word.lower()
has_e('Gadsby')
False
has_e('Emma')
True
7.6. Search#
Based on this simpler version of has_e
, let’s write a more general function called uses_any
that takes a second parameter that is a string of letters.
If returns True
if the word uses any of the letters and False
otherwise.
def uses_any(word, letters):
for letter in word.lower():
if letter in letters.lower():
return True
return False
Here’s an example where the result is True
.
uses_any('banana', 'aeiou')
True
And another where it is False
.
uses_any('apple', 'xyz')
False
uses_any
converts word
and letters
to lowercase, so it works with any combination of cases.
uses_any('Banana', 'AEIOU')
True
The structure of uses_any
is similar to has_e
.
It loops through the letters in word
and checks them one at a time.
If it finds one that appears in letters
, it returns True
immediately.
If it gets all the way through the loop without finding any, it returns False
.
This pattern is called a linear search. In the exercises at the end of this chapter, you’ll write more functions that use this pattern.
7.7. Doctest#
In Chapter 4 we used a docstring to document a function – that is, to explain what it does.
It is also possible to use a docstring to test a function.
Here’s a version of uses_any
with a docstring that includes tests.
def uses_any(word, letters):
"""Checks if a word uses any of a list of letters.
>>> uses_any('banana', 'aeiou')
True
>>> uses_any('apple', 'xyz')
False
"""
for letter in word.lower():
if letter in letters.lower():
return True
return False
Each test begins with >>>
, which is used as a prompt in some Python environments to indicate where the user can type code.
In a doctest, the prompt is followed by an expression, usually a function call.
The following line indicates the value the expression should have if the function works correctly.
In the first example, 'banana'
uses 'a'
, so the result should be True
.
In the second example, 'apple'
does not use any of 'xyz'
, so the result should be False
.
To run these tests, we have to import the doctest
module and run a function called run_docstring_examples
.
To make this function easier to use, I wrote the following function, which takes a function object as an argument.
from doctest import run_docstring_examples
def run_doctests(func):
run_docstring_examples(func, globals(), name=func.__name__)
We haven’t learned about globals
and __name__
yet – you can ignore them.
Now we can test uses_any
like this.
run_doctests(uses_any)
run_doctests
finds the expressions in the docstring and evaluates them.
If the result is the expected value, the test passes.
Otherwise it fails.
If all tests pass, run_doctests
displays no output – in that case, no news is good news.
To see what happens when a test fails, here’s an incorrect version of uses_any
.
def uses_any_incorrect(word, letters):
"""Checks if a word uses any of a list of letters.
>>> uses_any_incorrect('banana', 'aeiou')
True
>>> uses_any_incorrect('apple', 'xyz')
False
"""
for letter in word.lower():
if letter in letters.lower():
return True
else:
return False # INCORRECT!
And here’s what happens when we test it.
run_doctests(uses_any_incorrect)
**********************************************************************
File "__main__", line 4, in uses_any_incorrect
Failed example:
uses_any_incorrect('banana', 'aeiou')
Expected:
True
Got:
False
The output includes the example that failed, the value the function was expected to produce, and the value the function actually produced.
If you are not sure why this test failed, you’ll have a chance to debug it as an exercise.
7.8. Glossary#
loop variable:
A variable defined in the header of a for
loop.
file object: An object that represents an open file and keeps track of which parts of the file have been read or written.
method: A function that is associated with an object and called using the dot operator.
update: An assignment statement that give a new value to a variable that already exists, rather than creating a new variables.
initialize: Create a new variable and give it a value.
increment: Increase the value of a variable.
decrement: Decrease the value of a variable.
counter: A variable used to count something, usually initialized to zero and then incremented.
linear search: A computational pattern that searches through a sequence of elements and stops what it finds what it is looking for.
pass: If a test runs and the result is as expected, the test passes.
fail: If a test runs and the result is not as expected, the test fails.
7.9. Exercises#
# This cell tells Jupyter to provide detailed debugging information
# when a runtime error occurs. Run it before working on the exercises.
%xmode Verbose
Exception reporting mode: Verbose
7.9.1. Ask a virtual assistant#
In uses_any
, you might have noticed that the first return
statement is inside the loop and the second is outside.
def uses_any(word, letters):
for letter in word.lower():
if letter in letters.lower():
return True
return False
When people first write functions like this, it is a common error to put both return
statements inside the loop, like this.
def uses_any_incorrect(word, letters):
for letter in word.lower():
if letter in letters.lower():
return True
else:
return False # INCORRECT!
Ask a virtual assistant what’s wrong with this version.
7.9.2. Exercise#
Write a function named uses_none
that takes a word and a string of forbidden letters, and returns True
if the word does not use any of the forbidden letters.
Here’s an outline of the function that includes two doctests. Fill in the function so it passes these tests, and add at least one more doctest.
def uses_none(word, forbidden):
"""Checks whether a word avoid forbidden letters.
>>> uses_none('banana', 'xyz')
True
>>> uses_none('apple', 'efg')
False
"""
return None
7.9.3. Exercise#
Write a function called uses_only
that takes a word and a string of letters, and that returns True
if the word contains only letters in the string.
Here’s an outline of the function that includes two doctests. Fill in the function so it passes these tests, and add at least one more doctest.
def uses_only(word, available):
"""Checks whether a word uses only the available letters.
>>> uses_only('banana', 'ban')
True
>>> uses_only('apple', 'apl')
False
"""
return None
7.9.4. Exercise#
Write a function called uses_all
that takes a word and a string of letters, and that returns True
if the word contains all of the letters in the string at least once.
Here’s an outline of the function that includes two doctests. Fill in the function so it passes these tests, and add at least one more doctest.
def uses_all(word, required):
"""Checks whether a word uses all required letters.
>>> uses_all('banana', 'ban')
True
>>> uses_all('apple', 'api')
False
"""
return None
7.9.5. Exercise#
The New York Times publishes a daily puzzle called “Spelling Bee” that challenges readers to spell as many words as possible using only seven letters, where one of the letters is required. The words must have at least four letters.
For example, on the day I wrote this, the letters were ACDLORT
, with R
as the required letter.
So “color” is an acceptable word, but “told” is not, because it does not use R
, and “rat” is not because it has only three letters.
Letters can be repeated, so “ratatat” is acceptable.
Write a function called check_word
that checks whether a given word is acceptable.
It should take as parameters the word to check, a string of seven available letters, and a string containing the single required letter.
You can use the functions you wrote in previous exercises.
Here’s an outline of the function that includes doctests. Fill in the function and then check that all tests pass.
def check_word(word, available, required):
"""Check whether a word is acceptable.
>>> check_word('color', 'ACDLORT', 'R')
True
>>> check_word('ratatat', 'ACDLORT', 'R')
True
>>> check_word('rat', 'ACDLORT', 'R')
False
>>> check_word('told', 'ACDLORT', 'R')
False
>>> check_word('bee', 'ACDLORT', 'R')
False
"""
return False
According to the “Spelling Bee” rules,
Four-letter words are worth 1 point each.
Longer words earn 1 point per letter.
Each puzzle includes at least one “pangram” which uses every letter. These are worth 7 extra points!
Write a function called score_word
that takes a word and a string of available letters and returns its score.
You can assume that the word is acceptable.
Again, here’s an outline of the function with doctests.
def word_score(word, available):
"""Compute the score for an acceptable word.
>>> word_score('card', 'ACDLORT')
1
>>> word_score('color', 'ACDLORT')
5
>>> word_score('cartload', 'ACDLORT')
15
"""
return 0
7.9.6. Exercise#
You might have noticed that the functions you wrote in the previous exercises had a lot in common. In fact, they are so similar you can often use one function to write another.
For example, if a word uses none of a set forbidden letters, that means it doesn’t use any. So we can write a version of uses_none
like this.
def uses_none(word, forbidden):
"""Checks whether a word avoids forbidden letters.
>>> uses_none('banana', 'xyz')
True
>>> uses_none('apple', 'efg')
False
>>> uses_none('', 'abc')
True
"""
return not uses_any(word, forbidden)
There is also a similarity between uses_only
and uses_all
that you can take advantage of.
If you have a working version of uses_only
, see if you can write a version of uses_all
that calls uses_only
.
7.9.7. Exercise#
If you got stuck on the previous question, try asking a virtual assistant, “Given a function, uses_only
, which takes two strings and checks that the first uses only the letters in the second, use it to write uses_all
, which takes two strings and checks whether the first uses all the letters in the second, allowing repeats.”
Use run_doctests
to check the answer.
7.9.8. Exercise#
Now let’s see if we can write uses_all
based on uses_any
.
Ask a virtual assistant, “Given a function, uses_any
, which takes two strings and checks whether the first uses any of the letters in the second, can you use it to write uses_all
, which takes two strings and checks whether the first uses all the letters in the second, allowing repeats.”
If it says it can, be sure to test the result!
# Here's what I got from ChatGPT 4o December 26, 2024
# It's correct, but it makes multiple calls to uses_any
def uses_all(s1, s2):
"""Checks if all characters in s2 are in s1, allowing repeats."""
for char in s2:
if not uses_any(s1, char):
return False
return True
Copyright 2024 Allen B. Downey
Code license: MIT License
Text license: Creative Commons Attribution-NonCommercial-ShareAlike 4.0 International