Go语言的几个注意点

简介:

撸了一段时间的Go,觉得Go确实是个不错的语言,适合写高并发的程序,又自带各种强大的库。不过我最喜欢的还是C++,哈哈。

这里总结一下几个写Go的注意点。

Go语言的协程中,写死循环的注意点:

现象:

在写Go的多协程程序时,出现过几次无法理解的情况。

  • 有一次,我想写一个能跑满cpu的程序,最容易想到的就是,开几个Go的协程,每个协程里写死循环。没想到,运行的时候发现,协程就只开出了一个。
  • 另一次,我写了个程序,也是开了多个协程。因为如果不阻塞住主函数,主函数一结束,程序就会结束。所以我就在主函数结束前加了个死循环。然后就发现整个协程都被卡住了。

分析:

其实,这个东西是协程的特点。以前没用过协程,加上Go又说可以当线程用。所以想当然的写了死循环。

准确的说,是在Go语言里,写了死循环,并且死循环内并没有什么系统调用,只有简单的计算这类的。你就会发现,Go的协程调度就废掉了。

协程并非像线程那样,是由CPU中断来触发切换的。它不是应用程序能控制的(操作系统内核的某些关键操作会被保护,不被中断)。即使你在线程里写了死循环,只要周期一到,CPU产生终端,死循环会被打断,重新调度。但是,协程就不是这样了,协程的调度其实是在协程调用了某个系统调用时,自动跳到另一个协程执行。也就是这个“中断”是程序主动产生的,而不是被”中断”。

所以,协程中,如果你写了死循环,那你的死循环就会一直跑着,而不会让别的协程运行。主函数中也是一样,而且主函数中执行这个会让整个协程卡住,因为调度的代码没法被执行。

在Go语言中,如果你想写死循环,循环里面没有系统调用,又想让Go的协程能起作用,只需要在死循环里面加一条语句即可。估计系统调用时也是这个语句起的作用。

1 runtime.Gosched()
2 //主动让出时间片

Go的并发设置:

现象:

Go语言最大的优势就在于写高并并发的程序,能很方便的利用goroutine来充分利用系统资源,但估计你用协程写出的第一个高并发程序都没有充分的利用起CPU。最多就跑个100%,这让我几十核的CPU情何以堪啊。

分析:

因为Go默认情况下只用单线程。这就是说,你即使开了几百个goroutine,系统中同一时间在跑的只有一个线程,也就是一个协程。那是因为没有设置并发度。(╯’ – ‘)╯︵ ┻━┻

1 runtime.GOMAXPROCS()    //这个函数设置的是Go语言跑几个线程。
2 runtime.NumCPU()              //这个函数返回当前有的CPU数。

CPU并不知道协程,CPU只认识线程,CPU的核心数就是CPU能同时(同一个时间点)运行的线程的数量。协程则会挂在每个线程上,goroutine也会适当的调整协程,让它均匀的挂在每个线程上。

一般情况,线程的数量建议是CPU核数的2倍。所以我一般会这么设置:

1 runtime.GOMAXPROCS(runtime.NumCPU()*2)

Go语言可导出标识:

可导出就是说可以把import的包里的函数和变量暴露出来,可以被调用和访问。

这个有点类似于C++中,private和public这类访问控制。只是Go的控制是以包来划分,C++这个以类来划分。如果这个包的函数/变量是私有的,那么即使import了包,也无法调用里面的函数或访问里面的变量。

这会在编译的时候爆出错误。

1 //调用未导出的函数
2 xxxxx undefined (cannot refer to unexported field or method xxxxxx)
3  
4 //隐式赋值时的报错,因为那个被赋值的类变量是不被导出的。
5 implicit assignment of unexported field 'xxxxxx' in xxxxx literal

Go语言的导出规范:

  • 公有函数/变量的名字以大写字母开头
  • 私有函数/变量的名字以小写字母开头

声明的变量必须使用:

在Go语言中你必须使用所有被声明的变量。或者是import时导出的包。函数可以声明了但是不用。如果你写了个Go程序,发现编译时一堆报错,很可能里面大部分错误是因为你声明了变量但是没用。

真的是个反人类的设定。没办法,这个只能忍着。

总结:

总的来说,Go确实是个不错的语言。除了有一些反人类的设定。

要是C++的标准库能和Go一样,果断就抛弃Go了 ( ̄▽ ̄)~*。听说Google内部都是用C++的,就是因为有强大的类库。

转载请注明:旅途@KryptosX » Go语言的几个注意点

目录
相关文章
|
1月前
|
存储 安全 Java
【Golang】(4)Go里面的指针如何?函数与方法怎么不一样?带你了解Go不同于其他高级语言的语法
结构体可以存储一组不同类型的数据,是一种符合类型。Go抛弃了类与继承,同时也抛弃了构造方法,刻意弱化了面向对象的功能,Go并非是一个传统OOP的语言,但是Go依旧有着OOP的影子,通过结构体和方法也可以模拟出一个类。
159 1
|
3月前
|
Cloud Native 安全 Java
Go:为云原生而生的高效语言
Go:为云原生而生的高效语言
286 1
|
9月前
|
编译器 Go
揭秘 Go 语言中空结构体的强大用法
Go 语言中的空结构体 `struct{}` 不包含任何字段,不占用内存空间。它在实际编程中有多种典型用法:1) 结合 map 实现集合(set)类型;2) 与 channel 搭配用于信号通知;3) 申请超大容量的 Slice 和 Array 以节省内存;4) 作为接口实现时明确表示不关注值。此外,需要注意的是,空结构体作为字段时可能会因内存对齐原因占用额外空间。建议将空结构体放在外层结构体的第一个字段以优化内存使用。
|
9月前
|
运维 监控 算法
监控局域网其他电脑:Go 语言迪杰斯特拉算法的高效应用
在信息化时代,监控局域网成为网络管理与安全防护的关键需求。本文探讨了迪杰斯特拉(Dijkstra)算法在监控局域网中的应用,通过计算最短路径优化数据传输和故障检测。文中提供了使用Go语言实现的代码例程,展示了如何高效地进行网络监控,确保局域网的稳定运行和数据安全。迪杰斯特拉算法能减少传输延迟和带宽消耗,及时发现并处理网络故障,适用于复杂网络环境下的管理和维护。
|
3月前
|
Cloud Native Go API
Go:为云原生而生的高效语言
Go:为云原生而生的高效语言
362 0
|
3月前
|
Cloud Native Java Go
Go:为云原生而生的高效语言
Go:为云原生而生的高效语言
235 0
|
3月前
|
Cloud Native Java 中间件
Go:为云原生而生的高效语言
Go:为云原生而生的高效语言
212 0
|
3月前
|
Cloud Native Java Go
Go:为云原生而生的高效语言
Go:为云原生而生的高效语言
305 0
|
3月前
|
数据采集 Go API
Go语言实战案例:多协程并发下载网页内容
本文是《Go语言100个实战案例 · 网络与并发篇》第6篇,讲解如何使用 Goroutine 和 Channel 实现多协程并发抓取网页内容,提升网络请求效率。通过实战掌握高并发编程技巧,构建爬虫、内容聚合器等工具,涵盖 WaitGroup、超时控制、错误处理等核心知识点。
|
3月前
|
数据采集 JSON Go
Go语言实战案例:实现HTTP客户端请求并解析响应
本文是 Go 网络与并发实战系列的第 2 篇,详细介绍如何使用 Go 构建 HTTP 客户端,涵盖请求发送、响应解析、错误处理、Header 与 Body 提取等流程,并通过实战代码演示如何并发请求多个 URL,适合希望掌握 Go 网络编程基础的开发者。