Bitwise Operators

Download as pdf or txt
Download as pdf or txt
You are on page 1of 7

Arduino Bitwise operators

Bitwise Operators
& (bitwise and)
<< (bitshift left)
>> (bitshift right)
^ (bitwise xor)
| (bitwise or)
~ (bitwise not)

&
[Bitwise Operators]

Description

The bitwise AND operator in C++ is a single ampersand &, used between
two other integer expressions. Bitwise AND operates on each bit position of
the surrounding expressions independently, according to this rule: if both
input bits are 1, the resulting output is 1, otherwise the output is 0.

Another way of expressing this is:

0 0 1 1 operand1
0 1 0 1 operand2
----------
0 0 0 1 (operand1 & operand2) - returned
result
In Arduino, the type int is a 16-bit value, so using & between two int
expressions causes 16 simultaneous AND operations to occur.

Example Code

In a code fragment like:

int a = 92; // in binary: 0000000001011100


int b = 101; // in binary: 0000000001100101
int c = a & b; // result: 0000000001000100, or 68 in
decimal.

Each of the 16 bits in a and b are processed by using the bitwise AND, and
all 16 resulting bits are stored in c, resulting in the value 01000100 in
binary, which is 68 in decimal.

One of the most common uses of bitwise AND is to select a particular bit
(or bits) from an integer value, often called masking. See below for an
example (AVR architecture specific).

PORTD = PORTD & 0b00000011; // clear out bits 2 - 7, leave pins


PD0 and PD1 untouched (xx & 11 == xx)

<<
[Bitwise Operators]

Description

The left shift operator << causes the bits of the left operand to be
shifted left by the number of positions specified by the right operand.

Syntax

variable << number_of_bits;

Parameters

variable: Allowed data types: byte, int, long.


number_of_bits: a number that is < = 32. Allowed data types: int.

Example Code
int a = 5; // binary: 0000000000000101
int b = a << 3; // binary: 0000000000101000, or 40 in decimal

Notes and Warnings

When you shift a value x by y bits (x << y), the leftmost y bits in x are
lost, literally shifted out of existence:
int x = 5; // binary: 0000000000000101
int y = 14;
int result = x << y; // binary: 0100000000000000 - the first 1
in 101 was discarded

If you are certain that none of the ones in a value are being shifted into
oblivion, a simple way to think of the left-shift operator is that it multiplies
the left operand by 2 raised to the right operand power. For example, to
generate powers of 2, the following expressions can be employed:

Operation Result
--------- ------
1 << 0 1
1 << 1 2
1 << 2 4
1 << 3 8
...
1 << 8 256
1 << 9 512
1 << 10 1024
...

The following example can be used to print out the value of a received byte
to the serial monitor, using the left shift operator to move along the byte
from bottom(LSB) to top (MSB), and print out its Binary value:

// Prints out Binary value (1 or 0) of byte


void printOut1(int c) {
for (int bits = 7; bits > -1; bits--) {
// Compare bits 7-0 in byte
if (c & (1 << bits)) {
Serial.print("1");
}
else {
Serial.print("0");
}
}
}

>>
[Bitwise Operators]
Description

The right shift operator >> causes the bits of the left operand to be
shifted right by the number of positions specified by the right operand.

Syntax

variable >> number_of_bits;

Parameters

variable: Allowed data types: byte, int, long.


number_of_bits: a number that is < = 32. Allowed data types: int.

Example Code
int a = 40; // binary: 0000000000101000
int b = a >> 3; // binary: 0000000000000101, or 5 in decimal

Notes and Warnings

When you shift x right by y bits (x >> y), and the highest bit in x is a 1,
the behavior depends on the exact data type of x. If x is of type int, the
highest bit is the sign bit, determining whether x is negative or not, as we
have discussed above. In that case, the sign bit is copied into lower bits,
for esoteric historical reasons:

int x = -16; // binary: 1111111111110000


int y = 3;
int result = x >> y; // binary: 1111111111111110

This behavior, called sign extension, is often not the behavior you want.
Instead, you may wish zeros to be shifted in from the left. It turns out that
the right shift rules are different for unsigned int expressions, so you can
use a typecast to suppress ones being copied from the left:

int x = -16; // binary: 1111111111110000


int y = 3;
int result = (unsigned int)x >> y; // binary: 0001111111111110
If you are careful to avoid sign extension, you can use the right-shift
operator >> as a way to divide by powers of 2. For example:

int x = 1000;
int y = x >> 3; // integer division of 1000 by 8, causing y =
125.

^
[Bitwise Operators]

Description

There is a somewhat unusual operator in C++ called bitwise EXCLUSIVE


OR, also known as bitwise XOR. (In English this is usually pronounced "eks-
or".) The bitwise XOR operator is written using the caret symbol ^. A
bitwise XOR operation results in a 1 only if the input bits are different, else
it results in a 0.

Precisely,

0 0 1 1 operand1
0 1 0 1 operand2
----------
0 1 1 0 (operand1 ^ operand2) - returned result

Example Code
int x = 12; // binary: 1100
int y = 10; // binary: 1010
int z = x ^ y; // binary: 0110, or decimal 6

The ^ operator is often used to toggle (i.e. change from 0 to 1, or 1 to 0)


some of the bits in an integer expression. In a bitwise XOR operation if
there is a 1 in the mask bit, that bit is inverted; if there is a 0, the bit is not
inverted and stays the same.

// Note: This code uses registers specific to AVR


microcontrollers (Uno, Nano, Leonardo, Mega, etc.)
// it will not compile for other architectures
void setup() {
DDRB = DDRB | 0b00100000; // set PB5 (pin 13 on Uno/Nano, pin
9 on Leonardo/Micro, pin 11 on Mega) as OUTPUT
Serial.begin(9600);
}

void loop() {
PORTB = PORTB ^ 0b00100000; // invert PB5, leave others
untouched
delay(100);
}

|
[Bitwise Operators]

Description

The bitwise OR operator in C++ is the vertical bar symbol, |. Like the &
operator, | operates independently each bit in its two surrounding integer
expressions, but what it does is different (of course). The bitwise OR of two
bits is 1 if either or both of the input bits is 1, otherwise it is 0.

In other words:

0 0 1 1 operand1
0 1 0 1 operand2
----------
0 1 1 1 (operand1 | operand2) - returned result

Example Code
int a = 92; // in binary: 0000000001011100
int b = 101; // in binary: 0000000001100101
int c = a | b; // result: 0000000001111101, or 125 in
decimal.

One of the most common uses of the Bitwise OR is to set multiple bits in a
bit-packed number.

// Note: This code is AVR architecture specific


// set direction bits for pins 2 to 7, leave PD0 and PD1
untouched (xx | 00 == xx)
// same as pinMode(pin, OUTPUT) for pins 2 to 7 on Uno or Nano
DDRD = DDRD | 0b11111100
~
[Bitwise Operators]

Description

The bitwise NOT operator in C++ is the tilde character ~. Unlike & and |,
the bitwise NOT operator is applied to a single operand to its right. Bitwise
NOT changes each bit to its opposite: 0 becomes 1, and 1 becomes 0.

In other words:

0 1 operand1
-----
1 0 ~operand1

Example Code
int a = 103; // binary: 0000000001100111
int b = ~a; // binary: 1111111110011000 = -104

Notes and Warnings

You might be surprised to see a negative number like -104 as the result of
this operation. This is because the highest bit in an int variable is the so-
called sign bit. If the highest bit is 1, the number is interpreted as
negative. This encoding of positive and negative numbers is referred to as
two’s complement. For more information, see the Wikipedia article on two’s
complement.

As an aside, it is interesting to note that for any integer x, ~x is the same


as -x - 1.

At times, the sign bit in a signed integer expression can cause some
unwanted surprises.

You might also like