你如何在Go中设置和清除一个位?

Kev*_*rke 15 bit-manipulation go

在Golang中,如何设置和清除整数的各个位?例如,行为如下:

 clearBit(129, 7) // returns 1
 setBit(1, 7)     // returns 129
Run Code Online (Sandbox Code Playgroud)

Kev*_*rke 31

这是一个设置位的功能.首先,将数字1移动整数中指定的空格数(因此它变为0010,0100等).然后用原始输入OR.这使得其他位不受影响,但始终将目标位设置为1.

// Sets the bit at pos in the integer n.
func setBit(n int, pos uint) int {
    n |= (1 << pos)
    return n
}
Run Code Online (Sandbox Code Playgroud)

这是一个清除一点的功能.首先将数字1移位整数中指定的空格数(因此它变为0010,0100等).然后用^操作员翻转掩码中的每一位(因此0010变为1101).然后使用按位AND,它不会触及数字AND'ed with 1,但会取消设置为0的掩码中的值.

// Clears the bit at pos in n.
func clearBit(n int, pos uint) int {
    mask := ^(1 << pos)
    n &= mask
    return n
}
Run Code Online (Sandbox Code Playgroud)

最后,这是一个检查是否设置了一个位的函数.将数字1移动指定的空格数(因此它变为0010,0100等),然后将其与目标数字进行AND运算.如果结果数大于0(它将是1,2,4,8等),则设置该位.

func hasBit(n int, pos uint) bool {
    val := n & (1 << pos)
    return (val > 0)
}
Run Code Online (Sandbox Code Playgroud)

  • 请注意,只写 `x &amp;= ^(1&lt;&lt;7)` 或 `x |= 1&lt;&lt;7`(或 `x ^= 1&lt;&lt;7` 来切换,或 `(x &amp; 1 &lt;&lt;7 == 0)` 以查看是否设置了位)。这已经被广泛理解,不需要评论,但是如果您觉得应该解释一下(例如,您的团队不是来自一个乱七八糟的世界),只需评论您首次使用每个习语的意图。此外,人们经常定义表示位或位组合的常量,这样他们就不必每次都写出 `1&lt;&lt;n` 并记住正确的 `n`,以及 [`iota`](http://golang.org /ref/spec#Iota) 可以帮助使这些常量声明简洁。 (2认同)

wef*_*fa3 15

还有一个紧凑的符号可以清除一点.该操作员被&^称为"而不是".

使用此运算符clearBit可以像这样编写函数:

// Clears the bit at pos in n.
func clearBit(n int, pos uint) int {
    n &^= (1 << pos)
    return n
}
Run Code Online (Sandbox Code Playgroud)

或者像这样:

// Clears the bit at pos in n.
func clearBit(n int, pos uint) int {
    return n &^ (1 << pos)
}
Run Code Online (Sandbox Code Playgroud)