2021-10-19 00:53:10 +08:00
|
|
|
def get_1s_count(number: int) -> int:
|
|
|
|
"""
|
|
|
|
Count the number of set bits in a 32 bit integer using Brian Kernighan's way.
|
2022-10-16 21:45:25 +08:00
|
|
|
Ref - https://graphics.stanford.edu/~seander/bithacks.html#CountBitsSetKernighan
|
2021-10-19 00:53:10 +08:00
|
|
|
>>> get_1s_count(25)
|
|
|
|
3
|
|
|
|
>>> get_1s_count(37)
|
|
|
|
3
|
|
|
|
>>> get_1s_count(21)
|
|
|
|
3
|
|
|
|
>>> get_1s_count(58)
|
|
|
|
4
|
|
|
|
>>> get_1s_count(0)
|
|
|
|
0
|
|
|
|
>>> get_1s_count(256)
|
|
|
|
1
|
|
|
|
>>> get_1s_count(-1)
|
|
|
|
Traceback (most recent call last):
|
|
|
|
...
|
2022-11-20 18:59:25 +08:00
|
|
|
ValueError: Input must be a non-negative integer
|
2021-10-19 00:53:10 +08:00
|
|
|
>>> get_1s_count(0.8)
|
|
|
|
Traceback (most recent call last):
|
|
|
|
...
|
2022-11-20 18:59:25 +08:00
|
|
|
ValueError: Input must be a non-negative integer
|
|
|
|
>>> get_1s_count("25")
|
|
|
|
Traceback (most recent call last):
|
|
|
|
...
|
|
|
|
ValueError: Input must be a non-negative integer
|
2021-10-19 00:53:10 +08:00
|
|
|
"""
|
2022-11-20 18:59:25 +08:00
|
|
|
if not isinstance(number, int) or number < 0:
|
|
|
|
raise ValueError("Input must be a non-negative integer")
|
|
|
|
|
2021-10-19 00:53:10 +08:00
|
|
|
count = 0
|
|
|
|
while number:
|
|
|
|
# This way we arrive at next set bit (next 1) instead of looping
|
|
|
|
# through each bit and checking for 1s hence the
|
|
|
|
# loop won't run 32 times it will only run the number of `1` times
|
|
|
|
number &= number - 1
|
|
|
|
count += 1
|
|
|
|
return count
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
import doctest
|
|
|
|
|
|
|
|
doctest.testmod()
|