В разделе 2.6.2 приведен код попкорна:
// pc[i] is the population count of i.
var pc [256]byte
func init() {
for i := range pc {
pc[i] = pc[i/2] + byte(i&1)
}
}
// PopCount returns the population count (number of set bits) of x.
func PopCount(x uint64) int {
return int(pc[byte(x>>(0*8))] +
pc[byte(x>>(1*8))] +
pc[byte(x>>(2*8))] +
pc[byte(x>>(3*8))] +
pc[byte(x>>(4*8))] +
pc[byte(x>>(5*8))] +
pc[byte(x>>(6*8))] +
pc[byte(x>>(7*8))])
}
Это оптимизированная версия. init
создает справочную таблицу подсчета количества (количество битов, установленное в 1 в двоичном представлении) для каждого возможного байта, а PopCount
добавляет это для каждого байта из 8 байтов в uint64
.
Упражнения, следующие за этим разделом, просят вас реализовать различные варианты подсчета популярности и сравнить их производительность с этим фрагментом.
В частности, упражнение 2.4 спрашивает о переходе через 64-битные позиции. Для 64-битного типа int вы запускаете цикл, который повторяется 64 раза, в каждой итерации проверяется отдельный бит uint64
. Он считает, что все биты, которые он видит, установлены в 1. Вы должны реализовать это и сравнить производительность.