Pylint extension with performance anti-patterns
Project description
perflint
pylint extension for performance anti patterns
This project is an early beta. It will likely raise many false-positives in your code.
Installation
pip install perflint
Usage
pylint your_code/ --load-plugins=perflint
VS Code
Add these configuration properties to your .vscode/settings.json
file (create if it doesn't exist):
{
"python.linting.pylintEnabled": true,
"python.linting.enabled": true,
"python.linting.pylintArgs": [
"--load-plugins",
"perflint",
"--rcfile",
"${workspaceFolder}/.pylintrc"
],
}
Rules
W8101 : Unnecessary list()
on already iterable type (unnecessary-list-cast
)
Using a list()
call to eagerly iterate over an already iterable type is inefficient as a second list iterator is created, after first iterating the value:
def simple_static_tuple():
"""Test warning for casting a tuple to a list."""
items = (1, 2, 3)
for i in list(items): # [unnecessary-list-cast]
print(i)
W8102: Incorrect iterator method for dictionary (incorrect-dictionary-iterator
)
Python dictionaries store keys and values in two separate tables. They can be individually iterated. Using .items()
and discarding either the key or the value using _
is inefficient, when .keys()
or .values()
can be used instead:
def simple_dict_keys():
"""Check that dictionary .items() is being used correctly. """
fruit = {
'a': 'Apple',
'b': 'Banana',
}
for _, value in fruit.items(): # [incorrect-dictionary-iterator]
print(value)
for key, _ in fruit.items(): # [incorrect-dictionary-iterator]
print(key)
W8201: Loop invariant statement (loop-invariant-statement
)
The body of loops will be inspected to determine statements, or expressions where the result is constant (invariant) for each iteration of a loop. This is based on named variables which are not modified during each iteration.
For example:
def loop_invariant_statement():
"""Catch basic loop-invariant function call."""
x = (1,2,3,4)
for i in range(10_000):
# x is never changed in this loop scope,
# so this expression should be evaluated outside
print(len(x) * i) # [loop-invariant-statement]
# ^^^^^^
len(x)
should be evaluated outside the loop since x
is not modified within the loop.
def loop_invariant_statement():
"""Catch basic loop-invariant function call."""
x = (1,2,3,4)
n = len(x)
for i in range(10_000):
print(n * i) # [loop-invariant-statement]
The loop-invariance checker will underline expressions and subexpressions within the body using the same rules:
def loop_invariant_statement_more_complex():
"""Catch basic loop-invariant function call."""
x = [1,2,3,4]
i = 6
for j in range(10_000):
# x is never changed in this loop scope,
# so this expression should be evaluated outside
print(len(x) * i + j)
# ^^^^^^^^^^ [loop-invariant-statement]
Methods are blindly considered side-effects, so if a method is called on a variable, it is assumed to have possibly changed in value and therefore not loop-invariant:
def loop_invariant_statement_method_side_effect():
"""Catch basic loop-invariant function call."""
x = [1,2,3,4]
i = 6
for j in range(10_000):
print(len(x) * i + j)
x.clear() # x changes as a side-effect
The loop-invariant analysis will walk up the AST until it gets to the whole loop body, so an entire branch could be marked.
For example, the expression len(x) > 2
is invariant and therefore should be outside the loop. Also, because x * i
is invariant, that statement should also be outside the loop, therefore the entire branch will be marked:
def loop_invariant_branching():
"""Ensure node is walked up to find a loop-invariant branch"""
x = [1,2,3,4]
i = 6
for j in range(10_000):
# Marks entire branch
if len(x) > 2:
print(x * i)
Notes on loop invariance
Functions can have side-effects (print is a good example), so the loop-invariant scanner may give some false-positives.
It will also highlight dotted expresions, e.g. attribute lookups. This may seem noisy, but in some cases this is valid, e.g.
from os.path import exists
import os
def dotted_import():
for _ in range(100_000):
return os.path.exists('/')
def direct_import():
for _ in range(100_000):
return exists('/')
direct_import()
is 10-15% faster than dotted_import()
because it doesn't need to load the os
global, the path
attribute and the exists
method for each iteration.
W8202: Global name usage in a loop (loop-invariant-global-usage
)
Loading globals is slower than loading "fast" local variables. The difference is marginal, but when propagated in a loop, there can be a noticable speed improvement, e.g.:
d = {
"x": 1234,
"y": 5678,
}
def dont_copy_dict_key_to_fast():
for _ in range(100000):
d["x"] + d["y"]
d["x"] + d["y"]
d["x"] + d["y"]
d["x"] + d["y"]
d["x"] + d["y"]
def copy_dict_key_to_fast():
i = d["x"]
j = d["y"]
for _ in range(100000):
i + j
i + j
i + j
i + j
i + j
copy_dict_key_to_fast()
executes 65% faster than dont_copy_dict_key_to_fast()
R8203 : Try..except blocks have a significant overhead. Avoid using them inside a loop (loop-try-except-usage
).
Up to Python 3.10, try...except
blocks are computationally expensive compared with if
statements.
Avoid using them in a loop as they can cause significant overheads. Refactor your code to not require iteration specific details and put the entire loop in the body of a try
block.
W8204 : Looped slicing of bytes objects is inefficient. Use a memoryview() instead (memoryview-over-bytes
)
Slicing of bytes
is slow as it creates a copy of the data within the requested window. Python has a builtin type, memoryview
for zero-copy interactions:
def bytes_slice():
"""Slice using normal bytes"""
word = b'A' * 1000
for i in range(1000):
n = word[0:i]
# ^^^^^^^^^ memoryview-over-bytes
def memoryview_slice():
"""Convert to a memoryview first."""
word = memoryview(b'A' * 1000)
for i in range(1000):
n = word[0:i]
memoryview_slice()
is 30-40% faster than bytes_slice()
W8205 : Importing the "%s" name directly is more efficient in this loop. (dotted-import-in-loop
)
In Python you can import a module and then access submodules as attributes. You can also access functions as attributes of that module. This keeps your import statements minimal, however, if you use this method in a loop it is inefficient because each loop iteration it will load global, load attribute and then load method. Because the name isn't an object, "load method" falls back to load attribute via a slow internal path.
Importing the desired function directly is 10-15% faster:
import os # NOQA
def test_dotted_import(items):
for item in items:
val = os.environ[item] # Use `from os import environ`
def even_worse_dotted_import(items):
for item in items:
val = os.path.exists(item) # Use `from os.path import exists` instead
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Hashes for perflint-0.2.0-py2.py3-none-any.whl
Algorithm | Hash digest | |
---|---|---|
SHA256 | d4f6917527f7e56d56cd7d58801872d81d7c84abc3800778d6d44c801cd5a489 |
|
MD5 | 903a180963cb8f8e62d4962639f05f15 |
|
BLAKE2b-256 | cb2d1c67df15dd8aa90bf747b0d52c005229d884c2f934513fd037038eab2234 |