Bit Manipulation

Read this in other languages: français, 简体中文.

Get Bit

This method shifts the relevant bit to the zeroth position. Then we perform AND operation with one which has bit pattern like 0001. This clears all bits from the original number except the relevant one. If the relevant bit is one, the result is 1, otherwise the result is 0.

See getBit.js for further details.

Set Bit

This method shifts 1 over by bitPosition bits, creating a value that looks like 00100. Then we perform OR operation that sets specific bit into 1 but it does not affect on other bits of the number.

See setBit.js for further details.

Clear Bit

This method shifts 1 over by bitPosition bits, creating a value that looks like 00100. Than it inverts this mask to get the number that looks like 11011. Then AND operation is being applied to both the number and the mask. That operation unsets the bit.

See clearBit.js for further details.

Update Bit

This method is a combination of “Clear Bit” and “Set Bit” methods.

See updateBit.js for further details.

isEven

This method determines if the number provided is even. It is based on the fact that odd numbers have their last right bit to be set to 1.

  1. Number: 5 = 0b0101
  2. isEven: false
  3. Number: 4 = 0b0100
  4. isEven: true

See isEven.js for further details.

isPositive

This method determines if the number is positive. It is based on the fact that all positive numbers have their leftmost bit to be set to 0. However, if the number provided is zero or negative zero, it should still return false.

  1. Number: 1 = 0b0001
  2. isPositive: true
  3. Number: -1 = -0b0001
  4. isPositive: false

See isPositive.js for further details.

Multiply By Two

This method shifts original number by one bit to the left. Thus all binary number components (powers of two) are being multiplying by two and thus the number itself is being multiplied by two.

  1. Before the shift
  2. Number: 0b0101 = 5
  3. Powers of two: 0 + 2^2 + 0 + 2^0
  4. After the shift
  5. Number: 0b1010 = 10
  6. Powers of two: 2^3 + 0 + 2^1 + 0

See multiplyByTwo.js for further details.

Divide By Two

This method shifts original number by one bit to the right. Thus all binary number components (powers of two) are being divided by two and thus the number itself is being divided by two without remainder.

  1. Before the shift
  2. Number: 0b0101 = 5
  3. Powers of two: 0 + 2^2 + 0 + 2^0
  4. After the shift
  5. Number: 0b0010 = 2
  6. Powers of two: 0 + 0 + 2^1 + 0

See divideByTwo.js for further details.

Switch Sign

This method make positive numbers to be negative and backwards. To do so it uses “Twos Complement” approach which does it by inverting all of the bits of the number and adding 1 to it.

  1. 1101 -3
  2. 1110 -2
  3. 1111 -1
  4. 0000 0
  5. 0001 1
  6. 0010 2
  7. 0011 3

See switchSign.js for further details.

Multiply Two Signed Numbers

This method multiplies two signed integer numbers using bitwise operators. This method is based on the following facts:

  1. a * b can be written in the below formats:
  2. 0 if a is zero or b is zero or both a and b are zeroes
  3. 2a * (b/2) if b is even
  4. 2a * (b - 1)/2 + a if b is odd and positive
  5. 2a * (b + 1)/2 - a if b is odd and negative

The advantage of this approach is that in each recursive step one of the operands reduces to half its original value. Hence, the run time complexity is O(log(b)) where b is the operand that reduces to half on each recursive step.

See multiply.js for further details.

Multiply Two Unsigned Numbers

This method multiplies two integer numbers using bitwise operators. This method is based on that “Every number can be denoted as the sum of powers of 2”.

The main idea of bitwise multiplication is that every number may be split to the sum of powers of two:

I.e.

  1. 19 = 2^4 + 2^1 + 2^0

Then multiplying number x by 19 is equivalent of:

  1. x * 19 = x * 2^4 + x * 2^1 + x * 2^0

Now we need to remember that x * 2^4 is equivalent of shifting x left by 4 bits (x << 4).

See multiplyUnsigned.js for further details.

Count Set Bits

This method counts the number of set bits in a number using bitwise operators. The main idea is that we shift the number right by one bit at a time and check the result of & operation that is 1 if bit is set and 0 otherwise.

  1. Number: 5 = 0b0101
  2. Count of set bits = 2

See countSetBits.js for further details.

Count Bits to Flip One Number to Another

This methods outputs the number of bits required to convert one number to another. This makes use of property that when numbers are XOR-ed the result will be number of different bits.

  1. 5 = 0b0101
  2. 1 = 0b0001
  3. Count of Bits to be Flipped: 1

See bitsDiff.js for further details.

Count Bits of a Number

To calculate the number of valuable bits we need to shift 1 one bit left each time and see if shifted number is bigger than the input number.

  1. 5 = 0b0101
  2. Count of valuable bits is: 3
  3. When we shift 1 four times it will become bigger than 5.

See bitLength.js for further details.

Is Power of Two

This method checks if a number provided is power of two. It uses the following property. Let’s say that powerNumber is a number that has been formed as a power of two (i.e. 2, 4, 8, 16 etc.). Then if we’ll do & operation between powerNumber and powerNumber - 1 it will return 0 (in case if number is power of two).

  1. Number: 4 = 0b0100
  2. Number: 3 = (4 - 1) = 0b0011
  3. 4 & 3 = 0b0100 & 0b0011 = 0b0000 <-- Equal to zero, is power of two.
  4. Number: 10 = 0b01010
  5. Number: 9 = (10 - 1) = 0b01001
  6. 10 & 9 = 0b01010 & 0b01001 = 0b01000 <-- Not equal to zero, not a power of two.

See isPowerOfTwo.js for further details.

Full Adder

This method adds up two integer numbers using bitwise operators.

It implements full adder) electronics circuit logic to sum two 32-bit integers in two’s complement format. It’s using the boolean logic to cover all possible cases of adding two input bits: with and without a “carry bit” from adding the previous less-significant stage.

Legend:

  • A: Number A
  • B: Number B
  • ai: ith bit of number A
  • bi: ith bit of number B
  • carryIn: a bit carried in from the previous less-significant stage
  • carryOut: a bit to carry to the next most-significant stage
  • bitSum: The sum of ai, bi, and carryIn
  • resultBin: The full result of adding current stage with all less-significant stages (in binary)
  • resultDec: The full result of adding current stage with all less-significant stages (in decimal)
  1. A = 3: 011
  2. B = 6: 110
  3. ┌──────┬────┬────┬─────────┬──────────┬─────────┬───────────┬───────────┐
  4. bit ai bi carryIn carryOut bitSum resultBin resultDec
  5. ├──────┼────┼────┼─────────┼──────────┼─────────┼───────────┼───────────┤
  6. 0 1 0 0 0 1 1 1
  7. 1 1 1 0 1 0 01 1
  8. 2 0 1 1 1 0 001 1
  9. 3 0 0 1 0 1 1001 9
  10. └──────┴────┴────┴─────────┴──────────┴─────────┴───────────┴───────────┘

See fullAdder.js for further details. See Full Adder on YouTube.

References