Go 语言运算符

简介: Go 语言内置的运算符有:算术运算符关系运算符逻辑运算符位运算符赋值运算符其他运算符

算术运算符



算术运算符,所有的数据类型要相同


下表列出了所有Go语言的算术运算符。假定 A 值为 10,B 值为 20。


运算符 描述 实例
+ 相加 A + B 输出结果 30
- 相减 A - B 输出结果 -10
* 相乘 A * B 输出结果 200
/ 相除 B / A 输出结果 2
% 求余 B % A 输出结果 0
++ 自增 A++ 输出结果 11
-- 自减 A-- 输出结果 9


package main
import "fmt"
func main() {
 var a int = 21
 var b int = 10
 var c int
 c = a + b
 fmt.Printf("第一行 - c 的值为 %d\n", c)
 c = a - b
 fmt.Printf("第二行 - c 的值为 %d\n", c)
 c = a * b
 fmt.Printf("第三行 - c 的值为 %d\n", c)
 c = a / b
 fmt.Printf("第四行 - c 的值为 %d\n", c)
 c = a % b
 fmt.Printf("第五行 - c 的值为 %d\n", c)
  //a++ 等效于 a=a+1  等效于 a+=1
 a++
 fmt.Printf("第六行 - a 的值为 %d\n", a)
 a = 21 // 为了方便测试,a 这里重新赋值为 21
  // a-- 等效于 a=a-1  等效于a-=1
 a--
 fmt.Printf("第七行 - a 的值为 %d\n", a)
}


//输出结果:

第一行 - c 的值为 31

第二行 - c 的值为 11

第三行 - c 的值为 210

第四行 - c 的值为 2

第五行 - c 的值为 1

第六行 - a 的值为 22

第七行 - a 的值为 20


算术运算符,也可以对字符串类型使用,此时就是字符串拼接


//对字符串做 + 
package main
import "fmt"
func main() {
 a := "hello"
 b := "world"
 c := a + b
 fmt.Println(c)
}


//输出结果:

helloworld


关系运算符



下表列出了所有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
package main
import "fmt"
func main() {
   var a int = 21
   var b int = 10
   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" )
   } else {
      fmt.Printf("第三行 - a 不大于 b\n" )
   }
   /* Lets change value of a and b */
   a = 5
   b = 20
   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 main
import "fmt"
func main() {
   var a bool = true
   var b bool = false
   if ( a && b ) {
      fmt.Printf("第一行 - 条件为 true\n" )
   }
   if ( a || b ) {
      fmt.Printf("第二行 - 条件为 true\n" )
   }
   /* 修改 a 和 b 的值 */
   a = false
   b = true
   if ( a && b ) {
      fmt.Printf("第三行 - 条件为 true\n" )
   } else {
      fmt.Printf("第三行 - 条件为 false\n" )
   }
   if ( !(a && b) ) {
      fmt.Printf("第四行 - 条件为 true\n" )
   }
}


//输出结果:

第二行 - 条件为 true

第三行 - 条件为 false

第四行 - 条件为 true


位运算符



位运算符对整数在内存中的二进制位进行操作。

& 与 : 只有都为1 时,结果为1,否则为0

| 或 : 只有都为0 是,结果为0,否则为1

^ 异或:只有都相同时(都为0,或者都为1),结果为0 ,否则为1

下表列出了位运算符 &, |, 和 ^ 的计算:


p q p & q p | q p ^ q
0 0 0 0 0
0 1 0 1 1
1 1 1 1 0
1 0 0 1 1


假定 A = 60; B = 13; 其二进制数转换为:


A = 0011 1100
B = 0000 1101
-----------------
A&B = 0000 1100
A|B = 0011 1101
A^B = 0011 0001

 


Go 语言支持的位运算符如下表所示。假定 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
package main
import "fmt"
func main() {
 var a uint = 60 /* 60 = 0011 1100 */
 var b uint = 13 /* 13 = 0000 1101 */
 var c uint = 0
 c = a & b /* 12 = 0000 1100 */
 fmt.Printf("第一行 - c 的值为 %d\n", c)
 c = a | b /* 61 = 0011 1101 */
 fmt.Printf("第二行 - c 的值为 %d\n", c)
 c = a ^ b /* 49 = 0011 0001 */
 fmt.Printf("第三行 - c 的值为 %d\n", c)
 c = a << 2 /* 240 = 1111 0000 */
 fmt.Printf("第四行 - c 的值为 %d\n", c)
 c = a >> 2 /* 15 = 0000 1111 */
 fmt.Printf("第五行 - c 的值为 %d\n", c)
}

//输出结果为:

第一行 - c 的值为 12

第二行 - c 的值为 61

第三行 - c 的值为 49

第四行 - c 的值为 240

第五行 - c 的值为 15


赋值运算符



下表列出了所有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 main
import "fmt"
func main() {
 var a int = 21
 var c int
 c = a
 fmt.Printf("第 1 行 - =  运算符实例,c 值为 = %d\n", c)
 c += a
 fmt.Printf("第 2 行 - += 运算符实例,c 值为 = %d\n", c)
 c -= a
 fmt.Printf("第 3 行 - -= 运算符实例,c 值为 = %d\n", c)
 c *= a
 fmt.Printf("第 4 行 - *= 运算符实例,c 值为 = %d\n", c)
 c /= a
 fmt.Printf("第 5 行 - /= 运算符实例,c 值为 = %d\n", c)
 c = 200
 c <<= 2
 fmt.Printf("第 6行  - <<= 运算符实例,c 值为 = %d\n", c)
 c >>= 2
 fmt.Printf("第 7 行 - >>= 运算符实例,c 值为 = %d\n", c)
 c &= 2
 fmt.Printf("第 8 行 - &= 运算符实例,c 值为 = %d\n", c)
 c ^= 2
 fmt.Printf("第 9 行 - ^= 运算符实例,c 值为 = %d\n", c)
 c |= 2
 fmt.Printf("第 10 行 - |= 运算符实例,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

第 10 行 - |= 运算符实例,c 值为 = 2


其他运算符



下表列出了Go语言的其他运算符。


运算符 描述 实例
& 返回变量存储地址 &a; 将给出变量的实际地址。
* 指针变量。 *a; 是一个指针变量
package main
import "fmt"
func main() {
 var a int = 4
 var b int32
 var c float32
 var ptr *int
 /* 运算符实例 */
 fmt.Printf("第 1 行 - a 变量类型为 = %T\n", a)
 fmt.Printf("第 2 行 - b 变量类型为 = %T\n", b)
 fmt.Printf("第 3 行 - c 变量类型为 = %T\n", c)
 /*  & 和 * 运算符实例 */
 ptr = &a /* 'ptr' 包含了 'a' 变量的地址 */
 fmt.Printf("a 的值为  %d\n", a)
 fmt.Printf("*ptr 为 %d\n", *ptr)
}

//输出结果:

第 1 行 - a 变量类型为 = int

第 2 行 - b 变量类型为 = int32

第 3 行 - c 变量类型为 = float32

a 的值为  4

*ptr 为 4


运算符优先级



有些运算符拥有较高的优先级,二元运算符的运算方向均是从左至右。下表列出了所有运算符以及它们的优先级,由上至下代表优先级由高到低:


优先级 运算符
5 * / % << >> & &^
4 + - | ^
3 == != < <= > >=
2 &&
1 ||


当然,你可以通过使用括号来临时提升某个表达式的整体运算优先级。

package main
import "fmt"
func main() {
 var a int = 20
 var b int = 10
 var c int = 15
 var d int = 5
 var e int
 e = (a + b) * c / d // ( 30 * 15 ) / 5
 fmt.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) // (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) 的值为  : 90

a + (b * c) / d 的值为  : 50



相关文章
|
5天前
|
JSON 测试技术 Go
零值在go语言和初始化数据
【7月更文挑战第10天】本文介绍在Go语言中如何初始化数据,未初始化的变量会有对应的零值:bool为`false`,int为`0`,byte和string为空,pointer、function、interface及channel为`nil`,slice和map也为`nil`。。本文档作为指南,帮助理解Go的数据结构和正确使用它们。
53 22
零值在go语言和初始化数据
|
6天前
|
JSON Java Go
Go 语言性能优化技巧
在Go语言中优化性能涉及数字字符串转换(如用`strconv.Itoa()`代替`fmt.Sprintf()`)、避免不必要的字符串到字节切片转换、预分配切片容量、使用`strings.Builder`拼接、有效利用并发(`goroutine`和`sync.WaitGroup`)、减少内存分配、对象重用(`sync.Pool`)、无锁编程、I/O缓冲、正则预编译和选择高效的序列化方法。这些策略能显著提升代码执行效率和系统资源利用率。
42 13
|
1天前
|
Cloud Native Java Go
为什么要学习Go语言?
GO logo的核心理念,即简单胜于复杂。使用现代斜体无衬线字体与三条简单的运动线相结合,形成一个类似于快速运动的两个轮子的标记,传达速度和效率。字母的圆形暗示了GO地鼠的眼睛,创造了一个熟悉的形状,让标记和吉祥物很好地搭配在一起。
13 4
|
6天前
|
设计模式 Go
Go语言设计模式:使用Option模式简化类的初始化
在Go语言中,面对构造函数参数过多导致的复杂性问题,可以采用Option模式。Option模式通过函数选项提供灵活的配置,增强了构造函数的可读性和可扩展性。以`Foo`为例,通过定义如`WithName`、`WithAge`、`WithDB`等设置器函数,调用者可以选择性地传递所需参数,避免了记忆参数顺序和类型。这种模式提升了代码的维护性和灵活性,特别是在处理多配置场景时。
41 8
|
5天前
|
存储 Go
go语言中fmt格式化包和内置函数汇总
【7月更文挑战第10天】本文介绍fmt包和`Errorf`用于创建格式化的错误消息。`fmt`包还涉及一些接口,如`Formatter`、`GoStringer`、`ScanState`、`Scanner`和`Stringer`,支持自定义格式化和输入/输出处理。
17 1
|
5天前
|
Go
go语言中格式化输出的占位符
【7月更文挑战第10天】`fmt` 包在 Go 语言中用于格式化输出,包括不同类型的占位符:%v(默认格式)、%+v(带字段名的结构体)、%#v(Go语法表示)、%T(类型表示)、%%(百分号)。布尔值用%t,整数有%b、%c、%d、%o、%q、%x、%X和%U。浮点数和复数用%b、%e、%E、%f、%g、%G。字符串和字节切片用%s、%q、%x、%X。指针用%p。占位符可配合+、-、#、空格和0进行调整。宽度和精度控制输出格式,例如 %.4g 控制小数精度。Go 没有 `%u`,但无符号整数默认打印为正数。运算符包括逻辑、比较、加减、乘除、移位、按位和按位异或等。
17 1
|
7天前
|
存储 Go 索引
在go语言中自定义泛型的变长参数
【7月更文挑战第8天】在Go语言中,由于官方1.18以前的版本不支持泛型,可以通过空接口和反射模拟泛型。泛型适用于通用数据结构和函数,虽牺牲了一些性能,但提高了代码复用和类型安全性。
42 1
|
3天前
|
安全 Go
Go语言map并发安全,互斥锁和读写锁谁更优?
Go并发编程中,`sync.Mutex`提供独占访问,适合读写操作均衡或写操作频繁的场景;`sync.RWMutex`允许多个读取者并行,适用于读多写少的情况。明智选择锁可提升程序性能和稳定性。示例展示了如何在操作map时使用这两种锁。
6 0
|
3天前
|
安全 Go 开发者
Go语言map并发安全使用的正确姿势
在Go并发编程中,由于普通map不是线程安全的,多goroutine访问可能导致数据竞态。为保证安全,可使用`sync.Mutex`封装map或使用从Go 1.9开始提供的`sync.Map`。前者通过加锁手动同步,后者内置并发控制,适用于多goroutine共享。选择哪种取决于具体场景和性能需求。
6 0
|
3天前
|
存储 安全 Java
Go语言中的map为什么默认不是并发安全的?
Go语言的map默认不保证并发安全,以优化性能和简洁性。官方建议在需要时使用`sync.Mutex`保证安全。从Go 1.6起,并发读写map会导致程序崩溃,鼓励开发者显式处理并发问题。这样做的哲学是让代码更清晰,并避免不必要的性能开销。
4 0