How do I set, clear and toggle a bit in Rust?
Asked
Active
Viewed 2.2k times
33
-
2Are you looking for how to twiddle bits? – Matthieu M. Nov 07 '16 at 14:49
-
does Rust have bitwise operators? – phuclv Nov 07 '16 at 14:51
-
5Haven't you tried `&`, `|` and `^`? They're all available in Rust. – E_net4 - Krabbe mit Hüten Nov 07 '16 at 14:56
-
@LưuVĩnhPhúc: It has. – Matthieu M. Nov 07 '16 at 14:57
-
Then why don't just use what's in the other question? – phuclv Nov 07 '16 at 15:01
2 Answers
53
Like many other languages, the bitwise operators & (bitwise AND), | (bitwise OR), ^ (bitwise XOR) exist:
fn main() {
let mut byte: u8 = 0b0000_0000;
byte |= 0b0000_1000; // Set a bit
println!("0b{:08b}", byte);
byte &= 0b1111_0111; // Unset a bit
println!("0b{:08b}", byte);
byte ^= 0b0000_1000; // Toggle a bit
println!("0b{:08b}", byte);
}
The main difference from other languages is in bitwise NOT, which uses ! instead of ~:
fn main() {
let mut byte: u8 = 0b0000_0000;
byte = !byte; // Flip all bits
println!("0b{:08b}", byte);
}
You can also shift bits left or right:
fn main() {
let mut byte: u8 = 0b0000_1000;
byte <<= 1; // shift left one bit
println!("0b{:08b}", byte);
byte >>= 1; // shift right one bit
println!("0b{:08b}", byte);
}
There are many other conceptual things that ultimately do bit-level manipulation that are not expressed with operators. Check out the documentation for an integer for examples. One interesting example is leading_zeros. Here is how to rotate by a certain number of bits:
fn main() {
let mut byte: u8 = 0b1000_0000;
byte = byte.rotate_left(1); // rotate left one bit
println!("0b{:08b}", byte);
byte = byte.rotate_right(1); // rotate right one bit
println!("0b{:08b}", byte);
}
Shepmaster
- 326,504
- 69
- 892
- 1,159
-
1Interesting bit about `!` versus `~`. I would probably have tried the latter first. – Xavier T. Nov 08 '16 at 14:57
-
2@XavierT. yeah, it's definitely different, but IMO it does make more sense in the end. It is just a negation, after all. – Shepmaster Nov 08 '16 at 15:06
-
1fyi, the notation `"{:#08b}"` is designed to print the '0b' (or '0x', etc) before the binary number – Djzin Nov 08 '16 at 21:28
-
1@Djzin TIL! However, the zero-padding doesn't feel "right"; you actually have to use `{:#010b}` because it's an overall width :-( – Shepmaster Nov 08 '16 at 21:31
12
Rust has both bit-twiddling operators and binary format printing (very helpful for debugging):
fn bit_twiddling(original: u8, bit: u8) {
let mask = 1 << bit;
println!(
"Original: {:b}, Set: {:b}, Cleared: {:b}, Toggled: {:b}",
original,
original | mask,
original & !mask,
original ^ mask
);
}
fn main() {
bit_twiddling(0, 3);
bit_twiddling(8, 3);
}
It also has the compound assignment variants (|=, &= and ^=).
zzeroo
- 5,300
- 4
- 32
- 46
Matthieu M.
- 268,556
- 42
- 412
- 681