forked from SigmaQuan/Better-Python-59-Ways
-
Notifications
You must be signed in to change notification settings - Fork 2
/
item_17_be_defensive.py
247 lines (182 loc) · 8.3 KB
/
item_17_be_defensive.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
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
# Item 17: Be defensive when iterating over arguments
# When a function takes a list of objects as a parameter, it's often important
# to iterate over that list multiple times. For example, say you want to
# analyze tourism number for the U.S. state of Texas. Imagine the data set is
# the number of visitors to each city (in millions per year). You'd like to
# figure out what percentage of overall tourism each city receives.
# To do this you need a normalization function. It sums the inputs to
# determine the total number of tourists per year. Then it divides each city's
# individual visitor count by the total to find that city's contribution to
# the whole.
def normalize(numbers):
total = sum(numbers)
result = []
for value in numbers:
percent = 100 * value / total
result.append(percent)
return result
# This function works when given a list of visits.
visits = [15, 35, 80]
percentages = normalize(visits)
print(percentages)
# [11.538461538461538, 26.923076923076923, 61.53846153846154]
# To scale this up, I need to read the data from a file that contains every
# city in all of Texas. I define a generator to do this because then I can
# reuse the same function later when I want to compute tourism numbers for the
# whole world, a much larger data set (see Item 16: "Consider generators
# instead of returning lists").
def read_visits(data_path):
with open(data_path) as f:
for line in f:
yield int(line)
# Surprisingly, calling normalize on the generator's return value produces no
# results.
path = 'item_17_my_numbers.txt'
it = read_visits(path)
percentages = normalize(it)
print(percentages)
# []
# The cause of this behavior is that an iterator only produces results a
# single time. If you iterate over an iterator or generator that has
# already raised a StopIteration exception, you won't get any results the
# second time around.
it = read_visits(path)
print(list(it))
print(list(it))
# [15, 35, 80]
# []
# What's confusing is that you also won't get any errors when you iterate over
# an already exhausted iterator. for loops, the list constructor, and many
# other throughout the Python standard library expect the StopIteration
# exception to be raised during normal operation. These functions can't tell
# the difference between an iterator that has no output and an iterator that
# hand output and is now exhausted.
# To solve this problem, you can explicitly exhaust an input iterator and keep
# a copy of its entire contents in a list. You can then iterate over the list
# version of the data as many times as you need to. Here's the same function
# as before, but it defensively copies the input iterator:
def normalize_copy(numbers):
numbers = list(numbers) # Copy of the iterator
total = sum(numbers)
result = []
for value in numbers:
percent = 100 * value / total
result.append(percent)
return result
# Now the function works correctly on a generator's return value.
it = read_visits(path)
percentages = normalize_copy(it)
print(percentages)
# [11.538461538461538, 26.923076923076923, 61.53846153846154]
# The problem with this approach is the copy of the input iterator's contents
# could be large. Copying the iterator could cause your program to run out of
# memory and crash. One way around this is to accept a function that returns
# a new iterator each time it's called.
def normalize_func(get_iter):
total = sum(get_iter()) # New iterator
result = []
for value in get_iter(): # New iterator
percent = 100 * value / total
result.append(percent)
return result
# To use normalize_func, you can pass in a lambda expression that calls the
# generator and produces a new iterator each time.
percentages = normalize_func(lambda: read_visits(path))
# https://www.quora.com/What-is-lambda-function-in-Python-and-why-do-we-need-it
# https://www.zhihu.com/question/20125256
print(percentages)
# [11.538461538461538, 26.923076923076923, 61.53846153846154]
# Though it works, having to pass a lambda function like this is clumsy. The
# better way to achieve the same result is to provide a new container class
# that implements the iterator protocol.
# The iterator protocol is how Python for loops and related expressions
# traverse the contents of a container type. When Python sees a statement like
# for x in foo will actually call iter(foo). The iter built-in function call
# the foo.__iter__ special method in turn. The __iter__ method mush return
# an iterator object (which itself implements the __next__ special method).
# Then the for loop repeatedly calls the next built-in function on the
# iterator object until it's exhausted (and raise a StopIteration exception).
# It sounds complicated, but practically speaking you can achieve all of this
# behavior for your classes by implementing the __iter__ method as a
# generator. Here, I define an iterable container class that reads the files
# containing tourism data:
class ReadVisits(object):
def __init__(self, data_path):
self.data_path = data_path
def __iter__(self):
with open(self.data_path) as f:
for line in f:
yield int(line)
# This new container type works correctly when passed to the original function
# without any modifications.
visits = ReadVisits(path)
percentages = normalize(visits)
print(percentages)
# [11.538461538461538, 26.923076923076923, 61.53846153846154]
# This works because the sum method in normalize will call
# ReadVisits.__iter__ to allocate a new iterator object. The for loop to
# normalize the numbers will also call __iter__ to a second iterator object.
# Each of those iterators will be advanced and exhausted independently,
# ensuring that each unique iteration sees all of the input data values. The
# only downside of this approach is that it reads the input data multiple
# times.
# Now that you know how containers like ReadVisits work, you can write your
# functions to ensure that parameters aren't just iterator. The protocol
# states that when an iterator is passed to the iter built-in function, iter
# will return the iterator itself. In contrast, when a container type is
# passed to iter, a new iterator object will be returned each time. Thus,
# you can test an input value for this behavior and raise a TypeError to
# reject iterators.
def normalize_defensive(numbers):
if iter(numbers) is iter(numbers): # An iterator -- bad!
raise TypeError('Must supply a container')
total = sum(numbers)
result = []
for value in numbers:
percent = 100 * value / total
result.append(percent)
return result
# This is ideal if you don't want to copy the full input iterator like
# normalize_copy above, but you also need to iterate over the input data
# multiple times. This function works as expected for list and ReadVisits
# inputs because they are containers. It will work for any type of container
# that follows the iterator protocol.
visits = [15, 35, 80]
print('[] Defensive:')
print(normalize_defensive(visits))
# [] Defensive:
# [11.538461538461538, 26.923076923076923, 61.53846153846154]
visits = ReadVisits(path)
print('Path Denfensive:')
print(normalize_defensive(visits))
# Path Denfensive:
# [11.538461538461538, 26.923076923076923, 61.53846153846154]
# The function will raise an exception if the input is iterable but not a
# container.
it = iter(visits)
print('normalize:')
print(normalize(it))
# normalize:
# []
print('normalize_copy:')
print(normalize_copy(it))
# normalize_copy:
# []
print('normalize_func:')
# print(normalize_func(it))
# TypeError: 'generator' object is not callable
print('normalize_defensive:')
# print(normalize_defensive(it))
# TypeError: Must supply a container
# Things to remember
# 1. Beware of functions that iterate over input arguments multiple times. If
# these arguments are iterators, you may see strange behavior and missing
# values.
# 2. Python's iterator protocol defines how containers and iterators interact
# with the iter and next built-in functions, for loops, and related
# expression.
# 3. You can easily define your own iterable container type by implementing
# the __iter__ method as a generator.
# 4. You can detect that a value is an iterator (instead of a container) if
# calling iter on it twice produces the same result, which can then be
# progressed with the next built-in function.