golang开发:go并发的建议
这个是前段时间看到Go语言的贡献者与布道师 Dave Cheney对Go并发的建议或者叫使用的陷阱(不是我自己的建议),结合自己最近几年对gorotine的使用,再回头看这几条建议,真的会茅塞顿开,觉得特别重要。这篇文章对并发的建议的章节地址
https://dave.cheney.net/practical-go/presentations/qcon-china.html#_concurrency
Dave Cheney
Dave Cheney 是 Go 编程语言的开源贡献者和项目成员。David 是技术社区中备受尊敬的声音,他就软件设计、性能和 Go 编程语言等各种主题发表演讲。David 在go语言历程中,分享过很多关于Golang语言的正确使用的文章。这是他的博客地址。
https://dave.cheney.net/
8.1. Keep yourself busy or do the work yourself(让自己忙碌起来或自己做工作)
这个建议应该比较容易理解,启动一个gorotine应该是执行程序的,自己执行或者被人调用执行,不应该启动gorotine之后这个gorotine啥事都没干。
作者举了一个例子
package main
import (
"fmt"
"log"
"net/http"
)
func main() {
http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
fmt.Fprintln(w, "Hello, GopherCon SG")
})
go func() {
if err := http.ListenAndServe(":8080", nil); err != nil {
log.Fatal(err)
}
}()
for {
}
}
为了阻塞main gorotine不要直接退出,等待go func的执行,最后写了一个for的死循环,这样的话,main gorotine就是通常所说的啥事都没干,毫无结果地运行。我们当然可以使用WaitGroup去等待go func的结束。作者给我们的建议,既然我们只有一个任务需要做,main gorotine就可以完成,为什么要启动一个gorotine去做这个任务,而让main gorotine去等待,完全可以让main去做这个任务
import (
"fmt"
"log"
"net/http"
)
func main() {
http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
fmt.Fprintln(w, "Hello, GopherCon SG")
})
if err := http.ListenAndServe(":8080", nil); err != nil {
log.Fatal(err)
}
}
许多 Go 程序员过度使用 goroutine,尤其是在他们刚开始的时候。与生活中的所有事物一样,适度是成功的关键。
8.2. Leave concurrency to the caller(将并发留给调用者)
这个表述起来比较容易,平常开发中可能会被忽略,一个对象提供了启动使用goroutine的方法,那么就必须提供关闭goroutine的方法,而且一般得原则的是谁调用谁关闭。
举一个我们项目开发中的例子
timer_go.go
package main
import (
"fmt"
"sync"
"time"
)
type TimerGo struct {
quit chan bool
}
func NewTimerGo() *TimerGo {
timer := new(TimerGo)
timer.quit = make(chan bool)
return timer
}
func (this *TimerGo) Run(wg *sync.WaitGroup) {
defer wg.Done()
cfgTime := 3
t := time.NewTicker(time.Duration(cfgTime) * time.Second)
defer t.Stop()
for {
select {
case <- this.quit:
fmt.Println("quite")
return
case <-t.C:
this.Sync()
}
}
}
func (this *TimerGo) Sync() {
fmt.Printf("Sync")
}
func (this *TimerGo) Close() {
close(this.quit)
}
main.go
package main
func main() {
timergo := NewTimerGo()
wg := new(sync.WaitGroup)
wg.Add(1)
go timergo.Run(wg)
//start up
timergo.Close()
wg.Wait()
}
这个例子比较容易理解,我们需要每隔三秒执行一个异步的任务,这个工作我们启动一个goroutine去执行,所以我们在main函数执行go timergo.Run,我们也提供Close的方法,通过一个channal去关闭它。
原则就是,谁调用谁关闭。提供执行方法,就必须提供关闭方法。
8.3. Never start a goroutine without knowning when it will stop(永远不要在不知道何时停止的情况下启动 goroutine)
这个原则我觉得应该是最重要的原则,而且在开发中最容易遇到的问题。我们前期也写过很多这样的代码,而且我看大家使用的项目基本也都是在需要启动一个goroutine去执行代码的时候是这样写的
go AAA()
go BBB()
go CCC()
很少有人去关心启动的这三个goroutine应该在什么情况下去关闭,应该怎么关闭,他们得运行状态是怎么样的,在服务重新启动时候,是等待执行完毕还是强制中断。
这个原则应该会指引我们去做一些可靠的架构和规划。这个遇到的太多了,有必要花时间去整理这里。
永远不要在不知道何时停止的情况下启动 goroutine
golang开发:go并发的建议的更多相关文章
- golang开发:go并发的建议(完)
上次说了一下Go语言布道师 Dave Cheney对Go并发的建议,个人觉得最重要的一条,这次主要想说一下这个. 8.3. Never start a goroutine without knowni ...
- 用golang开发系统软件的一些细节
用golang开发系统软件的一些细节 作者:张富春(ahfuzhang),转载时请注明作者和引用链接,谢谢! cnblogs博客 zhihu Github 公众号:一本正经的瞎扯 (本文的pdf版本) ...
- window / Linux 下 Golang 开发环境的配置
一直专注于使用python语言进行程序开发,但是却又一直被它的性能问题所困扰,直到遇到了天生支持高并发的Golang,这似乎也成了学习go语言最理所当然的理由.下面介绍下Go语言开发环境搭建的步骤: ...
- Golang开发支持平滑升级(优雅重启)的HTTP服务
Golang开发支持平滑升级(优雅重启)的HTTP服务 - tabalt的博客 http://tabalt.net/blog/graceful-http-server-for-golang/ http ...
- Visual Studio Code配置GoLang开发环境
Visual Studio Code配置GoLang开发环境 在Visual Studio Code配置GoLang开发环境 作者:chszs,未经博主允许不得转载.经许可的转载需注明作者和博客主页: ...
- Linux下配置Golang开发环境
前几天无意间看到了微信推送的golang开发的消息,看到golang那么牛逼,突然心血来潮想学习一下go.工欲善其事必先利其器,想做go开发,必须先配置好go的开发环境(就像开发Java先安装配置jd ...
- [golang]Golang实现高并发的调度模型---MPG模式
Golang实现高并发的调度模型---MPG模式 传统的并发形式:多线程共享内存,这也是Java.C#或者C++等语言中的多线程开发的常规方法,其实golang语言也支持这种传统模式,另外一种是Go语 ...
- Golang 开发技能图谱
# Golang 开发技能图谱 ## Go 语言环境搭建- Go 安装- GOPATH 与工作空间- Go 命令- Go开发工具 ## GO 语言编程基础- 关键词和语法(Language Synta ...
- 关于Web开发里并发、同步、异步以及事件驱动编程的相关技术
一.开篇语 我的上篇文章<关于如何提供Web服务端并发效率的异步编程技术>又成为了博客园里“编辑推荐”的文章,这是对我写博客很大的鼓励,也许是被推荐的原因很多童鞋在这篇文章里发表了评论,有 ...
随机推荐
- tensorflow源码解析之framework-device
目录 什么是设备 设备属性描述 device_base 关系图 涉及的文件 迭代记录 1. 什么是设备 "设备"是一个很容易引起混淆的概念,在TF中,设备device专指能够执行实 ...
- Linux内存泄漏
0 什么是内存泄漏? 内存泄漏(Memory Leak)是指程序中已动态分配的堆内存由于某种原因程序未释放或无法释放,造成系统内存的浪费,导致程序运行速度减慢甚至系统崩溃等严重后果. 1 常见的造成内 ...
- java-Dos
打开CMD的方式 1.菜单打开 2.Windows+R 输入cmd 3.shift+鼠标右键 选择在此处打开命令行窗口 4.资源管理器地址栏前+cmd 空格 管理员身份运行 常用的Dos命令 #盘符切 ...
- JavaWeb 08_JSP+Dao+Bean+Servlet 实现登录注册(连接数据库,验证码登录,两周内免登陆等功能)
一.数据库db_01 表usert 字段username,password 二. 目录 三. 配置信息 四. 代码 index.jsp <script type="text/j ...
- JavaWeb 03_创建servlet项目(详细)
一.创建web项目 1. File--New--Project 2. 设置项目相关信息 3. 设置项目名称及工作空间 4. web项目目录结构如下 二.Servlet的实现 1. 新建包---类 ...
- Golang 常见设计模式之装饰模式
想必只要是熟悉 Python 的同学对装饰模式一定不会陌生,这类 Python 从语法上原生支持的装饰器,大大提高了装饰模式在 Python 中的应用.尽管 Go 语言中装饰模式没有 Python 中 ...
- (leetcode)二叉树的前序遍历-c语言实现
给定一个二叉树,返回它的 前序 遍历. 示例: 输入: [1,null,2,3] 1 \ 2 / 3 输出: [1,2,3] 进阶: 递归算法很简单,你可以通过迭代算法完成吗? 前序遍历 前序遍历首先 ...
- Basler acA1300-200uc相机使用教程
https://www.baslerweb.com/cn/products/cameras/area-scan-cameras/ace/aca1300-200uc/ 开发文档 https://zh.d ...
- Java反射详解篇--一篇入魂
1.反射概述 Java程序在运行时操作类中的属性和方法的机制,称为反射机制. 一个关键点:运行时 一般我们在开发程序时,都知道自己具体用了什么类,直接创建使用即可.但当你写一些通用的功能时没办法在编写 ...
- 半吊子菜鸟学Web开发4 --Html css学习2
1<a>标签,链接到另一个页面 <a href="目标网址" title="鼠标滑过显示的文本">链接显示的文本</a> H ...