Даже не думал что мне придется столкнуться с чем-то подобным, но как говорится все бывает в первый раз. Дублирую как заметку статью с Хабра для нового проекта по lineage 2.
Побитовые операции пременяются для быстрого выполнения вычислений и меньшего потребления ресурсов, связанных с этими вычислениями.
В Java существуют следующие виды побитовых операций:
Так же стоит выделить операции битового сдвига:
Применяются к каждой паре битов, которые стоят в одинаковых позициях в двух битовых последовательностях.
Побитовое ИЛИ (OR)
Оператор обозначается символом |
Выставляет значение в 1, если установлен соответствующий бит в первой или во второй последовательности, или вместе
Побитовое И (AND)
Обозначается символом &
Выставляет значение в 1, если установлены соответствующие биты в первой и второй последовательности одновременно
ИСКЛЮЧАЮЩЕЕ ИЛИ (XOR)
Обозначается символом ^
Выставляет значение в 1, если установлен соответствующий бит или в первой или во второй во второй последовательности, но не одновременно.
Если используется более двух последовательностей, то в результате будет единица тогда, когда общее количество единиц соответствующей позиции нечетное.
ПОБИТОВОЕ ОТРИЦАНИЕ (NOT)
Обозначается символом ~
Унарный оператор, т.е. применяется к одной последовательности. Превращает каждый бит в противоположный.
Знаковый оператор сдвига влево <<
Все биты смещаются влево. Число справа дополняется нулем. Операция используется для быстрого умножения на 2. Если оператор применяется к числу, умножение на 2 которого будет больше максимального значения int (2147483647), то в результате будет отрицательное число. Это происходит потому, что краний левый бит, который отвечает за знак числа, выставляется в единицу, что соответствует отрицательным числам.
Знаковый оператор сдвига вправо >>
Все биты смещаются вправо. Число слева дополняется нулем, если число положительное и единицей, если отрицательное. Операция используется для быстрого деления на 2. Если делится нечетное число, то остаток отбрасывается для положительных чисел и сохраняется для отрицательных.
Беззнаковый оператор сдвига >>>
Все биты смещаются вправо, число слева дополняется нулем, даже если операция выполняется с отрицательными числами. Отсюда и название оператора — беззнаковый. В результате применения оператора всегда получается положительное число, т.к. в Java левый бит отвечает за знак числа. Операция так же, как и знаковый оператор сдвига вправо, соответствует делению числа на два за исключением первого сдвига в отрицительном числе.
Первое, что нам нужно понимать, что за последовательностью битов скрывается какая-то информация, которой мы можем управлять посредством битовых операций, например, это может быть список булевых значений или просто числа, которыми удобнее манипулировать в двоичном представлении
Когда количество сдвигов превышает количество разрядов
При использовании битовых сдвигов важно помнить, что когда количество сдвигов достигает количества разрядов, следующий сдвиг вернет значение в исходное положение. Например, сдвиг влево:
Так же имеет смысл заметить, что после 31-го сдвига нет позиции, где все нули.
Приведение чисел к соответствующему типу данных
При использовании побитовых операций с типами данных byte/short, числа сначала приводятся к типу int, а если одно из чисел — long, то к long.
При сужении типа данных, левая часть битов просто отбрасывается, например:
Использование маски
Одним из приемов работы с битовыми данными является использование маски. Маска позволяет получать значения только определенных битов в последовательности. Например, у нас есть маска 00100100, она позволяет нам получать из последовательности только те биты, которые в ней установлены. В данном случае это 3-й и 7-й разряд. Для этого достаточно выполнить побитовое И с нашей маской и неким числом:
Битовая маска используется, например, при определении маски подсети в сетевой адресации.
Хранение в одной целочисленной переменной нескольких значений
При помощи битовых сдвигов можно хранить в одной целочисленной переменной несколько значений меньшей длины. Например, в первых нескольких битах можно хранить одно число, в следующих битах — другое. Требуется только знать, на сколько бит выполняется сдвиг и сколько бит занимает хранимое число. Для записи используется логическое ИЛИ, для получения — И.
В следующем примере в одном числе сохраняются три значения — возраст, рост, вес, а затем считываются из него. Недостатком такой системы является необходимость помнить, что хранимые значения не должны превышать количество бит, которые определены для них. Например, если в примере одно из значений будет превышать число 255, то мы получим ошибочный результат.
Работа с правами доступа
Еще один рапространенный пример применения побитовых операций — работа с правами доступа к папкам и файлам. Принцип следующий. Имеется последовательность из трех битов, где:
001 — первый бит отвечает за права на выполнение
010 — второй — запись
100 — третий — чтение
Имеем следующие константы.
Допустим, нам требуется дать пользователю полный доступ к ресурсу. Для этого должен быть выставлен каждый бит:
А теперь, допустим, что нам надо забрать у пользователя права на выполнение:
Обмен переменных местами без использования временной переменной
Исключающее ИЛИ может быть использовано для обмена двух переменных без создания временной переменной:
Правда, в реальной жизни способ с временной переменной работает быстрее. Данный пример приведен только в познавательных целях.
Шифр Вернама
На основе иключающего ИЛИ работает шифр Вернама, для которого была доказана абсолютная криптографическая стойкость. Шифр был «взломан» в фильме «Пароль «Рыба-меч»»
Быстрое умножение и деление
Операции сдвига рекомендуют использовать для быстрого умножения и деления целых чисел на числа, равные степени двойки. Например, выражение 3 << 4 соответствует умножению тройки на 2 в 4-й степени.
Хранение списка булевых значений и манипуляция ими
Побитовое ИЛИ может быть использовано для установки в 1 необходимых позиций битов в последовательности. Это позволяет, например, хранить список булевых значений в максимально сжатом виде
С помощью исключающего ИЛИ можно переключать заданные биты на противоположные, что позволяет манипулировать набором булевых значений.
Как проверить, является ли число степенью двойки?
С помощью побитовых операций можно за константное время проверить, является ли число степенью двойки:
Данное выражение работает для всех чисел, кроме 0, а 0 не является степенью двойки. Поэтому конечный вариант должен выглядеть так:
Как это работает:
В числе 2 в степени n в единицу выставлен только (n+1)-й бит. Все остальные — нули. Например, числа 16 и 32 имеют двоичный вид, соответственно, 10000 и 100000. Т.е. нам необходимо проверить, что в числе выставлен только один бит.
Допустим, число k — два в степени n. В числе k в единицу выставлен (n+1)-й бит. Тогда в числе k-1 в единицу будут выставлены все биты от 1 до n. Если с такими числами выполнить побитовое И, то в результате должен получиться 0. Например:
Вывод простой. Программист должен знать побитовые операции и уметь ими пользоваться. Это в некоторых случаях может упростить ему жизнь.
Вот еще много примеров применения побитовых операций:
Bit Twiddling Hacks
The Aggregate Magic Algorithms
All The Twiddled Bits
Code for the Algorithms in Hacker's Delight book
Ну и домашнее задание напоследок. Почему в HashMap в Java для определения индекса используется логическое И?
Bitwise and Bit Shift Operators
Битовые операции в PHP на примерах
Алгоритм обмена при помощи исключающего ИЛИ
Битовые сдвиги и приведения в Java: подводные камни
Битовая маска
Bitwise operation
How to check if a number is a power of 2
Побитовые операции пременяются для быстрого выполнения вычислений и меньшего потребления ресурсов, связанных с этими вычислениями.
В Java существуют следующие виды побитовых операций:
| | ИЛИ (OR) |
& | И (AND) |
^ | ИСКЛЮЧАЮЩЕЕ ИЛИ (XOR) |
~ | ОТРИЦАНИЕ (NOT) |
Так же стоит выделить операции битового сдвига:
<< | сдвиг влево |
>> | сдвиг вправо |
>>> | беззнаковый сдвиг вправо |
Побитовые операции
Применяются к каждой паре битов, которые стоят в одинаковых позициях в двух битовых последовательностях.
Побитовое ИЛИ (OR)
Оператор обозначается символом |
Выставляет значение в 1, если установлен соответствующий бит в первой или во второй последовательности, или вместе
Побитовое И (AND)
Обозначается символом &
Выставляет значение в 1, если установлены соответствующие биты в первой и второй последовательности одновременно
ИСКЛЮЧАЮЩЕЕ ИЛИ (XOR)
Обозначается символом ^
Выставляет значение в 1, если установлен соответствующий бит или в первой или во второй во второй последовательности, но не одновременно.
Если используется более двух последовательностей, то в результате будет единица тогда, когда общее количество единиц соответствующей позиции нечетное.
ПОБИТОВОЕ ОТРИЦАНИЕ (NOT)
Обозначается символом ~
Унарный оператор, т.е. применяется к одной последовательности. Превращает каждый бит в противоположный.
Знаковый оператор сдвига влево <<
Все биты смещаются влево. Число справа дополняется нулем. Операция используется для быстрого умножения на 2. Если оператор применяется к числу, умножение на 2 которого будет больше максимального значения int (2147483647), то в результате будет отрицательное число. Это происходит потому, что краний левый бит, который отвечает за знак числа, выставляется в единицу, что соответствует отрицательным числам.
Знаковый оператор сдвига вправо >>
Все биты смещаются вправо. Число слева дополняется нулем, если число положительное и единицей, если отрицательное. Операция используется для быстрого деления на 2. Если делится нечетное число, то остаток отбрасывается для положительных чисел и сохраняется для отрицательных.
Беззнаковый оператор сдвига >>>
Все биты смещаются вправо, число слева дополняется нулем, даже если операция выполняется с отрицательными числами. Отсюда и название оператора — беззнаковый. В результате применения оператора всегда получается положительное число, т.к. в Java левый бит отвечает за знак числа. Операция так же, как и знаковый оператор сдвига вправо, соответствует делению числа на два за исключением первого сдвига в отрицительном числе.
Применение на практике
Первое, что нам нужно понимать, что за последовательностью битов скрывается какая-то информация, которой мы можем управлять посредством битовых операций, например, это может быть список булевых значений или просто числа, которыми удобнее манипулировать в двоичном представлении
Когда количество сдвигов превышает количество разрядов
При использовании битовых сдвигов важно помнить, что когда количество сдвигов достигает количества разрядов, следующий сдвиг вернет значение в исходное положение. Например, сдвиг влево:
Так же имеет смысл заметить, что после 31-го сдвига нет позиции, где все нули.
Приведение чисел к соответствующему типу данных
При использовании побитовых операций с типами данных byte/short, числа сначала приводятся к типу int, а если одно из чисел — long, то к long.
При сужении типа данных, левая часть битов просто отбрасывается, например:
Использование маски
Одним из приемов работы с битовыми данными является использование маски. Маска позволяет получать значения только определенных битов в последовательности. Например, у нас есть маска 00100100, она позволяет нам получать из последовательности только те биты, которые в ней установлены. В данном случае это 3-й и 7-й разряд. Для этого достаточно выполнить побитовое И с нашей маской и неким числом:
Битовая маска используется, например, при определении маски подсети в сетевой адресации.
Хранение в одной целочисленной переменной нескольких значений
При помощи битовых сдвигов можно хранить в одной целочисленной переменной несколько значений меньшей длины. Например, в первых нескольких битах можно хранить одно число, в следующих битах — другое. Требуется только знать, на сколько бит выполняется сдвиг и сколько бит занимает хранимое число. Для записи используется логическое ИЛИ, для получения — И.
В следующем примере в одном числе сохраняются три значения — возраст, рост, вес, а затем считываются из него. Недостатком такой системы является необходимость помнить, что хранимые значения не должны превышать количество бит, которые определены для них. Например, если в примере одно из значений будет превышать число 255, то мы получим ошибочный результат.
Работа с правами доступа
Еще один рапространенный пример применения побитовых операций — работа с правами доступа к папкам и файлам. Принцип следующий. Имеется последовательность из трех битов, где:
001 — первый бит отвечает за права на выполнение
010 — второй — запись
100 — третий — чтение
Имеем следующие константы.
Допустим, нам требуется дать пользователю полный доступ к ресурсу. Для этого должен быть выставлен каждый бит:
А теперь, допустим, что нам надо забрать у пользователя права на выполнение:
Обмен переменных местами без использования временной переменной
Исключающее ИЛИ может быть использовано для обмена двух переменных без создания временной переменной:
Правда, в реальной жизни способ с временной переменной работает быстрее. Данный пример приведен только в познавательных целях.
Шифр Вернама
На основе иключающего ИЛИ работает шифр Вернама, для которого была доказана абсолютная криптографическая стойкость. Шифр был «взломан» в фильме «Пароль «Рыба-меч»»
Быстрое умножение и деление
Операции сдвига рекомендуют использовать для быстрого умножения и деления целых чисел на числа, равные степени двойки. Например, выражение 3 << 4 соответствует умножению тройки на 2 в 4-й степени.
Хранение списка булевых значений и манипуляция ими
Побитовое ИЛИ может быть использовано для установки в 1 необходимых позиций битов в последовательности. Это позволяет, например, хранить список булевых значений в максимально сжатом виде
С помощью исключающего ИЛИ можно переключать заданные биты на противоположные, что позволяет манипулировать набором булевых значений.
Как проверить, является ли число степенью двойки?
С помощью побитовых операций можно за константное время проверить, является ли число степенью двойки:
Данное выражение работает для всех чисел, кроме 0, а 0 не является степенью двойки. Поэтому конечный вариант должен выглядеть так:
Как это работает:
В числе 2 в степени n в единицу выставлен только (n+1)-й бит. Все остальные — нули. Например, числа 16 и 32 имеют двоичный вид, соответственно, 10000 и 100000. Т.е. нам необходимо проверить, что в числе выставлен только один бит.
Допустим, число k — два в степени n. В числе k в единицу выставлен (n+1)-й бит. Тогда в числе k-1 в единицу будут выставлены все биты от 1 до n. Если с такими числами выполнить побитовое И, то в результате должен получиться 0. Например:
Вывод
Вывод простой. Программист должен знать побитовые операции и уметь ими пользоваться. Это в некоторых случаях может упростить ему жизнь.
Вот еще много примеров применения побитовых операций:
Bit Twiddling Hacks
The Aggregate Magic Algorithms
All The Twiddled Bits
Code for the Algorithms in Hacker's Delight book
Ну и домашнее задание напоследок. Почему в HashMap в Java для определения индекса используется логическое И?
Использованная литература:
Bitwise and Bit Shift Operators
Битовые операции в PHP на примерах
Алгоритм обмена при помощи исключающего ИЛИ
Битовые сдвиги и приведения в Java: подводные камни
Битовая маска
Bitwise operation
How to check if a number is a power of 2
Комментариев нет:
Отправить комментарий