Как обезвреживать биты (безмолвие?)
-
01-10-2019 - |
Вопрос
Какой самый эффективный способ отменить межслойные биты из 32 бита? Для этого конкретного случая я только обеспокоен странными битами, хотя я уверен, что это просто обобщить любое решение для обоих наборов.
Например, я хочу преобразовать 0b01000101
в 0b1011
. Отказ Какой самый быстрый путь?
РЕДАКТИРОВАТЬ:
В этом приложении я могу гарантировать, что даже биты все нули. Могу ли я воспользоваться этим фактом, чтобы улучшить скорость или уменьшить пространство?
Решение
Учитывая, что вы знаете, что каждый другой бит 0 в вашем приложении, вы можете сделать это так:
x = (x | (x >> 1)) & 0x33333333;
x = (x | (x >> 2)) & 0x0f0f0f0f;
x = (x | (x >> 4)) & 0x00ff00ff;
x = (x | (x >> 8)) & 0x0000ffff;
Первый шаг выглядит так:
0a0b0c0d0e0f0g0h0i0j0k0l0m0n0o0p x
| 00a0b0c0d0e0f0g0h0i0j0k0l0m0n0o0 x >> 1
--------------------------------
= 0aabbccddeeffgghhiijjkkllmmnnoop x | (x >> 1)
& 00110011001100110011001100110011 0x33333333
--------------------------------
= 00ab00cd00ef00gh00ij00kl00mn00op (x | (x >> 1)) & 0x33333333
Затем второй шаг работает с двумя битами за раз, и так далее.
Другие советы
С точки зрения скорости, настольная таблица 16 бит ширина с 2 ^ 32 записками будет трудно победить! Но если у вас нет столько памяти, чтобы посмотреть, четыре поиска в таблице 256-входа, плюс несколько смещений и и скрещены их вместе, могут быть лучшим выбором. Или, возможно, сладкое место где-то между ... Это зависит от имеющихся ресурсов, и как стоимость инициализации таблицы поиска будет амортизировать количество поисков, которые вам необходимо выполнить.
Я не уверен, как быстро Это было бы, но вы могли бы сделать что-то вроде
int a = 0b01000101;
int b = 0;
int i = 0;
while (a > 0) {
b |= (a & 1) << i;
a >>= 2;
}
Который бы потянул всех странных битов и положить их на б.