2022-10-24 18:13:39 +08:00
|
|
|
from timeit import timeit
|
|
|
|
|
|
|
|
|
|
|
|
def get_set_bits_count_using_brian_kernighans_algorithm(number: int) -> int:
|
2021-02-12 10:29:24 +08:00
|
|
|
"""
|
|
|
|
Count the number of set bits in a 32 bit integer
|
2022-10-24 18:13:39 +08:00
|
|
|
>>> get_set_bits_count_using_brian_kernighans_algorithm(25)
|
2021-02-12 10:29:24 +08:00
|
|
|
3
|
2022-10-24 18:13:39 +08:00
|
|
|
>>> get_set_bits_count_using_brian_kernighans_algorithm(37)
|
2021-02-12 10:29:24 +08:00
|
|
|
3
|
2022-10-24 18:13:39 +08:00
|
|
|
>>> get_set_bits_count_using_brian_kernighans_algorithm(21)
|
2021-02-12 10:29:24 +08:00
|
|
|
3
|
2022-10-24 18:13:39 +08:00
|
|
|
>>> get_set_bits_count_using_brian_kernighans_algorithm(58)
|
2021-02-12 10:29:24 +08:00
|
|
|
4
|
2022-10-24 18:13:39 +08:00
|
|
|
>>> get_set_bits_count_using_brian_kernighans_algorithm(0)
|
2021-02-12 10:29:24 +08:00
|
|
|
0
|
2022-10-24 18:13:39 +08:00
|
|
|
>>> get_set_bits_count_using_brian_kernighans_algorithm(256)
|
2021-02-12 10:29:24 +08:00
|
|
|
1
|
2022-10-24 18:13:39 +08:00
|
|
|
>>> get_set_bits_count_using_brian_kernighans_algorithm(-1)
|
2021-02-12 10:29:24 +08:00
|
|
|
Traceback (most recent call last):
|
|
|
|
...
|
2022-10-24 18:13:39 +08:00
|
|
|
ValueError: the value of input must not be negative
|
2021-02-12 10:29:24 +08:00
|
|
|
"""
|
|
|
|
if number < 0:
|
2022-10-24 18:13:39 +08:00
|
|
|
raise ValueError("the value of input must not be negative")
|
|
|
|
result = 0
|
|
|
|
while number:
|
|
|
|
number &= number - 1
|
|
|
|
result += 1
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
|
|
|
def get_set_bits_count_using_modulo_operator(number: int) -> int:
|
|
|
|
"""
|
|
|
|
Count the number of set bits in a 32 bit integer
|
|
|
|
>>> get_set_bits_count_using_modulo_operator(25)
|
|
|
|
3
|
|
|
|
>>> get_set_bits_count_using_modulo_operator(37)
|
|
|
|
3
|
|
|
|
>>> get_set_bits_count_using_modulo_operator(21)
|
|
|
|
3
|
|
|
|
>>> get_set_bits_count_using_modulo_operator(58)
|
|
|
|
4
|
|
|
|
>>> get_set_bits_count_using_modulo_operator(0)
|
|
|
|
0
|
|
|
|
>>> get_set_bits_count_using_modulo_operator(256)
|
|
|
|
1
|
|
|
|
>>> get_set_bits_count_using_modulo_operator(-1)
|
|
|
|
Traceback (most recent call last):
|
|
|
|
...
|
|
|
|
ValueError: the value of input must not be negative
|
|
|
|
"""
|
|
|
|
if number < 0:
|
|
|
|
raise ValueError("the value of input must not be negative")
|
2021-02-12 10:29:24 +08:00
|
|
|
result = 0
|
|
|
|
while number:
|
|
|
|
if number % 2 == 1:
|
|
|
|
result += 1
|
2022-10-24 18:13:39 +08:00
|
|
|
number >>= 1
|
2021-02-12 10:29:24 +08:00
|
|
|
return result
|
|
|
|
|
|
|
|
|
2022-10-24 18:13:39 +08:00
|
|
|
def benchmark() -> None:
|
|
|
|
"""
|
|
|
|
Benchmark code for comparing 2 functions, with different length int values.
|
|
|
|
Brian Kernighan's algorithm is consistently faster than using modulo_operator.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def do_benchmark(number: int) -> None:
|
|
|
|
setup = "import __main__ as z"
|
|
|
|
print(f"Benchmark when {number = }:")
|
|
|
|
print(f"{get_set_bits_count_using_modulo_operator(number) = }")
|
|
|
|
timing = timeit("z.get_set_bits_count_using_modulo_operator(25)", setup=setup)
|
|
|
|
print(f"timeit() runs in {timing} seconds")
|
|
|
|
print(f"{get_set_bits_count_using_brian_kernighans_algorithm(number) = }")
|
|
|
|
timing = timeit(
|
|
|
|
"z.get_set_bits_count_using_brian_kernighans_algorithm(25)",
|
|
|
|
setup=setup,
|
|
|
|
)
|
|
|
|
print(f"timeit() runs in {timing} seconds")
|
|
|
|
|
|
|
|
for number in (25, 37, 58, 0):
|
|
|
|
do_benchmark(number)
|
|
|
|
print()
|
|
|
|
|
|
|
|
|
2021-02-12 10:29:24 +08:00
|
|
|
if __name__ == "__main__":
|
|
|
|
import doctest
|
|
|
|
|
|
|
|
doctest.testmod()
|
2022-10-24 18:13:39 +08:00
|
|
|
benchmark()
|