Notes from Raymond Hettinger’s talk at pycon US 2013 video, slides.
The code examples and direct quotes are all from Raymond’s talk. I’ve reproduced them here for my own edification and the hopes that others will find them as handy as I have!
for i in [0, 1, 2, 3, 4, 5]:
    print i**2
for i in range(6):
    print i**2for i in xrange(6):
    print i**2xrange creates an iterator over the range producing the values one at a time. This approach is much more memory efficient than range. xrange was renamed to range in python 3.
colors = ['red', 'green', 'blue', 'yellow']
for i in range(len(colors)):
    print colors[i]for color in colors:
    print colorcolors = ['red', 'green', 'blue', 'yellow']
for i in range(len(colors)-1, -1, -1):
    print colors[i]for color in reversed(colors):
    print colorcolors = ['red', 'green', 'blue', 'yellow']
for i in range(len(colors)):
    print i, '--->', colors[i]for i, color in enumerate(colors):
    print i, '--->', colorIt’s fast and beautiful and saves you from tracking the individual indices and incrementing them.
Whenever you find yourself manipulating indices [in a collection], you’re probably doing it wrong.
names = ['raymond', 'rachel', 'matthew']
colors = ['red', 'green', 'blue', 'yellow']
n = min(len(names), len(colors))
for i in range(n):
    print names[i], '--->', colors[i]
for name, color in zip(names, colors):
    print name, '--->', colorfor name, color in izip(names, colors):
    print name, '--->', colorzip creates a new list in memory and takes more memory. izip is more efficient than zip.
Note: in python 3 izip was renamed to zip and promoted to a builtin replacing the old zip.
colors = ['red', 'green', 'blue', 'yellow']
# Forward sorted order
for color in sorted(colors):
    print colors
# Backwards sorted order
for color in sorted(colors, reverse=True):
    print colorscolors = ['red', 'green', 'blue', 'yellow']
def compare_length(c1, c2):
    if len(c1) < len(c2): return -1
    if len(c1) > len(c2): return 1
    return 0
print sorted(colors, cmp=compare_length)print sorted(colors, key=len)The original is slow and unpleasant to write. Also, comparison functions are no longer available in python 3.
blocks = []
while True:
    block = f.read(32)
    if block == '':
        break
    blocks.append(block)blocks = []
for block in iter(partial(f.read, 32), ''):
    blocks.append(block)iter takes two arguments. The first you call over and over again and the second is a sentinel value.
def find(seq, target):
    found = False
    for i, value in enumerate(seq):
        if value == target:
            found = True
            break
    if not found:
        return -1
    return idef find(seq, target):
    for i, value in enumerate(seq):
        if value == target:
            break
    else:
        return -1
    return iInside of every for loop is an else.
d = {'matthew': 'blue', 'rachel': 'green', 'raymond': 'red'}
for k in d:
    print k
for k in d.keys():
    if k.startswith('r'):
        del d[k]When should you use the second and not the first? When you’re mutating the dictionary.
If you mutate something while you’re iterating over it, you’re living in a state of sin and deserve what ever happens to you.
d.keys() makes a copy of all the keys and stores them in a list. Then you can modify the dictionary.
Note: in python 3 to iterate through a dictionary you have to explicidly write: list(d.keys()) because d.keys() returns a “dictionary view” (an iterable that provide a dynamic view on the dictionary’s keys). See documentation.
# Not very fast, has to re-hash every key and do a lookup
for k in d:
    print k, '--->', d[k]
# Makes a big huge list
for k, v in d.items():
    print k, '--->', vfor k, v in d.iteritems():
    print k, '--->', viteritems() is better as it returns an iterator.
Note: in python 3 there is no iteritems() and items() behaviour is close to what iteritems() had. See documentation.
names = ['raymond', 'rachel', 'matthew']
colors = ['red', 'green', 'blue']
d = dict(izip(names, colors))
# {'matthew': 'blue', 'rachel': 'green', 'raymond': 'red'}For python 3: d = dict(zip(names, colors))
colors = ['red', 'green', 'red', 'blue', 'green', 'red']
# Simple, basic way to count. A good start for beginners.
d = {}
for color in colors:
    if color not in d:
        d[color] = 0
    d[color] += 1
# {'blue': 1, 'green': 2, 'red': 3}d = {}
for color in colors:
    d[color] = d.get(color, 0) + 1
# Slightly more modern but has several caveats, better for advanced users
# who understand the intricacies
d = defaultdict(int)
for color in colors:
d[color] += 1names = ['raymond', 'rachel', 'matthew', 'roger',
         'betty', 'melissa', 'judith', 'charlie']
# In this example, we're grouping by name length
d = {}
for name in names:
    key = len(name)
    if key not in d:
        d[key] = []
    d[key].append(name)
# {5: ['roger', 'betty'], 6: ['rachel', 'judith'], 7: ['raymond', 'matthew', 'melissa', 'charlie']}
d = {}
for name in names:
    key = len(name)
    d.setdefault(key, []).append(name)d = defaultdict(list)
for name in names:
    key = len(name)
    d[key].append(name)d = {'matthew': 'blue', 'rachel': 'green', 'raymond': 'red'}
while d:
    key, value = d.popitem()
    print key, '-->', valuepopitem is atomic so you don’t have to put locks around it to use it in threads.
defaults = {'color': 'red', 'user': 'guest'}
parser = argparse.ArgumentParser()
parser.add_argument('-u', '--user')
parser.add_argument('-c', '--color')
namespace = parser.parse_args([])
command_line_args = {k:v for k, v in vars(namespace).items() if v}
# The common approach below allows you to use defaults at first, then override them
# with environment variables and then finally override them with command line arguments.
# It copies data like crazy, unfortunately.
d = defaults.copy()
d.update(os.environ)
d.update(command_line_args)d = ChainMap(command_line_args, os.environ, defaults)ChainMap has been introduced into python 3. Fast and beautiful.
twitter_search('@obama', False, 20, True)twitter_search('@obama', retweets=False, numtweets=20, popular=True)Is slightly (microseconds) slower but is worth it for the code clarity and developer time savings.
# Old testmod return value
doctest.testmod()
# (0, 4)
# Is this good or bad? You don't know because it's not clear.# New testmod return value, a namedTuple
doctest.testmod()
# TestResults(failed=0, attempted=4)A namedTuple is a subclass of tuple so they still work like a regular tuple, but are more friendly.
To make a namedTuple:
TestResults = namedTuple('TestResults', ['failed', 'attempted'])p = 'Raymond', 'Hettinger', 0x30, 'python@example.com'
# A common approach / habit from other languages
fname = p[0]
lname = p[1]
age = p[2]
email = p[3]fname, lname, age, email = pThe second approach uses tuple unpacking and is faster and more readable.
def fibonacci(n):
    x = 0
    y = 1
    for i in range(n):
        print x
        t = y
        y = x + y
        x = tdef fibonacci(n):
    x, y = 0, 1
    for i in range(n):
        print x
        x, y = y, x + yProblems with first approach
The second approach is more high-level, doesn’t risk getting the order wrong and is fast.
tmp_x = x + dx * t
tmp_y = y + dy * t
tmp_dx = influence(m, x, y, dx, dy, partial='x')
tmp_dy = influence(m, x, y, dx, dy, partial='y')
x = tmp_x
y = tmp_y
dx = tmp_dx
dy = tmp_dyx, y, dx, dy = (x + dx * t,
                y + dy * t,
                influence(m, x, y, dx, dy, partial='x'),
                influence(m, x, y, dx, dy, partial='y'))Basically, just don’t move data around unecessarily.
names = ['raymond', 'rachel', 'matthew', 'roger',
         'betty', 'melissa', 'judith', 'charlie']
s = names[0]
for name in names[1:]:
    s += ', ' + name
print sprint ', '.join(names)names = ['raymond', 'rachel', 'matthew', 'roger',
         'betty', 'melissa', 'judith', 'charlie']
del names[0]
# The below are signs you're using the wrong data structure
names.pop(0)
names.insert(0, 'mark')names = deque(['raymond', 'rachel', 'matthew', 'roger',
               'betty', 'melissa', 'judith', 'charlie'])
# More efficient with deque
del names[0]
names.popleft()
names.appendleft('mark')# Mixes business / administrative logic and is not reusable
def web_lookup(url, saved={}):
    if url in saved:
        return saved[url]
    page = urllib.urlopen(url).read()
    saved[url] = page
    return page@cache
def web_lookup(url):
    return urllib.urlopen(url).read()Note: since python 3.2 there is a decorator for this in the standard library: functools.lru_cache.
# Saving the old, restoring the new
old_context = getcontext().copy()
getcontext().prec = 50
print Decimal(355) / Decimal(113)
setcontext(old_context)with localcontext(Context(prec=50)):
    print Decimal(355) / Decimal(113)f = open('data.txt')
try:
    data = f.read()
finally:
    f.close()with open('data.txt') as f:
    data = f.read()# Make a lock
lock = threading.Lock()
# Old-way to use a lock
lock.acquire()
try:
    print 'Critical section 1'
    print 'Critical section 2'
finally:
    lock.release()# New-way to use a lock
with lock:
    print 'Critical section 1'
    print 'Critical section 2'try:
    os.remove('somefile.tmp')
except OSError:
    passwith ignored(OSError):
    os.remove('somefile.tmp')ignored is is new in python 3.4, documentation.
Note: ignored is actually called suppress in the standard library.
To make your own ignored context manager in the meantime:
@contextmanager
def ignored(*exceptions):
    try:
        yield
    except exceptions:
        passStick that in your utils directory and you too can ignore exceptions
# Temporarily redirect standard out to a file and then return it to normal
with open('help.txt', 'w') as f:
    oldstdout = sys.stdout
    sys.stdout = f
    try:
        help(pow)
    finally:
        sys.stdout = oldstdoutwith open('help.txt', 'w') as f:
    with redirect_stdout(f):
        help(pow)redirect_stdout is proposed for python 3.4, bug report.
To roll your own redirect_stdout context manager
@contextmanager
def redirect_stdout(fileobj):
    oldstdout = sys.stdout
    sys.stdout = fileobj
    try:
        yield fieldobj
    finally:
        sys.stdout = oldstdoutTwo conflicting rules:
Raymond’s rule:
result = []
for i in range(10):
s = i ** 2
    result.append(s)
print sum(result)print sum(i**2 for i in xrange(10))First way tells you what to do, second way tells you what you want.