14 KiB
This article is also published in The FOSS Albatross.
Known as an easy-to-use and flexible programming language, Python nevertheless still has plenty of tricks you can use to make your code prettier and faster to write. No matter if you’re new to Python or have years of experience, read more to find a tip for you!
1. F-strings
When printing out lots of text, you might find that you have to add lots of text together, which you might do by string concatenation:
= "John"
name = "Doe"
lastname print("My name is " + name + " and my last name is " + lastname + ".")
As you can see, it gets quite long and cumbersome if the string is long enough and if you’re adding enough strings. This is where f-strings come in to make the code readable and actually more performant! These special strings begin with the letter “f” immediately before the opening quote of a string, and any expression in curly braces will be evaluated.
print(f"My name is {name} and my last name is {lastname}.")
Completely clear, right?
That’s not all they can do, either! F-strings also provide some ways to easily format the data by providing how you want the data to be displayed after a colon in the expression.
For example, you can limit the number of decimal places shown in a float to two decimal places…
= 123.456789
num print(f"{num:.2f}")
…and even dates and times!
from datetime import date
= date(2022, 8, 21)
today print(f"{today:%m/%d/%Y}")
Output:
08/21/2022
2. Using “if” to its maximum potential
Like any language, Python has its quirks with values that evaluate to
True
and False
in if statements. Specifically,
only the following are False and every other value is
True.
- Zero (
0
and0.0
) - Empty containers, such as lists, tuples, dictionaries, sets, and strings
False
None
This means that actions like checking for empty containers can be drastically shortened from:
= []
array if len(array) != 0:
print("something is in the list!")
to:
= []
array if array:
print("something is in the list!")
3. List comprehensions
You can generate a new list very easily with an inline
for
.
= [i for i in range(10)]
array print(array)
Output:
0, 1, 2, 3, 4, 5, 6, 7, 8, 9] [
List comprehensions are just syntactic sugar for a regular for loop, so the above code is equivalent to:
= []
array for i in range(10):
array.append(i)print(array)
They’re also useful for performing operations on the
elements, much like map
in other functional programming
languages.
= [i for i in range(10)]
array = [i ** 2 for i in array]
new_array print(new_array)
Output:
0, 1, 4, 9, 16, 25, 36, 49, 64, 81] [
In addition, as an alternative to filter
, you can only
include specific elements by adding an if
at the end:
= [i ** 2 for i in range(10) if i % 2 == 0]
array print(array)
Output:
0, 4, 16, 36, 64] [
This is equivalent to:
= []
array for i in range(10):
if i % 2 == 0:
** 2)
array.append(i print(array)
4. Slicing strings and arrays
Compared to more traditional languages that only allow array accesses
array[i]
for one element, Python lets you “slice” a list or
any other sequence (like strings) by a colon between the two
indexes.
= "hello world!"
string = [1, 2, 3, 4, 5, 6]
array
print(string[2:7])
print(array[:-2])
Output:
llo w
[1, 2, 3, 4]
5. Tuple expansion
Tuple expansion is one of Python’s most powerful features, allowing you to expand or assign multiple values in a single line.
= 1, 2
a, b = "baguette", "tomato" + "cheese"
s1, s2 print(a, b)
print(s1, s2)
Output:
1 2
baguette tomatocheese
The code above effectively assigns a tuple (a, b)
to
another tuple (1, 2)
and is equivalent to the code
below:
= 1
a = 2
b = "baguette"
s1 = "tomato" + "cheese" s2
This can let you return multiple values from a function.
def init():
return (5, 6)
= init() a, b
And it’s also really useful in swapping variables!
= 2
a = 4
b
# oops, assigned them wrong
= a, b b, a
Tuple expansion can even be used in for loops to iterate over tuples or dictionaries!
= {
items "apple": "juice",
"orange": "pulp",
"banana": "smoothie",
"mango": "slushie"
}
for key, value in items.items():
print(key, value)
Output:
apple juice
banana smoothie
mango slushie
orange pulp
6. Shorter code with shortcircuiting
Imagine you want to assign a variable based on the value of another variable. You might do it this way:
= "John"
name if name == "John":
= "Doe"
last_name else:
= "Unknown" last_name
But that’s long, and last_name
appears twice. You can
have an inline if to do it in just one line:
= "John"
name = "Doe" if name == "John" else "Unknown" last_name
Much nicer!
You also might be familiar with logical boolean operators and how they’re used in if statements.
if password == "hunter2" and name == "Joe":
print("password accepted")
But you might not know some of their quirks or exactly how they work.
If you have two things opposite an and
, for example
x and y
:
- If
x
evaluates to True,y
is returned. - If
x
evaluates to False,x
is returned.
This might not seem that useful, but then look at or
in
x or y
:
- If
x
evaluates to True,x
is returned. - If
x
evaluates to False,y
is returned.
This might also not seem that useful until you learn that logical operators can also be used in assignments, where they’re most commonly used for fallback values.
= input("Accept the EULA? (Y/n) ") or "y" confirm
Because input
returns a string, if the user doesn’t
enter anything, it evaluates to False, so the value "y"
is
assigned to confirm
.
7. Safer file handing with context managers
In most languages, when you write something to a file, it’s actually held in a buffer until you close it manually or automatically when the program ends.
In Python, you would run the following to write to a file:
file = open("myfile.txt")
file.write("hello file")
file.close()
But that’s a lot of lines, and you might forget to run
file.close()
, particularly in long programs with lots of
stuff being written to the file.
That’s where context managers come in, abstracting
the whole process and calling file.close()
automagically:
with open("myfile.txt") as file:
file.write("hello file")
8. Nicer iteration with zip() and enumerate()
Python’s for loop is commonly known in other programming languages as
a for-each loop. This is great if you just want each item in an
iterable, but sometimes you want the index too! Instead of having to
resort to range(len(array))
, instead you can use
enumerate()
and tuple expansion to easily get both the
index of the element and the element itself:
= ["a", "b", "c", "d", "e"]
array for i, c in enumerate(array):
print(i, c)
Output:
0 a
1 b
2 c
3 d
4 e
In a similar vein, if you have two arrays you want to process at the
same time, you don’t need to use range(len(array))
when you
have zip()
, which will bundle the different iterators into
one big one as big as the smallest iterable.
= [1, 2, 3, 4]
ints = ("pomme", "poutine", "pinterest", "pear")
strs = [1.0, 2.0, 3.0, 4.0, 5.0, 6.0]
floats for i, s, f in zip(ints, strs, floats):
print(i, s, f)
Output:
1 pomme 1.0
2 poutine 2.0
3 pinterest 3.0
4 pear 4.0
9. Nicer paths with pathlib
Python is a cross-platform language, and if there’s one annoying
difference between Windows and macOS / Linux, it’s that their path
separators are different: Windows uses the backslash (\
)
while macOS and Linux use the forward slash (/
), and
attempting to access a path the wrong way will result in file not
found.
So, in the olden days, you would have to string together a long
string with os.path.join
just to be safe:
import os
= os.path.join("folder", os.path.join("subfolder", "subsubfile.txt")) path
Doesn’t that look clunky?
Luckily, a “new” (read: years-old) addition to the standard library makes that much easier:
from pathlib import Path
= Path("folder") / "subfolder" / "subsubfile.txt" path
It also includes a bunch of helper methods to check for things and navigate the filesystem tree just because Python loves making your life easier.
from pathlib import Path
= Path("folder")
path
if path.exists() and path.is_file():
print("yay!")
print(path.parent)
print(path.suffix)
if not path.exists():
### create the folder if it doesn't exist path.mkdir()
10. Iterable unpacking
n competitive programming, often you have to print out space-separated results. This can be done by the mildly inconveniencing
print(" ".join([1, 2, 3, 4]))
or heaven forbid, via iteration:
for i in [1, 2, 3, 4]:
print(i, end=" ")
print()
which is where iterable unpacking comes in, and you can go straight to
= [1, 2, 3, 4]
array print(*array)
The unpacking operator (the asterisk) basically gets rid of the container and throws all of the elements inside directly into the print function as parameters.
That means that the above line of code is equivalent to:
print(1, 2, 3, 4)
which nicely prints out the integers separated by spaces with a newline at the end.
But wait, there’s more! The unpacking operator is also commonly used in function definitions as a catch-all parameter for extra arguments, stuffing them into a list.
def init(a, b, *args):
print(a, b)
print(args)
1, 2, "pomme", 4, 6.0) init(
Output:
1, 2
['pomme', 4, 6.0]
You can also use this in normal assignment to, say, only get the first and last elements of an array:
*args, last = [1, 2, 3, 4, 5]
first, print(first)
print(args)
print(last)
Output:
1
[2, 3, 4]
5
11. Else outside of if
Everyone knows what if-else does. But did you know that Python also lets you use it after loops and exception blocks?
- In loops,
else
is run only if the loop did notbreak
. - In exception blocks,
else
is run only if there was no exception.
These are especially useful so you don’t have to add an indicator variable yourself:
for i in array:
if i.is_tomato():
print("found a tomato!")
break
else:
print("no tomato found :(")
…is equivalent to
= False
found_tomato for i in array:
if i.is_tomato():
= True
found_tomato break
if found_tomato:
print("found a tomato!")
else:
print("non tomato found :(")
Similarly, in a try-except block:
try:
= int(input("Enter a number: "))
number except ValueError:
print("That's not a number >:(")
else:
print("You know how to follow instructions! :D")
…which is equivalent to:
= True
is_number try:
= int(input("Enter a number: "))
number except ValueError:
= False
is_number
if is_number:
print("You know how to follow instructions! :D")
else:
print("That's not a number >:(")
12. Type hinting
When you use a library, your IDE often knows what types a function will accept so you don’t have to guess.
These come from type hints in the code, which you can use in your own code, especially if there is a lot of reused code and it’s a long program.
A colon after a variable shows its type:
int = 4 a:
…while an arrow after a function shows its return value.
def pow(x: int, y: int) -> int:
return x ** 2
def largest(array: list[int]) -> int:
return max(array)
In more complex programs, type hinting is especially useful as your IDE can provide autocomplete and better syntax highlighting as it knows the limits of your program, so if you accidentally assume the wrong type of an object, your IDE will complain and find the bug before it causes a runtime crash.
13. The walrus operator (:=)
Say you’re doing something over and over again and checking for a condition until it’s true. You could do it in a while loop:
import requests
= requests.get("https://google.com")
r while r.status_code != 200:
print(r.status_code)
1)
time.sleep(= requests.get("https://google.com") r
But the assignment is repeated twice!
Introducing the walrus operator, which assigns and returns the assigned value in the same statement, allowing for less repeated code and some pretty crazy one-liners:
import requests
while (r := requests.get("https://google.com")).status_code != 200:
print(r.status_code)
1) time.sleep(
Although thirteen of them are covered here, there are endless ways to optimise your code so that it’s faster to read, write, and run. Python is a “batteries included” language — chances are that the way you’re used to doing things in other languages have a shorter and more concise method in Python.
The most important tip I can give you is to check the standard library if you want to do something — from image format recognition to config file management to basic database operations, the standard library is chock full of useful tools included over the years.