basicbenchmark is a simple benchmarking tool for timing Python callables.
It's so basic, that probably you don't need it. But if it saves you a moment, here it is.
It runs a callable a number of times and returns the average time it took. Optionally, you can also get some basic statistics: fastest and slowest run time, and the standard deviation.
No dependencies, just a simple wrapper around Python standard library timeit.Timer
and time.perf_counter
.
pip install basicbenchmark
The package provides a decorator function, @basicbenchmark
, that can be used to time functions each time they are called.
In addition, there are two functions that can be called directly: benchmark
and benchmark_stats
.
These are useful for timing functions, keeping timing benchmarks separate from source code and storing the results in variables.
Add the @basicbenchmark
decorator to a function to time the execution and write the results to the console.
from basicbenchmark import basicbenchmark
@basicbenchmark
def my_function(x, y=2):
return x**y
result = my_function(x=2) # result is 4
my_function: 1 runs, mean time per run: 2.50 µs.
You can specify the number of runs to perform by passing the n_runs
argument to the decorator.
It will in addition print the fastest and slowest run times, along with the standard deviation.
@basicbenchmark(n_runs=100_000)
def my_function(x, y=2):
return x**y
result = my_function(x=2) # result is 4
my_function: 100,000 runs, mean time per run: 385.44±700.90 ns.
Fastest run: 200.00 ns. Slowest run: 123700.00 ns.
You can also pass the pre_run=True
argument to the decorator, which will run the function once before starting the benchmark.
This is useful for correctly timing JIT compiled functions, for example.
The two functions that can be called directly, benchmark
and benchmark_stats
, are used to time a callable and return the timing results.
The benchmark
function takes a callable, optional arguments and keyword arguments, and runs it a number of times, returning only the average time it took to run the callable in seconds.
It supports auto ranging the number of runs to get an accurate result, as it is a wrapper around timeit.Timer
, or you can specify the number of runs yourself with the n_runs
argument.
The benchmark_stats
function takes the same input arguments as benchmark
, but instead return a dictionary with timing results and the function return value.
The timing results in the dictionary include the average time, the standard deviation, the fastest and the slowest time in seconds.
This is useful if you want to know more about the distribution of the times it took to run the callable, and not just the average. In some cases the fastest time is of more interest than the average.
Both functions print the results to the console in more readable time units, but this can be disabled by setting the print_result
argument to False
.
Both functions also support a pre_run
argument, which runs the callable once before performing the benchmark. Useful for example when calling JIT compiled functions.
from basicbenchmark import benchmark, benchmark_stats
def my_function(x, y=2):
return x**y
The most basic way to time a function is to use the benchmark
function, which uses auto ranging to decide the number
of runs required (if argument n_runs
is not passed), and allows for passing arguments and keyword arguments to the callable.
The example below times the function my_function
with x=2
and y=2
, and returns the average time to avg_time
:
avg_time = benchmark(my_function, args=(2,))
Result is printed to the console
my_function: 5,000,000 runs, mean time per run: 72.37 ns.
To also get some basic statistics, use the benchmark_stats
function. This function does not support automatically determining the number of runs, so you need to specify the number of runs yourself.
The example below uses the benchmark_stats
with keyword arguments to pass to the callable:
benchmark_result = benchmark_stats(my_function, kwargs={'x': 100, 'y': 3}, n_runs=100_000)
It returns the following dictionary:
benchmark_result
{'return_value': 1000000,
'mean': 2.2017200006757777e-07,
'stdev': 1.883448346591418e-05,
'min': 9.999985195463523e-08,
'max': 0.005955899999662506}
And prints to console:
my_function: 100,000 runs, mean time per run: 220.17±18834.48 ns.
Fastest run: 100.00 ns. Slowest run: 5955900.00 ns.