-
Notifications
You must be signed in to change notification settings - Fork 222
/
Copy pathitem_04_helper_function.py
149 lines (107 loc) · 5.15 KB
/
item_04_helper_function.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
# Item 4: Write helper functions instead of complex expressions
# Python's pithy syntax makes it easy to write single-line expressions that
# implement a lot of logic. For example, say you want to decode the query
# string from a URL. Here, each query string parameter represents an integer
# value:
from urllib.parse import parse_qs
my_values = parse_qs('red=5&blue=0&green=', keep_blank_values=True)
print(repr(my_values))
# $ python3 item_04_helper_function.py
# {'red': ['5'], 'green': [''], 'blue': ['0']}
# Some query string parameters may have multiple values, some may have single
# values, some may be present but have blank values, and some may be missing
# entirely. Using the get method on the result dictionary will return
# different values in each circumstance.
print("Red: ", my_values.get('red'))
print("Green: ", my_values.get('green'))
print("Opacity: ", my_values.get('opacity'))
# $ python3 item_04_helper_function.py
# Red: ['5']
# Green: ['']
# Opacity: None
# It'd be nice if a default value of 0 was assigned when a parameter isn't
# supplied or is blank. You might choose to do this with Boolean expressions
# because it feels like this logic doesn't merit a whole if statement or
# helper function quite yet.
# Python's syntax makes this choice all too easy. The trick here is that the
# empty string, the empty list, and zero all evaluate to False implicitly.
# Thus, the expressions below will evaluate to the subexpression after the or
# operator when the first subexpression is False.
# For query string 'red=5&blue=0&green='
red = my_values.get('red', [''])[0] or 0
green = my_values.get('green', [''])[0] or 0
opacity = my_values.get('opacity', [''])[0] or 0
print("Red: %r" % red)
print("Green: %r" % green)
print("Opacity: %r" % opacity)
# Red: '5'
# Green: 0
# Opacity: 0
# The red case works because the key is present in the my_values dictionary.
# The value is a list with one member: the string '5'. This string implicitly
# evaluates to True, so red is assigned to the first part of the or
# expression.
# The green case works because the value in the my_values dictionary is a list
# with one member: an empty string. The empty string implicitly evaluates to
# False, causing the or expression to evaluate to 0.
# The opacity case works because the value in the my_values dictionary is
# missing altogether. The behavior of the get method is to return its second
# argument if the key doesn't exist in the dictionary. The default value in
# this case is a list with one member, an empty string. When opacity isn't
# found in the dictionary, this code does exactly the same thing as the green
# case.
# However, this expression is difficult to read and it still doesn't do
# everything you need. You'd also want to ensure that all the parameter values
# are integers so you can use them in mathematical expressions. To do that,
# you'd wrap each expression with the int built-in function to parse the
# string as an integer.
red = int(my_values.get('red', [''])[0] or 0)
print('red: ', red)
# red: 5
# This is now extremely hard to read. There's so much visual noise. The code
# isn't approachable. A new reader of the code would have to spend too much
# time picking apart the expression to figure out what it actually does. Even
# though it's nice to keep things short, it's not worth trying to fit this all
# on one line.
# Python 2.5 added if/else conditional-or ternary-expressions to make cases
# like this clearer while keeping the code short.
red = my_values.get('red', [''])
red = int(red[0]) if red[0] else 0
print('red: ', red)
# red: 5
# This is better. For less complicated situations, if/else conditional
# expressions can make things very clear. But the example above is still not
# as clear as the alternative of a full if/else statement over multiple lines.
# Seeing all of the logic spread out like this makes the dense version seem
# even more complex.
green = my_values.get('green', [''])
if green[0]:
green = int(green[0])
else:
green = 0
# Writing a helper function is the way to go, especially if you need to use
# this logic repeatedly.
def get_first_int(values, key, default=0):
found = values.get(key, [''])
if found[0]:
found = int(found[0])
else:
found = default
return found
# The calling code is much clearer than complex expression using or and the
# two-line version using the if/else expression.
green = get_first_int(my_values, 'green')
print('green: ', green)
# green: 0
# As soon as your expressions get complicated, it's time to consider
# splitting them into smaller pieces and moving logic into helper functions.
# What you gain in readability always outweighs what brevity may have
# afforded you. Don't let Python's pithy syntax for complex expressions get
# you into a mess like this.
# Things to remember
# 1. Python's syntax makes it all too easy to write single-line expressions
# that are overly complicated and difficult to read.
# 2. Move complex expressions into helper functions, especially if you need to
# use the same logic repeatedly.
# 3. The if/else expression provides a more readable alternative to using
# Boolean operators like or and adn in expressions.