# Chapter 5. Bit manipulation ## 5.1. Two's complement and negative numbers Positive numbers are represented as themselves, negative numbers as the two's complement of its absolute value, with a 1 in its sign bit, indicating it's negative. The two's complement of an N-bit number (N: \# bits used for the number, excluding the sign bit), is the complement of the number with respect to 2N. The binary representation of -K as a N-bit number is concat(1, 2N-1 - K). ## 5.2. Arithmetic vs. Logical right shift Arithmetic right shift: division by two. Logical right shift: 'shifting the bits'. ### 5.2.1. Logical shift Indicated with a >>> operator, all bits are shifted, sign bit too. * -75: **1**0110101 * 90: **0**1011010 ```c++ int repeatedLogicalShift(int x, int count) { for (int i = 0; i < count; i++) { x >>>= 1; // Logical shift by 1 } } ``` ### 5.2.2. Arithmetic shift The sign bit is kept, and all bits (sign bit too) are shifted. * -75: **1**0110101 * -38: **1**1011010 ```c++ int repeatedArithmeticShift(int x, int count) { for (int i = 0; i < count; i++) { x >>= 1; // Arithmetic shift by 1 } } ``` ## 5.3. Common bit tasks: getting and setting ### Get bit This method shifts 1 over by i bits, creating a value that looks like 00010000. By performing an AND with `num`, we clear all bits other than the bit at bit `i`. Then we compare it to 0. If the new value isn't 0, then bit i must have a 1. Otherwise, bit i is a 0. ```c++ boolean getBit(int num, int i) { return ((num & (1 << i)) != 0); } ``` ### Set bit This function shifts 1 over by i bits, creating a value like 00010000. By performing an OR with `num`, only the value at bit i will change. All other bits of the mask are zero, and won't affect `num`. It sets the bit to 1 if it's 0, otherwise leaves the number unchanged. ```c++ boolean setBit(int num, int i) { return num | (1 << i); } ``` ### Clear bit This is the opposite of `setBit`. We first create a number like 11101111. Then, we AND it with `num`. This clears the bit i and leaves the rest unchanged. ```c++ boolean clearBit(int num, int i) { return num & ~(1 << i); } ``` To clear all bits from the most significant bit through i (inclusive), we create a mask with a 1 at the ith bit (1 << i). Then we substract 1 from it, giving us a sequence of zeros followed by i ones. We AND `num` with this mask, to leave just the last i bits. ```c++ boolean clearBitMSBthroughI(int num, int i) { return num & ((1 << i) - 1); } ``` To clear all bits from i through 0 (inclusive), we take a sequence of all ones (which represents -1) and shift it left by i+1 bits. This gives a sequence of ones followed by i+1 zeros. ```c++ boolean clearBitIthrough0(int num, int i) { return num & (-1 << (i+1)); } ``` ### Update bit To set the ith bit to a value v, we first clear the bit at position i, then we shift the value v, left by i bits. This will create a number with bit i equal to v and all other bits equal to 0. Finally, we OR these 2 numbers, updating the ith bit if v=1 and leaving it as 0 otherwise. ```c++ boolean updateBit(int num, int i, boolean bitIs1) { int value = bitIs1 ? 1 : 0; int mask = ~(1 << i); return (num & mask) | (value << i); } ```