Golang 中的 条件变量(sync.Cond)详解

本篇文章面向的读者: 已经基本掌握Go中的 协程(goroutine)通道(channel)互斥锁(sync.Mutex)读写锁(sync.RWMutex) 这些知识。如果对这些还不太懂,可以先回去把这几个知识点解决了。

首先理解以下三点再进入正题:

  • Go中的一个协程 可以理解成一个独立的人,多个协程是多个独立的人
  • 多个协程都需要访问的 共享资源(比如共享变量) 可以理解成 多人要用的某种公共社会资源
  • 上锁 其实就是加入到某个共享资源的争抢组中上锁完成 就是从争抢组中被选出,得到了期待的共享资源;解锁 就是退出某个共享资源的争抢组

假如有这样一个现实场景:在一个公园中有一个公共厕所,这个厕所一次只能容纳一个人上厕所,同时这个厕所中有个放卷纸的位置,其一次只能放一卷纸,一卷纸的总长度是 5 米,而每个人上一次厕所需要用掉 1 米的纸。而当一卷纸用完后,公园管理员要负责给厕所加上一卷新纸,以便大家可以继续使用厕所。 那么对于这个单人公共厕所,大家只能排队上厕所,当每个人进到厕所的时候,当然会把厕所门锁好,以便任何人都进不来(包括管理员)。管理员若要进到厕所查看用纸情况并加卷纸,也需要排队(因为插队总是不文明对吧)。

那么怎么用 Golang 去模拟上述场景呢?

首先我们先不用 sync.Cond,看如何实现?那么请看下面这段代码:

package main

import (
    "fmt"
    "time"
    "sync"
)

var 卷纸 int
var m sync.Mutex
var wg sync.WaitGroup

func 上厕所(姓名 string){
    m.Lock()
    defer func(){
        m.Unlock()
        wg.Done()
    }()
    fmt.Printf("%s 进到厕所\t",姓名)
    if 卷纸 >= 1 {  // 进到厕所第一件事是看还有没有纸
        fmt.Printf("正在拉屎中...\n")
        time.Sleep(time.Second)
        卷纸 -= 1
        fmt.Printf("%s 已用完厕所,正在离开\n",姓名)
        return
    }
    fmt.Printf("发现纸用完了,无奈先离开厕所\n")
}

func 加厕纸(){
    m.Lock()
    defer func(){
        m.Unlock()
        wg.Done()
    }()
    fmt.Printf("公园管理员 进到厕所\t")
    if 卷纸 <= 0 { // 管理员进到厕所是看纸有没有用完
        fmt.Printf("公园管理员 正在加新纸...\n")
        time.Sleep(time.Millisecond*500)
        卷纸 = 5
        fmt.Printf("公园管理员 已加上新厕纸,正在离开\n")
    }else{
        fmt.Printf("发现纸还没用完,先离开厕所\n")
    }
}

func main() {
    卷纸 = 5 // 厕所一开始就准备好了一卷纸,长度5米
    要排队上厕所的人 := [...]string{"老王","小李","老张","小刘","阿明","欣欣","西西","芳芳"}
    for _,谁 := range 要排队上厕所的人 {
        wg.Add(1)
        go 上厕所(谁)
    }
    wg.Add(1)
    go 加厕纸()
    wg.Wait()
}

/*
输出(由于协程执行顺序的不可预测性,因此每次输出的顺序都可能不一样):

公园管理员 进到厕所     发现纸还没用完,先离开厕所
阿明 进到厕所   正在拉屎中...
阿明 已用完厕所,正在离开
老王 进到厕所   正在拉屎中...
老王 已用完厕所,正在离开
小刘 进到厕所   正在拉屎中...
小刘 已用完厕所,正在离开
小李 进到厕所   正在拉屎中...
小李 已用完厕所,正在离开
老张 进到厕所   正在拉屎中...
老张 已用完厕所,正在离开
欣欣 进到厕所   发现纸用完了,无奈先离开厕所
芳芳 进到厕所   发现纸用完了,无奈先离开厕所
西西 进到厕所   发现纸用完了,无奈先离开厕所
*/

上面的代码已经能看出一些效果,但还是有问题:最后三个人因为厕纸用完,都直接离开厕所后就没有后续了?应该是他们离开厕所后再次尝试排队,直到需求解决,就离开厕所不再参与排队了,否则要不断去排队上厕所。而公园管理员呢,他要一直去排队进到厕所里看还有没有纸,而不是看一次就再也不管了。 那么请看下面的完善代码:

package main

import (
    "fmt"
    "sync"
    "time"
)

var (
    卷纸    int
    m     sync.Mutex
    wg    sync.WaitGroup
    厕所的排队 chan string
)

func 上厕所(姓名 string) {
    m.Lock() // 该语句的调用只说明本执行体(可理解成该姓名所指的那个人)加入到了厕所资源的争抢组中;
             // 而该语句的完成调用,才代表了从争抢组中脱颖而出,抢到了厕所;在完成调用之前,会一直阻塞在这里(可理解为这个人正在争抢中)
    defer func() {
        m.Unlock()
        wg.Done()
    }()
    fmt.Printf("%s 进到厕所\t", 姓名)
    if 卷纸 >= 1 { // 进到厕所第一件事是看还有没有纸
        fmt.Printf("正在拉屎中...\n")
        time.Sleep(time.Second)
        卷纸 -= 1
        fmt.Printf("%s 已用完厕所,正在离开\n", 姓名)
        return
    }
    fmt.Printf("发现纸用完了,无奈先离开厕所\n")
    厕所的排队 <- 姓名 // 再次加入厕所排队,期望下次可以成功如厕
}

func 加厕纸() {
    m.Lock()
    defer m.Unlock()
    fmt.Printf("公园管理员 进到厕所\t")
    if 卷纸 <= 0 { // 管理员进到厕所是看纸有没有用完
        fmt.Printf("公园管理员 正在加新纸...\n")
        time.Sleep(time.Millisecond * 500)
        卷纸 = 5
        fmt.Printf("公园管理员 已加上新厕纸,正在离开\n")
    } else {
        fmt.Printf("发现纸还没用完,先离开厕所\n")
    }
}

func main() {
    卷纸 = 5                                                                // 厕所一开始就准备好了一卷纸,长度5米
    要上厕所的人 := [...]string{"老王", "小李", "老张", "小刘", "阿明", "欣欣", "西西", "芳芳"} // 这里只是举几个人名例子,假设此处有源源不断的人去上厕所(读者可以随意改造人名来源)
    厕所的排队 = make(chan string, len(要上厕所的人))
    for _, 谁 := range 要上厕所的人 {
        厕所的排队 <- 谁
    }
    go func() { // 在这个执行体中,会不断从 厕所排队 中把人加入到 对厕所资源的争抢组中
        for 谁 := range 厕所的排队 {
            wg.Add(1)
            go 上厕所(谁)
        }
    }()
    wg.Add(1)
    go func() { // 在这个执行体中,代表公园管理员的个人时间线,他会每隔一段时间去加入争抢组进到厕所,检查纸还有没有
        for {
            time.Sleep(time.Millisecond * 1200)
            加厕纸()
        }
    }()
    wg.Wait()
}

/*
输出:

老王 进到厕所   正在拉屎中...
老王 已用完厕所,正在离开
芳芳 进到厕所   正在拉屎中...
芳芳 已用完厕所,正在离开
阿明 进到厕所   正在拉屎中...
阿明 已用完厕所,正在离开
小刘 进到厕所   正在拉屎中...
小刘 已用完厕所,正在离开
欣欣 进到厕所   正在拉屎中...
欣欣 已用完厕所,正在离开
小李 进到厕所   发现纸用完了,无奈先离开厕所
老张 进到厕所   发现纸用完了,无奈先离开厕所
西西 进到厕所   发现纸用完了,无奈先离开厕所
公园管理员 进到厕所     公园管理员 正在加新纸...
公园管理员 已加上新厕纸,正在离开
西西 进到厕所   正在拉屎中...
西西 已用完厕所,正在离开
小李 进到厕所   正在拉屎中...
小李 已用完厕所,正在离开
老张 进到厕所   正在拉屎中...
老张 已用完厕所,正在离开
公园管理员 进到厕所     发现纸还没用完,先离开厕所
公园管理员 进到厕所     发现纸还没用完,先离开厕所
公园管理员 进到厕所     发现纸还没用完,先离开厕所
*/

上面这个代码在功能上基本是完善了,成功模拟了上述 多人上公厕 的场景。但仔细一想,这个场景其实有些地方是不合常理的:如果有个人进到厕所发现没纸,难道他会出来紧接着再去排队吗?如果排了三次五次甚至十次还是没有纸,还要这样不断地反复排队进去出来又排队?而公园管理员,要是这样不断反复排队进厕所查看,那么他这一天其他啥事都干不了。

所以更合理实际的情况应该是:如果一个人进到厕所发现没纸,他应该先去在旁边歇着或在附近干别的,当公园管理员加完纸后,会通过喇叭吆喝一声:“新纸已加上”。这样,附近所有因为没厕纸而歇着的人就会听到这个通知,此时,他们再去尝试排队进厕所;而公园管理员也不用不断去排队进厕所检查纸用完了没有,因为经过升级,厕所加装了一个功能,有一个纸用尽的报警按钮装在纸盒旁边,当上完厕所的人发现纸用完的时候,他会先按下这个报警按钮,再离开厕所。这个报警的声音在整个公园的各处都可以听到,所以管理员无论在哪里干啥,他都能收到这个纸用尽的报警信号,然后他才去进厕所加纸。

其实这种被动通知的模式就是 sync.Cond 的核心思想,它会减少资源消耗,达到更优的效果,下面就是改良为 sync.Cond 的实现代码:

package main

import (
    "fmt"
    "math"
    "strconv"
    "sync"
    "time"
)

var (
    卷纸   int
    m    sync.Mutex
    cond = sync.NewCond(&m)
)

func 上厕所(姓名 string) {
    m.Lock() // 该语句的调用只说明本执行体(可理解成该姓名所指的那个人)加入到了厕所资源的争抢组中;
             // 而该语句的完成调用,才代表了从争抢组中脱颖而出,抢到了厕所;在完成调用之前,会一直阻塞在这里(可理解为这个人正在争抢中)
    defer m.Unlock()
    fmt.Printf("%s 进到厕所\t", 姓名)
    for 卷纸 < 1 { // 进到厕所第一件事是看还有没有纸
        fmt.Printf("发现纸用完了,先离开厕所在附近歇息等待信号\n")
        cond.Wait() // 该语句的调用 相当于调用了 m.Unlock() 也就是退出了争抢组,而是先歇着等待纸加上的信号;
                    // 当收到纸加上的信号后,该语句会自动执行 m.Lock(),也就是会重新加入到厕所的争抢组中;
                    // 该语句的完成调用说明已经再次成功争抢到了厕所;
        fmt.Printf("%s 等到了厕纸已加的信号,并去再次抢到了厕所\t", 姓名)
    }
    fmt.Printf("正在拉屎中...\n")
    time.Sleep(time.Second)
    卷纸 -= 1
    fmt.Printf("%s 已用完厕所\t", 姓名)
    if 卷纸 < 1 { // 注意这里:在他用完厕所离开前,他需要看是不是纸已经用完了,如果用完了,就按下纸用尽的报警按钮,给公园管理员发送信号
        cond.Broadcast() // 想想,这里为什么不用 Signal() ?因为 Signal 只能通知到一个等待者,这样就有可能通知不到 公园管理员。可以试着把这里换成 Signal() 试下
        fmt.Printf("发现厕纸已用完,并按下了报警\t")
    }
    fmt.Printf("正在离开厕所\n")
}

func 加厕纸() {
    m.Lock()
    defer m.Unlock()
    fmt.Printf("公园管理员 进到厕所\t")
    for 卷纸 > 0 { // 管理员进到厕所是看纸有没有用完
        fmt.Printf("发现纸还没用完,先离开厕所在等纸用尽的报警消息\n")
        cond.Wait() // 如果纸没用完,就先去干其他工作,等纸用尽的报警消息
        fmt.Printf("公园管理员 等到了纸用尽的报警消息,并再次抢到了厕所\n")
    }
    fmt.Printf("公园管理员 正在加新纸...\n")
    time.Sleep(time.Millisecond * 500)
    卷纸 = 5
    cond.Broadcast() // 注意:公园管理员加完新纸后,要通过喇叭喊一声 “纸已加上” 的消息通知所有 因没纸而等待上厕所的人
    fmt.Printf("公园管理员 已加上新厕纸,并通过喇叭通知了该消息,并正在离开厕所\n")
}

func main() {
    卷纸 = 5  // 厕所一开始就准备好了一卷纸,长度5米
    要上厕所的人 := [...]string{"老王", "小李", "老张", "小刘", "阿明", "欣欣", "西西", "芳芳"} // 上厕所的人名模板
    go func() { // 在这个执行体中,代表厕所及厕所队列的时间线,厕所永远运营下去
        for i := 0; i < math.MaxInt; i++ { // 此循环通过编号加上上面的姓名模板来 创建源源不断 上厕所的人
            for _, 人名模板 := range 要上厕所的人 {
                谁 := 人名模板 + strconv.Itoa(i)
                go 上厕所(谁)
                time.Sleep(time.Millisecond * 500) // 平均每半秒有一个人去上厕所
            }
            fmt.Printf("\n====================>> 屏幕停止输出后,请按Enter键继续 <<====================\n\n")
            fmt.Scanln()
        }
    }()
    go func() { // 在这个执行体中,代表公园管理员的个人时间线,管理员永不退休
        for {
            // 注意:相比上个版本,此处不用再加 Sleep 函数了,因为 加厕纸() 函数中的 cond.Wait() 会在有纸的时候等待信号
            加厕纸()
        }
    }()
    end := make(chan bool)
    <-end
}

/*
输出:

公园管理员 进到厕所     发现纸还没用完,先离开厕所在等纸用尽的报警消息
老王0 进到厕所  正在拉屎中...
老王0 已用完厕所        正在离开厕所
小李0 进到厕所  正在拉屎中...
小李0 已用完厕所        正在离开厕所
老张0 进到厕所  正在拉屎中...
老张0 已用完厕所        正在离开厕所
小刘0 进到厕所  正在拉屎中...
小刘0 已用完厕所        正在离开厕所
阿明0 进到厕所  正在拉屎中...

====================>> 屏幕停止输出后,请按Enter键继续 <<====================

阿明0 已用完厕所        发现厕纸已用完,并按下了报警    正在离开厕所
欣欣0 进到厕所  发现纸用完了,先离开厕所在附近歇息等待信号
西西0 进到厕所  发现纸用完了,先离开厕所在附近歇息等待信号
芳芳0 进到厕所  发现纸用完了,先离开厕所在附近歇息等待信号
公园管理员 等到了纸用尽的报警消息,并再次抢到了厕所
公园管理员 正在加新纸...
公园管理员 已加上新厕纸,并通过喇叭通知了该消息,并正在离开厕所
公园管理员 进到厕所     发现纸还没用完,先离开厕所在等纸用尽的报警消息
欣欣0 等到了厕纸已加的信号,并去再次抢到了厕所  正在拉屎中...
欣欣0 已用完厕所        正在离开厕所
芳芳0 等到了厕纸已加的信号,并去再次抢到了厕所  正在拉屎中...
芳芳0 已用完厕所        正在离开厕所
西西0 等到了厕纸已加的信号,并去再次抢到了厕所  正在拉屎中...
西西0 已用完厕所        正在离开厕所

老王1 进到厕所  正在拉屎中...
老王1 已用完厕所        正在离开厕所
小李1 进到厕所  正在拉屎中...
小李1 已用完厕所        发现厕纸已用完,并按下了报警    正在离开厕所
老张1 进到厕所  发现纸用完了,先离开厕所在附近歇息等待信号
公园管理员 等到了纸用尽的报警消息,并再次抢到了厕所
公园管理员 正在加新纸...
公园管理员 已加上新厕纸,并通过喇叭通知了该消息,并正在离开厕所
公园管理员 进到厕所     发现纸还没用完,先离开厕所在等纸用尽的报警消息
小刘1 进到厕所  正在拉屎中...
小刘1 已用完厕所        正在离开厕所
阿明1 进到厕所  正在拉屎中...

====================>> 屏幕停止输出后,请按Enter键继续 <<====================
*/

用了 sync.Cond 的代码显然要精简了很多,而且还节省了计算资源,只会在收到通知的时候 才去抢公共厕所,而不是不断地反复去抢公共厕所。通过这个对现实场景的模拟,我们就很容易从使用者的角度理解 sync.Cond 是什么,它的字面意思就是 “条件”,这就已经点出了这东西的核心要义,就是满足条件才执行,条件是什么,信号其实就是条件,当一个执行体收到信号之后,它才去争抢共享资源,否则就会挂起等待(这种等待底层其实会让出线程,所以这种等待并不会空耗资源),比起不断轮寻去抢资源,这种方式要节省得多。

最后留给读者一个思考的问题:就是上面最后一版的代码,为什么 当纸用完后按报警按钮通知 公园管理员 要用 sync.Broadcast() 方法去广播通知?不是只通知管理员一个人吗,单独通知他不就行了,用 sync.Signal() 为什么不行?

到此这篇关于Golang 中的 条件变量(sync.Cond)详解的文章就介绍到这了,更多相关Golang 中的 条件变量(sync.Cond)内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Go语言Http调用之Post请求详解

    目录 前言 POST 请求 小结 前言 上篇文章 Go HTTP 调用(上) 介绍了如何进行 HTTP 调用,并通过 GET 请求的例子,讲述了 query 参数和 header 参数如何设置,以及响应体的获取方法. 本文继上文,接下来会通过 POST 请求,对其他参数的设置进行介绍. POST 请求 发起 HTTP POST 请求时,携带 json 格式的 body 参数是最常见的,这是因为 json 格式的参数可读性好,对于层级结构较为复杂的数据也能应对,并且这符合 RestFul API

  • Go简单实现协程方法

    目录 为什么需要协程 协程的本质 协程如何在线程中执行 GMP调度模型 协程并发 为什么需要协程 协程的本质是将一段数据的运行状态进行打包,可以在线程之间调度,所以协程就是在单线程的环境下实现的应用程序级别的并发,就是把本来由操作系统控制的切换+保存状态在应用程序里面实现了. 所以我们需要协程的目的其实就是它更加节省资源.可以在有限的资源内支持更高的并发,体现在以下三个方面: 资源利用:程可以利用任何的线程去运行,不需要等待CPU的调度. 快速调度:协程可以快速地调度(避开了系统调用和切换),快

  • Go map发生内存泄漏解决方法

    目录 正文 hamp 结构体代码 查看占用的内存数量 对于 map 内存泄漏的解法 正文 Go 程序运行时,有些场景下会导致进程进入某个“高点”,然后就再也下不来了. 比如,多年前曹大写过的一篇文章讲过,在做活动时线上涌入的大流量把 goroutine 数抬升了不少,流量恢复之后 goroutine 数也没降下来,导致 GC 的压力升高,总体的 CPU 消耗也较平时上升了 2 个点左右. 有一个 issue 讨论为什么 allgs(runtime 中存储所有 goroutine 的一个全局 sl

  • 详解Go语言如何进行Http调用

    目录 前言 前置知识 GET 请求 小结 前言 无论是微服务还是单体架构等,服务间都有相互通信的时候,而最直接的通信方法就是 HTTP 调用,本文将会介绍在 Go 语言里,如何进行 HTTP 调用,并举例说明. 前置知识 HTTP 调用需要通过 http 包里的 Client 结构体里的 Do 方法去实现,因此需要先声明一个 Client 结构体变量,该结构体可以设置超时时间等配置. 对于一个请求里的 URL,查询参数,请求 method 等参数,需要 http 包里的 Request 结构体去

  • Vue路由跳转方式区别汇总(push,replace,go)

    目录 声明式导航router-link 1. 不带参数 2.带参数 编程式导航 1.this.$router.push 2.this.$router.replace 3.this.$router.go(n) 总结区别: 在浏览器中,点击链接实现导航的方式,叫做声明式导航.例如:普通网页中点击 a标签链接.vue项目中点击router-link标签链接都属于声明式导航.在浏览器中,调用API方法实现导航的方式,叫做编程式导航.例如:普通网页中调用location.href跳转到新页面的方式,属于编

  • Golang 中的 条件变量(sync.Cond)详解

    本篇文章面向的读者: 已经基本掌握Go中的 协程(goroutine),通道(channel),互斥锁(sync.Mutex),读写锁(sync.RWMutex) 这些知识.如果对这些还不太懂,可以先回去把这几个知识点解决了. 首先理解以下三点再进入正题: Go中的一个协程 可以理解成一个独立的人,多个协程是多个独立的人 多个协程都需要访问的 共享资源(比如共享变量) 可以理解成 多人要用的某种公共社会资源 上锁 其实就是加入到某个共享资源的争抢组中:上锁完成 就是从争抢组中被选出,得到了期待的

  • Golang 中的 unsafe.Pointer 和 uintptr详解

    目录 前言 uintptr unsafe.Pointer 使用姿势 常规类型互转 Pointer => uintptr 指针算数计算:Pointer => uintptr => Pointer reflect 包中从 uintptr => Ptr 实战案例 string vs []byte sync.Pool 前言 日常开发中经常看到大佬们用各种 unsafe.Pointer, uintptr 搞各种花活,作为小白一看到 unsafe 就发憷,不了解二者的区别和场景,自然心里没数.

  • Golang中的错误处理的示例详解

    目录 1.panic 2.包装错误 3.错误类型判断 4.错误值判断 1.panic 当我们执行panic的时候会结束下面的流程: package main import "fmt" func main() { fmt.Println("hello") panic("stop") fmt.Println("world") } 输出: go run 9.go hellopanic: stop 但是panic也是可以捕获的,我们可

  • SQL Server查询条件IN中能否使用变量的示例详解

    在SQL Server的查询条件中,能否在IN里面使用变量呢? 如果可以的话,有没有需要注意的地方或一些限制呢?在回答这个问题前,我们先来看看这个例子: IF EXISTS (SELECT 1 FROM sys.objects WHERE name='TEST' AND type='U') BEGIN DROP TABLE TEST; END GO CREATE TABLE TEST ( ID INT, NAME VARCHAR(16) ); GO INSERT INTO dbo.TEST SE

  • golang中为什么不存在三元运算符详解

    三元运算符广泛存在于其他语言中,比如: python: val = trueValue if expr else falseValue javascript: const val = expr ? trueValue : falseValue c.c++: const char *val = expr ? "trueValue" : "falseValue"; 然而,被广泛支持的三目运算符在golang中却是不存在的!如果我们写出类似下面的代码: val := ex

  • Go中的条件语句Switch示例详解

    Switch简介 Go的switch的基本功能和C.Java类似: switch 语句用于基于不同条件执行不同动作,每一个 case 分支都是唯一的,从上至下逐一测试,直到匹配为止. 匹配项后面也不需要再加 break. 特点: switch 默认情况下 case 最后自带 break 语句,匹配成功后就不会执行其他 case 重点介绍Go当中的Switch的两个特别点:** 表达式判断为true还需要执行后面的 case,可以使用 fallthrough type-switch 来判断某个 i

  • Golang中for循环的用法示例详解

    目录 Golang中for循环的用法 for循环 基本语法 注意事项和使用细节 Golang中for循环的用法 for循环 就是让一段代码循环的执行. 基本语法 for循环变量初始化:循环条件:循环变量迭代{ 循环操作(语句) } package main import "fmt" func main(){ for i := 1; i <= 10; i++ { fmt.Println("666",i) } } for循环的四个要素: 1.循环变量初始化 2.循

  • golang中defer的关键特性示例详解

    前言 大家都知道golang的defer关键字,它可以在函数返回前执行一些操作,最常用的就是打开一个资源(例如一个文件.数据库连接等)时就用defer延迟关闭改资源,以免引起内存泄漏.本文主要给大家介绍了关于golang中defer的关键特性,分享出来供大家参考学习,下面话不多说,来一起看看详细的介绍: 一.defer 的作用和执行时机 go 的 defer 语句是用来延迟执行函数的,而且延迟发生在调用函数 return 之后,比如 func a() int { defer b() return

  • Golang中数据结构Queue的实现方法详解

    前言 本文主要给大家介绍了关于Golang中数据结构Queue实现的相关内容,分享出来供大家参考学习,下面话不多说了,来一起看看详细的介绍吧. 需求 队列的特性较为单一,基本操作即初始化.获取大小.添加元素.移除元素等.最重要的特性就是满足先进先出. 实现 接下来还是按照以前的套路,一步一步来分析如何利用Go的语法特性实现Queue这种数据结构. 定义 首先定义每个节点Node结构体,照例Value的值类型可以是任意类型,节点的前后指针域指针类型为node type node struct {

  • 关于Golang中range指针数据的坑详解

    前言 在Golang中使用 for range 语句进行迭代非常的便捷,但在涉及到指针时就得小心一点了. 下面的代码中定义了一个元素类型为 *int 的通道 ch : package main import ( "fmt" ) func main() { ch := make(chan *int, 5) //sender input := []int{1,2,3,4,5} go func(){ for _, v := range input { ch <- &v } cl

随机推荐