今天,咱们来说下Go语言的算术运算符、比较运算符和逻辑运算符。学习
+ sum integers, floats, complex values (复数), strings
- difference integers, floats, complex values
* product integers, floats, complex values
/ quotient integers, floats, complex values
% remainder integers
位运算符:
& bitwise AND (按位与) integers
| bitwise OR (按位或) integers
^ bitwise XOR (按位异或) integers
&^ bit clear (AND NOT 按位置零) integers
移位运算符:
<< left shift integer << unsigned integer
>> right shift integer >> unsigned integer
复制代码
算术运算符+
、-
、*
、/
适用于整数、浮点数和复数,+
也适用于字符串,用于字符串链接;取模运算%
仅适用整数间的运算,Go语言中,%
取模运算符的结果的符号和被取模数的符号老是一致的,所以-5%3
和-5%-3
结果都是-2
。除法运算符/
的结果依赖于操做数是否为全为整数,好比5.0/4.0
的结果是1.25
,可是5/4
的结果是1
,由于整数除法会向着0方向截断余数;位运算和移位运算只适用整型。 位操做运算符^
,做为一元运算符,表示按位取反,就是说,它返回一个每一个bit位都取反的数:做用相似在C、C#、Java语言中中符号~
,对于有符号的整数来讲,是按照补码进行取反操做的(快速计算方法:对数a取反,结果为-(a+1)
);对于无符号整数来讲就是按位取反。ui
var a int8 = 3
var b uint8 = 3
var c int8 = -3
fmt.Printf("^%b=%b %d\n", a, ^a, ^a) // ^11=-100 -4
fmt.Printf("^%b=%b %d\n", b, ^b, ^b) // ^11=11111100 252
fmt.Printf("^%b=%b %d\n", c, ^c, ^c) // ^-11=10 2
复制代码
输出:spa
^11=-100 -4
^11=11111100 252
^-11=10 2
复制代码
做为二元运算符时是按位异或:对应位,相同为0,相异为1。.net
var a int8 = 3
var c int8 = 5
fmt.Printf("a: %08b\n",a)
fmt.Printf("c: %08b\n",c)
fmt.Printf("a^c: %08b\n",a ^ c)
复制代码
输出:code
a: 00000011
c: 00000101
a^c: 00000110
复制代码
加法和减法运算符:+
、-
也能够做为一元操做符: 对于整数,+x
是0+x
的简写,-x
则是0-x
的简写;对于浮点数和复数,+x
就是x
,-x
则是x
的负数。 须要给你们提一下的是这个操做符:按位置零&^
,表达式z = x &^ y
,若是 y 中的bit位为1,则 z 对应bit位为0,不然 z 对应bit位等于 x 中相应的bit位的值。不知道你们发现没有,咱们还能够这样理解或操做符|
,表达式z = x | y
,若是 y 中的bit位为1,则 z 对应bit位为1,不然 z 对应bit位等于 x 中相应的bit位的值,与&^
彻底相反。cdn
var x uint8 = 214
var y uint8 = 92
fmt.Printf("x: %08b\n",x)
fmt.Printf("y: %08b\n",y)
fmt.Printf("x|y: %08b\n",x | y)
fmt.Printf("x&^y: %08b\n",x &^ y)
复制代码
输出:blog
x: 11010110
y: 01011100
x|y: 11011110
x&^y: 10000010
复制代码
移位运算符x<<n
、x>>n
,若是左边的操做数是有符号整型,则是算术移位;若是左边的操做数是无符号整型,则是逻辑移位。逻辑移位不用考虑符号位,而算术移位须要考虑符号位,这样能保证移位操做和乘除的操做一致。对于x<<1
,至关于x*2
;x>>1
,至关于x/2
;左移算术移位结果会向下取整:rem
var a int8 = -51
fmt.Println("a:")
fmt.Printf("%08b\n",a)
fmt.Printf("%d: %08b\n", a>>1, a>>1)
var b int8 = 51
fmt.Println("b:")
fmt.Printf("%08b\n",b)
fmt.Printf("%d: %08b\n", b>>1, b>>1)
var i uint8 = 1
fmt.Println("i:")
fmt.Printf("%d: %08b\n", i<<1, i<<1) //2: 00000010
fmt.Printf("%d: %08b\n", i<<7, i<<7) //128: 10000000
fmt.Printf("%d: %b\n", i<<8, i<<8) //0: 0
var i2 int8 = 1
fmt.Println("i2:")
fmt.Printf("%d: %08b\n", i2<<1, i2<<1) //2: 00000010
fmt.Printf("%d: %08b\n", i2<<7, i2<<7) //-128: -10000000
fmt.Printf("%d: %b\n", i2<<8, i2<<8) //0: 0
var i3 int8 = -1
fmt.Println("i3:")
fmt.Printf("%d: %08b\n", -i3<<1, -i3<<1) //2: 00000010
fmt.Printf("%d: %08b\n", -i3<<7, -i3<<7) //-128: -10000000
fmt.Printf("%d: %b\n", -i3<<8, -i3<<8) //0: 0
var i4 int8 = -128
fmt.Println("i4:")
fmt.Printf("%d: %08b\n", -i4>>0, -i4>>0) //-128: -10000000
fmt.Printf("%d: %08b\n", -i4>>1, -i4>>1) //-64: -1000000
fmt.Printf("%d: %b\n", -i4>>2, -i4>>2) //-32: -100000
复制代码
输出:字符串
a:
-0110011
-26: -0011010
b:
00110011
25: 00011001
i:
2: 00000010
128: 10000000
0: 0
i2:
2: 00000010
-128: -10000000
0: 0
i3:
2: 00000010
-128: -10000000
0: 0
i4:
-128: -10000000
-64: -1000000
-32: -100000
复制代码
字符串链接可使用+
、+=
运算符:get
c := " Seekload"
s := "Hi" + string(c)
s += ", good bye"
复制代码
整型溢出: 有符号整数的+
、-
、*
、<<
的操做的结果的溢出不会致使异常,但结果可能不是你想要的,好比x < x+1
并不老是成立,好比:
var x int8 = 127;
fmt.Printf("%d\n",x+2) //输出 -127
复制代码
无符号整数的+、-、*、<<的操做的结果会取模2^n, 也就是溢出的位会被丢掉, 好比:
var x uint8 = 255;
fmt.Printf("%d\n",x+2) // 输出:1
复制代码
两个相同的整数类型可使用下面的二元比较运算符进行比较,比较表达式的结果是布尔类型。
符号 说明
&& 逻辑与 (p && q) p、q全为true,(p && q)才为true
|| 逻辑或 (p || q) p、q至少一个为true,(p || q)就为true
! 逻辑非 !p 若p为true,则!p为false;若p为false,则!p为true;
复制代码
一元运算符有最高的优先级。操做符++
、--
只能构成语句,而不是表达式,由于它们不属于运算符序列。 下面是关于算术运算、逻辑运算和比较运算的二元运算符,按照优先级递减的顺序排列:
优先级 操做符
5 * / % << >> & &^
4 + - | ^
3 == != < <= > >=
2 &&
1 ||
复制代码
从上面能够看出,二元运算符有五种优先级。在同一个优先级,使用左优先结合规则,可是使用括号能够明确优先顺序,使用括号也能够用于提高优先级,例如:mask & (1 << 28)
原创文章,若需转载请注明出处!
欢迎扫码关注公众号「Golang来啦」或者移步 seekload.net ,查看更多精彩文章。
公众号「Golang来啦」给你准备了一份神秘学习大礼包,后台回复【电子书】领取!