运算符用于在程序运行时执行数学或逻辑运算,Go 语言内置的运算符有:
- 算术运算符
- 赋值运算符
- 关系运算符
- 逻辑运算符
- 位运算符
- 其他运算符
算术运算符
下表列出了所有Go语言的算术运算符,假定 A 值为 10,B 值为 20
| 运算符 | 描述 | 实例 | 运算结果 |
|---|---|---|---|
| + | 相加 | A + B | 30 |
| - | 相减 | A - B | -10 |
| * | 相乘 | A * B | 200 |
| / | 相除 | A / B | 0 |
| % | 求余 | A % B | 10 |
| ++ | 自增 | A++ | 11 |
| — | 自减 | B— | 19 |
package mainimport "fmt"func main() {var a int = 10var b int = 20var c intc = a + bfmt.Printf("第一行: c的值为 %d\n", c ) //30c = a - bfmt.Printf("第二行: c的值为 %d\n", c ) //-10c = a * bfmt.Printf("第三行: c的值为 %d\n", c ) //200c = a / bfmt.Printf("第四行: c的值为 %d\n", c ) //0c = a % bfmt.Printf("第五行: c的值为 %d\n", c ) //10a++fmt.Printf("第六行: a的值为 %d\n", a ) //11b=20b--fmt.Printf("第七行: b的值为 %d\n", b ) //19}
Go 的自增,自减只能作为表达式使用,而不能用于赋值语句
package mainimport "fmt"func main() {var a int =10a++ // 这是允许的,类似 a = a + 1a--a = a++ // 这是不允许的,会出现变异错误 syntax error: unexpected ++ at end of statementfmt.Println(a)}
赋值运算符
下表列出了所有Go语言的赋值运算符
| 运算符 | 描述 | 实例 | 等价与 |
|---|---|---|---|
| = | 简单的赋值运算符 将一个表达式的值赋给一个左值 |
C = A + B | 将 A + B 表达式结果赋值给 C |
| := | 海象运算符 |
||
| += | 相加后再赋值 | C += A | C = C + A |
| -= | 相减后再赋值 | C -= A | C = C - A |
| *= | 相乘后再赋值 | C *= A | C = C * A |
| /= | 相除后再赋值 | C /= A | C = C / A |
| %= | 求余后再赋值 | C %= A | C = C % A |
| <<= | 左移后赋值 | C <<= 2 | C = C << 2 |
| >>= | 右移后赋值 | C >>= 2 | C = C >> 2 |
| &= | 按位与后赋值 | C &= 2 | C = C & 2 |
| ^= | 按位异或后赋值 | C ^= 2 | C = C ^ 2 |
| |= | 按位或后赋值 | C |= 2 | C = C | 2 |
package mainimport "fmt"func main() {var a int = 21var c intc = afmt.Printf("第1行: = 运算符实例,c 值为 = %d\n", c )c += afmt.Printf("第2行: += 运算符实例,c 值为 = %d\n", c )c -= afmt.Printf("第3行: -= 运算符实例,c 值为 = %d\n", c )c *= afmt.Printf("第4行: *= 运算符实例,c 值为 = %d\n", c )c /= afmt.Printf("第5行: /= 运算符实例,c 值为 = %d\n", c )c = 200;c <<= 2fmt.Printf("第6行: <<= 运算符实例,c 值为 = %d\n", c )c >>= 2fmt.Printf("第7行: >>= 运算符实例,c 值为 = %d\n", c )c &= 2fmt.Printf("第8行: &= 运算符实例,c 值为 = %d\n", c )c ^= 2fmt.Printf("第9行: ^= 运算符实例,c 值为 = %d\n", c )c |= 2fmt.Printf("第0行: = 运算符实例,c 值为 = %d\n", c )}/*第1行: = 运算符实例,c 值为 = 21第2行: += 运算符实例,c 值为 = 42第3行: -= 运算符实例,c 值为 = 21第4行: *= 运算符实例,c 值为 = 441第5行: /= 运算符实例,c 值为 = 21第6行: <<= 运算符实例,c 值为 = 800第7行: >>= 运算符实例,c 值为 = 200第8行: &= 运算符实例,c 值为 = 0第9行: ^= 运算符实例,c 值为 = 2第0行: = 运算符实例,c 值为 = 2*/
关系运算符
下表列出了所有Go语言的关系运算符。假定 A 值为 10,B 值为 20
| 运算符 | 描述 | 实例 | 运行结果 |
|---|---|---|---|
==
| 检查两个值是否相等
如果相等返回 True 否则返回 False | A == B | False |
|
!=
| 检查两个值是否不相等
如果不相等返回 True 否则返回 False | A != B | True |
|
>
| 检查左边值是否大于右边值
如果是返回 True 否则返回 False | A > B | False |
|
<
| 检查左边值是否小于右边值
如果是返回 True 否则返回 False | A < B | True | |
>=
| 检查左边值是否大于等于右边值
如果是返回 True 否则返回 False | A >= B | False |
|
<=
| 检查左边值是否小于等于右边值
如果是返回 True 否则返回 False | A <= B | True |
| ~~ ? : ~~ | Go语言中无 三元表达式 | | |
package mainimport "fmt"func main() {var a int = 10var b int = 20if( a == b ) {fmt.Printf("第一行: a等于b\n" )} else {fmt.Printf("第一行: a不等于b\n" )}if ( a < b ) {fmt.Printf("第二行: a小于b\n" )} else {fmt.Printf("第二行: a不小于b\n" )}if ( a > b ) {fmt.Printf("第三行: a大于b\n" )} else {fmt.Printf("第三行: a不大于b\n" )}if ( a <= b ) {fmt.Printf("第四行: a小于等于b\n" )}if ( b >= a ) {fmt.Printf("第五行: b大于等于a\n" )}}/*第一行: a不等于b第二行: a小于b第三行: a不大于b第四行: a小于等于b第五行: b大于等于a*/
逻辑运算符
下表列出了所有Go语言的逻辑运算符。假定 A 值为 True,B 值为 False
| 运算符 | 描述 | 实例 | 结果 |
|---|---|---|---|
| && | 逻辑与 (and) 如果两边的操作数都是 True,则条件 True,否则为 False |
A && B | False |
| || | 逻辑或 (or) 如果两边的操作数有一个 True,则条件 True,否则为 False |
A || B) | True |
| ! | 逻辑非 (not) 如果条件为 True,则逻辑 NOT 条件 False,否则为 True |
!(A && B) | True |
package mainimport "fmt"func main() {var a bool = truevar b bool = falseif ( a && b ) {fmt.Printf("第一行: 条件为 true\n" )}if ( a || b ) {fmt.Printf("第二行: 条件为 true\n" )}/* 修改 a 和 b 的值 */a = falseb = trueif ( a && b ) {fmt.Printf("第三行: 条件为 true\n" )} else {fmt.Printf("第三行: 条件为 false\n" )}if ( !(a && b) ) {fmt.Printf("第四行: 条件为 true\n" )}}/*第二行: 条件为 true第三行: 条件为 false第四行: 条件为 true*/
位运算符
位运算符对整数在内存中的二进制位进行操作,下表列出了位运算符 & | ^ << >>的功能 以下涉及到的运算全部都是假定 A = 60; B = 13
| 运算符 | 描述 | 实例 | 运行结果 |
|---|---|---|---|
| & | 按位与运算符”&”是双目运算符 其功能是参与运算的两数各对应的二进位相与 |
A & B | 12 二进制 0000 1100 |
| | | 按位或运算符”|”是双目运算符 其功能是参与运算的两数各对应的二进位相或 |
A | B | 61 二进制 0011 1101 |
| ^ | 按位异或运算符”^”是双目运算符 其功能是参与运算的两数各对应的二进位相异或,当两对应的二进位相异时,结果为1 |
A ^ B | 49 二进制 0011 0001 |
| << | 左移运算符”<<”是双目运算符。左移n位就是乘以2的n次方 其功能把”<<”左边的运算数的各二进位全部左移若干位,由”<<”右边的数指定移动的位数,高位丢弃,低位补0 |
A << 2 | 240 二进制 1111 0000 |
| >> | 右移运算符”>>”是双目运算符。右移n位就是除以2的n次方 其功能是把”>>”左边的运算数的各二进位全部右移若干位,”>>”右边的数指定移动的位数 |
A >> 2 | 15 二进制 0000 1111 |
| p | q | p & q | p | q | p ^ q |
|---|---|---|---|---|
| 0 | 0 | 0 | 0 | 0 |
| 0 | 1 | 0 | 1 | 1 |
| 1 | 0 | 0 | 1 | 1 |
| 1 | 1 | 1 | 1 | 0 |
| A | B | A&B | A|B | A^B |
| 0011 1100 | 0000 1101 | 0000 1100 | 0011 1101 | 0011 0001 |
| 对应的十进制整数 | ||||
| 60 | 13 | 12 | 61 | 49 |
package mainimport "fmt"func main() {var a uint = 60 // 60 = 0011 1100var b uint = 13 // 13 = 0000 1101var c uint = 0c = a & b // 12 = 0000 1100fmt.Printf("第一行: c 的值为 %d\n", c )c = a | b // 61 = 0011 1101fmt.Printf("第二行: c 的值为 %d\n", c )c = a ^ b // 49 = 0011 0001fmt.Printf("第三行: c 的值为 %d\n", c )c = a << 2 // 240 = 1111 0000fmt.Printf("第四行: c 的值为 %d\n", c )c = a >> 2 // 15 = 0000 1111fmt.Printf("第五行: c 的值为 %d\n", c )}/*第一行: c 的值为 12第二行: c 的值为 61第三行: c 的值为 49第四行: c 的值为 240第五行: c 的值为 15*/
指针运算符
下表列出了Go语言的其他运算符
| 运算符 | 描述 | 实例 |
|---|---|---|
| & | 返回变量存储地址 | &a 将给出变量的实际地址 |
| * | 指针变量 | *a 是一个指针变量 |
package mainimport "fmt"func main() {var a int = 4var b int32var c float32var ptr *int/* 运算符实例 */fmt.Printf("第1行: a变量类型为 = %T\n", a );fmt.Printf("第2行: b变量类型为 = %T\n", b );fmt.Printf("第3行: c变量类型为 = %T\n\n", c );/* & 和 * 运算符实例 */ptr = &a /* 'ptr' 包含了 'a' 变量的地址 */fmt.Printf("a的值为 %d\n", a);fmt.Printf("ptr的值为 %d\n", ptr);fmt.Printf("*ptr为 %d\n", *ptr);}/*第1行: a变量类型为 = int第2行: b变量类型为 = int32第3行: c变量类型为 = float32a的值为 4ptr的值为 824633819264*ptr为 4*/
指针变量 * 和地址值 & 的区别*
指针变量保存的是一个地址值,会分配独立的内存来存储一个整型数字。当变量前面有 标识时,才等同于 & 的用法,否则会直接输出一个整型数字
package mainimport "fmt"func main() {var a int = 4var ptr *intptr = &afmt.Println("a的值为", a); // 4fmt.Println("*ptr为", *ptr); // 4fmt.Println("ptr为", ptr); // 0xc000018080}
运算符优先级
有些运算符拥有较高的优先级,二元运算符的运算方向均是从左至右。下表列出了所有运算符以及它们的优先级,由上至下代表优先级由高到低:
| 优先级 | 运算符 |
|---|---|
| 5 | * / % << >> & &^ |
| 4 | + - | ^ |
| 3 | == != < <= > >= |
| 2 | && |
| 1 | || |
当然,你可以通过使用括号来临时提升某个表达式的整体运算优先级。
package mainimport "fmt"func main() {var a int = 20var b int = 10var c int = 15var d int = 5var e int;e = (a + b) * c / d; // ( 30 * 15 ) / 5fmt.Printf("(a + b) * c / d 的值为 : %d\n", e );e = ((a + b) * c) / d; // (30 * 15 ) / 5fmt.Printf("((a + b) * c) / d 的值为 : %d\n" , e );e = (a + b) * (c / d); // (30) * (15/5)fmt.Printf("(a + b) * (c / d) 的值为 : %d\n", e );e = a + (b * c) / d; // 20 + (150/5)fmt.Printf("a + (b * c) / d 的值为 : %d\n" , e );}/*(a + b) * c / d 的值为 : 90((a + b) * c) / d 的值为 : 90(a + b) * (c / d) 的值为 : 90a + (b * c) / d 的值为 : 50*/
