


























At my job, we have just upgraded Python from 3.7 to 3.9, and I got super excited about all the new features. This is a blog post of highlights from these two releases.
This was a bit of a controversial feature during Python 3.8’s development and PEP 572 was even part of the reason Guido resigned as a benevolent dictator.
The new feature is pretty straightforward, here are some examples straight from the PEP:
# Handle a matched regex
if (match := pattern.search(data)) is not None: # Do something with match
...
# A loop that can't be trivially rewritten using 2-arg iter()
while chunk := file.read(8192):
process(chunk)
# Reuse a value that's expensive to compute
[y := f(x), y**2, y**3]
# Share a subexpression between a comprehension filter clause and its output
filtered_data = [y for x in data if (y := f(x)) is not None]
I remember using the assignment expression only once in a personal project so far but I do feel it’s a nifty way to save a line of code here and there.
PEP 570 added positional-only arguments. These arguments have no externally-usable name and therefore cannot be called with kwargs. You would use them when you don’t want to expose the function parameter name so that changing it later won’t break anyone’s code.
def f(a, b, /, c, d, \*, e, f):
The function definition has two positional-only arguments (a, b), two parameters that can be both positional or keyword (c, d) and two keyword only parameters (e, f).
# Valid cals:
f(1, 2, c=3, d=4, e=5, f=6)
f(1, 2, 3, d=4, e=5, f=6)
f(1, 2, 3, 4, e=5, f=6)
# Invalid calls
f(a=1, b=2, c=3, d=4, e=5, f=6)
f(1, 2, 3, 4, 5, 6)
Positional only-arguments were already used in the Python standard library and I like that they became a core feature. They might not be super interesting in our day-to-day, but library authors do appreciate them.
This is a feature that I use a lot when debugging. Appending the = character to the f-string expression will print out the name of the variable used:
# Valid cals:
>>> user = "anze_pecar"
>>> member_since = date(2012, 1, 26)
>>> f"{user=} {member_since=}"
'user=anze_pecar member_since=datetime.date(2012, 1, 26)'
Very useful for debugging, but remember that using f-strings isn’t advised for logging calls.
Python 3.9 added a union operator | to dicts (PEP 584), so joining dicts became a lot easier:
>>> d = {'spam': 1, 'eggs': 2, 'cheese': 3}
>>> e = {'cheese': 'cheddar', 'aardvark': 'Ethel'}
>>> d | e
{'spam': 1, 'eggs': 2, 'cheese': 'cheddar', 'aardvark': 'Ethel'}
>>> e | d
{'cheese': 3, 'aardvark': 'Ethel', 'spam': 1, 'eggs': 2}
This was a small but good addition that saves you an extra import anytime you want to use a list, tuple, or a dict in a type definition (see PEP 585 for the full list of generics that we can now use).
l: list[dict[str, str]] = []
The LL(1) parser was removed in favor of a PEG parser (PEP 617). A big change for Python, but nothing tangible in the current release. This paved the way for things like the match expression and better error reporting in Python 3.10.
PEP 615 added a new module zoneinfo so that we no longer need to use 3rd party packages (pytz) for dealing with time zone data.
Django switched to using zoneinfo in version 4.0 so that’s probably going to be a fun upgrade for us 😅
There will now be a new Python release every year, Python 3.10 being the first such release. 🎉
Besides the ones mentioned there have been many more improvements in 3.8 so jump over to the changelogs for 3.8 and 3.9 to read them all.
此内容由惯性聚合(RSS阅读器)自动聚合整理,仅供阅读参考。 原文来自 — 版权归原作者所有。