.get() More with Less from Python Dictionaries

I recently started volunteering to help host a few interview practice nights for Seattle’s local Python user group, PuPPy (Puget Sound Programming Python), and have seen some amazing people coming together to solve whiteboarding problems in the dead of night. During these practice nights I’ve been delighted by the opportunities I’ve had to share one of my absolute favorite Python tricks and today I’d like to share it with you.

Have you been in a situation where you needed to store the frequency of keys within a dictionary and have written something similar to this?

# Example 1 
s = 'Hello World!'
counts = {}

for char in s:
    try:
        counts[char] += 1
        # Equivalent to counts[char] = counts[char] + 1     except KeyError:
        counts[char] = 1

I don’t particularly care for the look of that try/except block.

The problem here is that dictionaries do not have a default return value when a program uses bracket notation to access a key that is not already present within the dictionary. Instead Python handily crashes the program with a KeyError.

In the case above, if the code in the try clause throws a KeyError we know that the key is not present in the dictionary. The program then enters the except clause where we add the key to the dictionary by explicitly assigning it a value of 1. The next time that key is encountered the try clause will not throw an error and the key’s value will be incremented by 1.

What if there was a better way, something sleeker, something that doesn’t call out that you may be knowingly throwing errors within your code?

Well it turns out there is a method within dictionaries that does just that, .get(). Calling .get() on a dictionary while passing in the desired key will return the value associated with that key, or None if the key is not present. This method allows us to modify Example 1 into something a little less hacky in appearance.

# Example 2 
s = 'Hello World!'
counts = {}

for char in s:
    if counts.get(char):
        counts[char] += 1
    else:
        counts[char] = 1

That just feels better to look at, right? No errors on this horizon!

Now if the dictionary doesn’t contain the key, the default None is returned. Since None is considered falsey, the code within the if clause is not executed and we move into the else clause to assign the initial value for the given key.

Be careful when using .get() in logic though, if the value associated with the key is 0, that is also considered falsey and the code within the if clause would not execute. It wouldn’t make a difference with this specific example but it could in other scenarios. In those cases I would suggest something along the lines of if counts.get(char) != None: to check if the value associated with the key is anything other than None.

🤔 Hmm, but Example 2 doesn’t seem to have any less lines of code than Example 1. Well, there’s more goodness stashed away in .get(). This method also accepts an optional second argument that defines the default return value. Not satisfied with None? Then let’s do something about it!

# Example 3 
s = 'Hello World!'
counts = {}

for char in s:
    counts[char] = counts.get(char, 0) + 1

It’s… beautiful!

A lot is happening on that last line. The .get() method checks the dictionary for the given key and returns either the value associated with the key or the default return value we specified as the second argument, 0. We add 1 to whatever is returned from .get(), assign the sum as the value for the key, and call it a day.

The examples above have dealt with a simple counting scenario but .get() can be used practically anywhere you’re checking keys in a dictionary. If you’re specifically in the counting business, I’d recommend you that you also take a peek at the incredible Counter class from the Python Standard Library’s collections module.

# Example 4 
from collections import Counter

counts = Counter('Hello World!')

Thank you for reading my love letter to .get()! Let me know how you’ve used .get() or if you know a Python trick that you feel everyone should be embracing.

原文链接:.get() More with Less from Python Dictionaries

© 版权声明
THE END
喜欢就支持一下吧
点赞6 分享
评论 抢沙发

请登录后发表评论

    暂无评论内容