.. SPDX-FileCopyrightText: 2021 Veit Schiele
..
.. SPDX-License-Identifier: BSD-3-Clause
Performance
===========
Python can be used to write and test code quickly because it is an interpreted
language that types dynamically. However, these are also the reasons it is slow
when performing simple tasks repeatedly, for example in loops.
When developing code, there can often be trade-offs between different
implementations. However, at the beginning of the development of an algorithm,
it is usually counterproductive to worry about the efficiency of the code.
«We should forget about small efficiencies, say about 97% of the time:
premature optimization is the root of all evil. Yet we should not pass up
our opportunities in that critical 3%.»[#]_
.. [#] `Donald Knuth, founder of Literate Programming
`_, in Computer Programming as an
Art (1974)
k-Means example
---------------
In the following, I show examples of the `k-means algorithm
`_ to form a previously known
number of groups from a set of objects. This can be achieved in the following
three steps:
#. Choose the first :samp:`k` elements as cluster centres
#. Assign each new element to the cluster with the least increase in variance.
#. Update the cluster centre
Steps 2 and 3 are repeated until the assignments no longer change.
A possible implementation with pure Python could look like this:
.. literalinclude:: py_kmeans.py
:caption: py_kmeans.py
:name: py_kmeans.py
We can create sample data with:
.. code-block:: python
from sklearn.datasets import make_blobs
points, labels_true = make_blobs(
n_samples=1000, centers=3, random_state=0, cluster_std=0.60
)
And finally, we can perform the calculation with:
.. code-block:: python
kmeans(points, 10)
Performance measurements
------------------------
Once you have worked with your code, it can be useful to examine its efficiency
more closely. The :doc:`ipython-profiler` or :doc:`scalene` can be used for
this.
.. seealso::
* `airspeed velocity (asv) `_
is a tool for benchmarking Python packages during their lifetime. Runtime,
memory consumption and even user-defined values can be recorded and
displayed in an interactive web frontend.
* `Python Speed Center `_
* `Tracing the Python GIL
`_
.. toctree::
:hidden:
:titlesonly:
:maxdepth: 0
ipython-profiler.ipynb
scalene.ipynb
Search for existing implementations
-----------------------------------
You should not try to reinvent the wheel: If there are existing implementations,
you should use them. There are even two implementations for the k-means
algorithm:
* `sklearn.cluster.KMeans
`_
.. code-block:: python
from sklearn.cluster import KMeans
KMeans(10).fit_predict(points)
* `dask_ml.cluster.KMeans
`_
.. code-block:: python
from dask_ml.cluster import KMeans
KMeans(10).fit(points).predict(points)
The best that could be said against these existing solutions is that they could
create a considerable overhead in your project if you are not already using
`scikit-learn `_ or `Dask-ML
`_ elsewhere. In the following, I will therefore show you
further possibilities to optimise your own code.
Find anti-patterns
------------------
Then you can use :doc:`perflint` to search your code for the most common
performance anti-patterns in Python.
.. toctree::
:hidden:
:titlesonly:
:maxdepth: 0
perflint
.. seealso::
* `Effective Python `_
Vectorisations with NumPy
-------------------------
:doc:`../workspace/numpy/index` moves repetitive operations into a statically
typed compiled layer, combining the fast development time of Python with the
fast execution time of C. You may be able to use
:doc:`../workspace/numpy/ufunc`, :doc:`vectorisation
<../workspace/numpy/vectorisation>` and
:doc:`../workspace/numpy/indexing-slicing` in all combinations to move
repetitive operations into compiled code to avoid slow loops.
With NumPy we can do without some loops:
.. literalinclude:: np_kmeans.py
:caption: np_kmeans.py
:name: np_kmeans.py
:lines: 1-8
The advantages of NumPy are that the Python overhead only occurs per array and
not per array element. However, because NumPy uses a specific language for array
operations, it also requires a different mindset when writing code. Finally, the
batch operations can also lead to excessive memory consumption.
Special data structures
-----------------------
:doc:`../workspace/pandas/index`
for SQL-like :doc:`../workspace/pandas/group-operations` and
:doc:`../workspace/pandas/aggregation`.
This way you can also bypass the loop in the ``compute_centers`` method:
.. literalinclude:: pd_kmeans.py
:caption: pd_kmeans.py
:name: pd_kmeans.py
:lines: 2-4, 11-15
`scipy.spatial `_
for spatial queries like distances, nearest neighbours, k-Means :abbr:`etc
(et cetera)`.
Our ``find_labels`` method can then be written more specifically:
.. literalinclude:: sp_kmeans.py
:caption: sp_kmeans.py
:name: sp_kmeans.py
:lines: 6-9
`scipy.sparse `_
`sparse matrices `_
for 2-dimensional structured data.
`Sparse `_
for N-diemensional structured data.
`scipy.sparse.csgraph `_
for graph-like problems, for example searching for shortest paths.
`Xarray `_
for grouping across multiple dimensions.
.. toctree::
:hidden:
:titlesonly:
:maxdepth: 0
parallelise-pandas
Select compiler
---------------
Faster CPython
~~~~~~~~~~~~~~
At PyCon US in May 2021, Guido van Rossum presented `Faster CPython
`_, a project that aims to double the speed
of Python 3.11. The cooperation with the other Python core developers is
regulated in :pep:`PEP 659 – Specializing Adaptive Interpreter <659>`. There is
also an open `issue tracker `_
and various `tools for collecting bytecode statistics
`_. CPU-intensive Python code in
particular is likely to benefit from the changes; code already written in C,
I/O-heavy processes and multithreaded code, on the other hand, are unlikely to
benefit.
.. seealso::
* `Faster CPython `__
If you don’t want to wait with your project until the release of Python 3.11 in
the final version probably on 24 October 2022, you can also have a look at the
following Python interpreters:
Cython
~~~~~~
For intensive numerical operations, Python can be very slow, even if you have
avoided all anti-patterns and used vectorisations with NumPy. In this case,
translating code into `Cython `_ can be helpful.
Unfortunately, the code often has to be restructured and thus increases in
complexity. Explicit type annotations and the provision of code also become more
cumbersome.
Our example could then look like this:
.. literalinclude:: cy_kmeans.pyx
:caption: cy_kmeans.pyx
:name: cy_kmeans.pyx
:lines: 1-28
.. seealso::
* `Cython Tutorials
`_
Numba
~~~~~
`Numba `_ is a JIT compiler that translates mainly
scientific Python and NumPy code into fast machine code, for example:
.. literalinclude:: nb_kmeans.py
:caption: nb_kmeans.py
:name: nb_kmeans.py
:lines: 1-25
However, Numba requires `LLVM `_ and some
Python constructs are not supported.
Task planner
------------
:doc:`jupyter-tutorial:hub/ipyparallel/index`, :doc:`dask` and `Ray
`_ can distribute tasks in a cluster. In doing
so, they have different focuses:
* ``ipyparallel`` simply integrates into a :doc:`jupyter-tutorial:hub/index`.
* :doc:`dask` imitates pandas, NumPy iterators, Toolz und PySpark when it
distributes their tasks.
* Ray provides a simple, universal API for building distributed applications.
* `RLlib `_ will scale
reinforcement learning in particular.
* A `backend for joblib
`_ supports
distributed `scikit-learn `_ programs.
* `XGBoost-Ray
`_
is a backend for distributed `XGBoost
`_.
* `LightGBM-Ray
`_ is a
backend for distributed `LightGBM
`_.
* `Collective Communication Lib
`_ offers a
set of native collective primitives for `Gloo
`_ and the `NVIDIA Collective
Communication Library (NCCL)
`_.
Our example could look like this with Dask:
.. literalinclude:: ds_kmeans.py
:caption: ds_kmeans.py
:name: ds_kmeans.py
:lines: 1-
.. toctree::
:hidden:
:titlesonly:
:maxdepth: 0
dask.ipynb
Multithreading, Multiprocessing and Async
-----------------------------------------
After a brief :doc:`overview `, three examples
of :doc:`threading `, :doc:`multiprocessing
` and :doc:`async ` illustrate the rules and
best practices.
.. toctree::
:hidden:
:titlesonly:
:maxdepth: 0
multiprocessing-threading-async
threading-example.ipynb
multiprocessing.ipynb
threading-forking-combined.ipynb
asyncio-example.ipynb