加 Golang学习 QQ群共同学习进步成家立业工作 ^-^ 群号:96933959

Goroutine

Go语言的主要的功能在于令人简易使用的并行设计,这个方法叫做Goroutine,通过Goroutine能够让你的程序以异步的方式运行,而不需要担心一个函数导致程序中断,因此Go语言也非常地适合网络服务。

我们通过go让其中一个函数同步运行,如此就不需要等待该函数运行完后才能运行下一个函数。

func main() {
// 通过 `go`,我们可以把这个函数异步执行,这样就不会阻塞往下执行。
go loop()
// 执行 Other
}

Goroutine是类似线程的概念(但Goroutine并不是线程)。线程属于系统层面,通常来说创建一个新的线程会消耗较多的资源且管理不易。而 Goroutine就像轻量级的线程,但我们称其为并发,一个Go程序可以运行超过数万个 Goroutine,并且这些性能都是原生级的,随时都能够关闭、结束。一个核心里面可以有多个Goroutine,通过GOMAXPROCS参数你能够限制Gorotuine可以占用几个系统线程来避免失控。

在内置的官方包中也不时能够看见Goroutine的应用,像是net/http中用来监听网络服务的函数实际上是创建一个不断运行循环的Goroutine。

设置同时执行的cpu数(GOMAXPROCS)

GOMAXPROCS 在调度程序优化后会去掉,默认用系统所有资源。

func main() {
num := runtime.NumCPU() //本地机器的逻辑CPU个数
runtime.GOMAXPROCS(num) //设置可同时执行的最大CPU数,并返回先前的设置
fmt.Println(num)
}

Goroutine中使用recover

应用场景,如果某个goroutine panic了,而且这个goroutine里面没有捕获(recover),那么整个进程就会挂掉。所以,好的习惯是每当go产生一个goroutine,就需要写下recover。

var (
domainSyncChan = make(chan int, )
) func domainPut(num int) {
defer func() {
err := recover()
if err != nil {
fmt.Println("error to chan put.")
}
}()
domainSyncChan <- num panic("error....")
} func main() {
for i := ; i < ; i++ {
domainName := i
go domainPut(domainName)
}
time.Sleep(time.Second * )
}

Goroutine 栗子

package main

import (
"fmt"
"sync"
"time"
) var (
m = make(map[int]uint64)
lock sync.Mutex //申明一个互斥锁
) type task struct {
n int
} func calc(t *task) {
defer func() {
err := recover()
if err != nil {
fmt.Println("error...")
return
}
}() var sum uint64
sum =
for i := ; i < t.n; i++ {
sum *= uint64(i)
} lock.Lock() //写全局数据加互斥锁
m[t.n] = sum
lock.Unlock() //解锁
} func main() {
for i := ; i < ; i++ {
t := &task{n: i}
go calc(t) // Goroutine来执行任务
} time.Sleep(time.Second) // Goroutine异步,所以等一秒到任务完成 lock.Lock() //读全局数据加锁
for k, v := range m {
fmt.Printf("%d! = %v\n", k, v)
}
fmt.Println(len(m))
lock.Unlock() //解锁
}

Goroutine 栗子(等待所有任务退出主程序再退出)

package main

import (
"sync"
"fmt"
"time"
) func calc(w *sync.WaitGroup, i int) {
fmt.Println("calc: ", i)
time.Sleep(time.Second)
w.Done()
} func main() {
wg := sync.WaitGroup{}
for i:=; i<; i++ {
wg.Add()
go calc(&wg, i)
}
wg.Wait()
fmt.Println("all goroutine finish")
}

Channel

channel,管道、队列,先进先出,用来异步传递数据。channel加上goroutine,就形成了一种既简单又强大的请求处理模型,使高并发和线程同步之间代码的编写变得异常简单。

线程安全,多个goroutine同时访问,不需要加锁。

channel是有类型的,一个整数的channel只能存放整数。

channel使用

//chan申明
var userChan chan interface{} // chan里面放interface类型
userChan = make(chan interface{}, ) // make初始化,大小为10 var readOnlyChan <-chan int // 只读chan
var writeOnlyChan chan<- int // 只写chan
//chan放取数据
userChan <- "nick"
name := <- userChan
name, ok := <- userChan
//关闭chan
intChan := make(chan int, )
intChan <-
close(intChan)
// range chan
intChan := make(chan int, )
for i := ; i < ; i++ {
intChan <- i
}
close(intChan) for v := range intChan {
fmt.Println(v)
}

放入chan数据个数超过初始化指定大小会怎样?

userChan := make(chan interface{})
userChan <- "nick"
// 错误!fatal error: all goroutines are asleep - deadlock!
// 开启race会一直阻塞

开启一个goroutine来放入初始化未指定大小的chan不会报错。

即放即走,在等放入时有来拿数据的,就直接拿走。

userChan := make(chan interface{})
go func() {
userChan <- "nick"
}()
name := <- userChan
userChan := make(chan interface{})
go func() {
for {
userChan <- "nick"
}
}()
for {
name := <- userChan
fmt.Println(name)
time.Sleep(time.Millisecond)
}

chan关闭与不关闭

关闭chan后再放入数据会 panic: send on closed channel。

chan不关闭取超数据的情况会报 deadlock

func main() {
intChan := make(chan int, ) for i := ; i < ; i++ {
intChan <- i
}
for {
//十次后 fatal error: all goroutines are asleep - deadlock!
i := <- intChan
fmt.Println(i)
time.Sleep(time.Second)
}
}

chan关闭的情况取超出值为类型默认值,如int为0

func main() {
intChan := make(chan int, ) for i := ; i < ; i++ {
intChan <- i
}
close(intChan) for {
i := <- intChan
//十次后i值都为0,不报错
time.Sleep(time.Second)
fmt.Println(i)
}
}

判断chan是否取完

func main() {
intChan := make(chan int, ) for i := ; i < ; i++ {
intChan <- i
}
close(intChan) for {
i, ok := <- intChan
if !ok {
fmt.Println("channel is close.")
return
}
fmt.Println(i)
}
}

channel 栗子

栗子一

func sendData(ch chan<- string) {
ch <- "go"
ch <- "java"
ch <- "c"
ch <- "c++"
ch <- "python"
close(ch)
} func getData(ch <-chan string, chColse chan bool) {
for {
str, ok := <-ch
if !ok {
fmt.Println("chan is close.")
break
}
fmt.Println(str)
}
chColse <- true
} func main() {
ch := make(chan string, )
chColse := make(chan bool, )
go sendData(ch)
go getData(ch, chColse)
<-chColse
close(chColse)
}

栗子二:interface类型chan,取出后转化为对应类型。

type user struct {
Name string
} func main() {
userChan := make(chan interface{}, ) u := user{Name: "nick"}
userChan <- &u
close(userChan) var u1 interface{}
u1 = <-userChan var u2 *user
u2, ok := u1.(*user)
if !ok {
fmt.Println("cant not convert.")
return
}
fmt.Println(u2)
}

channel 超时处理

利用select来处理chan超时。

for {
select {
case v := <-chan1:
fmt.Println(v)
case v := <-chan2:
fmt.Println(v)
default:
time.Sleep(time.Second)
fmt.Println("timeout...")
}
}

time.After()定时器来做处理。

在time.After()计时器触发之前,底层计时器不会被垃圾收集器回收。

select {
case m := <-c:
handle(m)
case <-time.After( * time.Minute):
fmt.Println("timed out")
}
    t := time.NewTicker(time.Second)
fmt.Println(t)
for v := range t.C {
fmt.Println(v)
}
t.Stop()

定时器栗子

Goroutine+Channel 栗子

栗子一

多个goroutine处理任务;

等待一组channel的返回结果。

func calc(taskChan, resChan chan int, exitChan chan bool) {
defer func() {
err := recover()
if err != nil {
fmt.Println("error...")
return
}
}() for v := range taskChan {
// 任务处理逻辑
flag := true
for i := ; i < v; i++ {
if v%i == {
flag = false
break
}
}
if flag {
//结果进chan
resChan <- v
}
}
//处理完进退出chan
exitChan <- true
} func main() {
//任务chan
intChan := make(chan int, )
//结果chan
resChan := make(chan int, )
//退出chan
exitChan := make(chan bool, ) go func() {
for i := ; i < ; i++ {
intChan <- i
}
close(intChan)
}() //启动8个goroutine做任务
for i := ; i < ; i++ {
go calc(intChan, resChan, exitChan)
} go func() {
//等所有goroutine结束
for i := ; i < ; i++ {
<-exitChan
}
close(resChan)
close(exitChan)
}() for v := range resChan {
fmt.Println(v)
}
}

栗子二

等待一组channel的返回结果 sync.WaitGroup 的解决方法。

WaitGroup用于等待一组线程的结束。父线程调用Add方法来设定应等待的线程的数量。每个被等待的线程在结束时应调用Done方法。同时,主线程里可以调用Wait方法阻塞至所有线程结束。

func merge(cs <-chan int) <-chan int {
var wg sync.WaitGroup
out := make(chan int) output := func(c <-chan int) {
for n := range c {
out <- n
}
wg.Done()
}
wg.Add(len(cs)) for _, c := range cs {
go output(c)
} go func() {
wg.Wait()
close(out)
}()
return out
}

Go语言学习笔记(七)杀手锏 Goroutine + Channel的更多相关文章

  1. Go语言学习笔记七: 函数

    Go语言学习笔记七: 函数 Go语言有函数还有方法,神奇不.这有点像python了. 函数定义 func function_name( [parameter list] ) [return_types ...

  2. Go语言学习笔记五: 条件语句

    Go语言学习笔记五: 条件语句 if语句 if 布尔表达式 { /* 在布尔表达式为 true 时执行 */ } 竟然没有括号,和python很像.但是有大括号,与python又不一样. 例子: pa ...

  3. Go语言学习笔记二: 变量

    Go语言学习笔记二: 变量 今天又学了一招如何查看go的版本的命令:go version.另外上一个笔记中的代码还可以使用go run hello.go来运行,只是这种方式不会生成exe文件. 定义变 ...

  4. go微服务框架kratos学习笔记七(kratos warden 负载均衡 balancer)

    目录 go微服务框架kratos学习笔记七(kratos warden 负载均衡 balancer) demo demo server demo client 池 dao service p2c ro ...

  5. Go语言学习笔记(1)——顺序编程

    Go语言学习笔记这一堆主要是<Go语言编程>(人民邮电出版社)的读书笔记.中间会穿插一些零碎的点,比如源码学习之类的.大概就是这样吧. 1. 顺序编程 1.1 变量 变量的声明: var ...

  6. Java IO学习笔记七:多路复用从单线程到多线程

    作者:Grey 原文地址:Java IO学习笔记七:多路复用从单线程到多线程 在前面提到的多路复用的服务端代码中, 我们在处理读数据的同时,也处理了写事件: public void readHandl ...

  7. C语言学习 第七次作业总结

    C语言学习 第七次作业总结 数组可以分为数组和多下标数组(在传统的国内C语言书本中,将其称为二/多维数组). 数组名称 在之前的课程中,大家应该都有印象,对于int a这样的定义,会为变量 a 声明一 ...

  8. HTML语言学习笔记(会更新)

    # HTML语言学习笔记(会更新) 一个html文件是由一系列的元素和标签组成的. 标签: 1.<html></html> 表示该文件为超文本标记语言(HTML)编写的.成对出 ...

  9. (转)Qt Model/View 学习笔记 (七)——Delegate类

    Qt Model/View 学习笔记 (七) Delegate  类 概念 与MVC模式不同,model/view结构没有用于与用户交互的完全独立的组件.一般来讲, view负责把数据展示 给用户,也 ...

  10. 2017-04-21周C语言学习笔记

    C语言学习笔记:... --------------------------------- C语言学习笔记:学习程度的高低取决于.自学能力的高低.有的时候生活就是这样的.聪明的人有时候需要.用笨的方法 ...

随机推荐

  1. UE4 difference between servertravel and openlevel(多人游戏的关卡切换)

    多人游戏的关卡切换分为无缝和非无缝.非无缝切换时,客户端将跟服务器断开连接,然后重新连接到同一个服务器,服务器则加载一个新地图.无缝切换不会发生这样的情况. 有三个函数供我们使用:UEngine::B ...

  2. 在Cenos系统安装Python3.5版本,使P2和P3共存

    首先Cenos安装好后,系统自带python2.6版本 输入>>>exit()     退出 使用迅雷下载python3.5 链接:https://www.python.org/ft ...

  3. 【Android Developers Training】 83. 实现高效网络访问来优化下载

    注:本文翻译自Google官方的Android Developers Training文档,译者技术一般,由于喜爱安卓而产生了翻译的念头,纯属个人兴趣爱好. 原文链接:http://developer ...

  4. java的三大特性,封装,继承,多态

    封装 /** * 所谓封装,就是将对象具有的成员变量和成员函数包装和隐藏起来,让外界无法直接使用, * 被封装的成员只能通过某些特定的方式才能访问. * 实现封装有两个步骤: *   1.将不能暴露的 ...

  5. ReactJS基础(续)

    前边的ReactJS基础,我们可以了解到,对于React,可以说是万物皆组件 React的组件应该具有 可组合(Composeable)可重用(Reusable)可维护(Maintainable)的特 ...

  6. HTML5 进阶系列:文件上传下载

    前言 HTML5 中提供的文件API在前端中有着丰富的应用,上传.下载.读取内容等在日常的交互中很常见.而且在各个浏览器的兼容也比较好,包括移动端,除了 IE 只支持 IE10 以上的版本.想要更好地 ...

  7. xUtils使用详细介绍

    xUtils3使用详解 一.xUtils简介: xUtils是基于Afinal开发的目前功能比较完善的一个Android开源框架,官网:https://github.com/wyouflf/xUtil ...

  8. Unity 游戏框架搭建 (二) 单例的模板

      上一篇文章中说到的manager of managers,其中每个manager都是单例的实现,当然也可以使用静态类实现,但是相比于静态类的实现,单例的实现更为通用,可以适用大多数情况. 如何设计 ...

  9. How to install MySQL on CentOS

    1)chekc centos中是否安装了MySQL [root@localhost MySQL]# rpm -qa | grep mariadb mariadb-libs-5.5.52-1.el7.x ...

  10. gradle的安装,配置,构建,研究,初体验......(入职一周研究的第一个大知识点)

    (1)Gradle是一个基于Apache Ant和Apache Maven概念的项目自动化构建工具.它使用一种基于Groovy的特定领域语言(DSL)来声明项目设置,抛弃了基于XML的各种繁琐配置.更 ...