美文网首页
Goroutine同步

Goroutine同步

作者: 高稚商de菌 | 来源:发表于2018-03-08 00:09 被阅读0次

在goroutine执行的过程中,需要进行goroutine的同步。Go语言提供了sync包和channel机制来解决goroutine的同步问题

1. sync.WaitGroup

// 代码来自官方文档
package main
import (
    "fmt"
    "io/ioutil"
    "sync"
    "net/http"
)

var wg sync.WaitGroup
var urls = []string{
    "http://www.golang.org/",
    "http://www.google.com/",
    "http://www.somestupidname.com/",
}
func main() {
    for _, url := range urls {
        // Increment the WaitGroup counter.
        wg.Add(1)
        // Launch a goroutine to fetch the URL.
        go func(url string) {
            // Decrement the counter when the goroutine completes.
            defer wg.Done()
            // Fetch the URL.
            resp, err := http.Get(url)
            if err != nil {
                fmt.Println("Error is: ", err)
                return
            }
            defer resp.Body.Close()
            body, err := ioutil.ReadAll(resp.Body)
            fmt.Println(string(body))
        }(url)
    }
    // Wait for all HTTP fetches to complete.
    wg.Wait()
}
<html>
<body>
Some stupid name....
Error is:  Get http://www.google.com/: dial tcp 75.126.2.43:80: getsockopt: connection refused
Error is:  Get http://www.golang.org/: dial tcp 216.239.37.1:80: getsockopt: connection refused
// 因为没有翻墙。。。

2. chan

package main
import (
    "fmt"
    "io/ioutil"
    "net/http"
)

var urls = []string{
    "http://www.golang.org/",
    "http://www.google.com/",
    "http://www.somestupidname.com/",
}
var ch = make(chan string, len(urls))
func main() {
    for _, url := range urls {
        go func(url string) {
            resp, err := http.Get(url)
            if err != nil {
                str := "Error is: " + err.Error() 
                ch <- str
                return
            }
            defer resp.Body.Close()
            body, err := ioutil.ReadAll(resp.Body)
            ch <- string(body)
        }(url)
    }
    for i := 0; i < len(urls); i++ {
        str := <-ch
        fmt.Println(str)
    }
}
<html>
<body>
Some stupid name....
Error is: Get http://www.google.com/: dial tcp 173.252.102.16:80: getsockopt: connection refused
Error is: Get http://www.golang.org/: dial tcp 216.239.37.1:80: getsockopt: connection refused

channel是一种 阻塞管道 , 是自动阻塞 的. 如果 channel 满了, 对channel放入数据的操作就会阻塞, 直到有某个routine从channel中取出数据, 这个放入数据的操作才会执行. 相反同理, 如果管道是空的, 一个从channel取出数据的操作就会阻塞,直到某个routine向这个channel中放入数据, 这个取出数据的操作才会执行(原理非常类似阻塞型socket的读写缓冲区)

如果channel已经close了, 在进行读写操作会怎么样呢?
总结:多数情况下建议使用channel。处理比较简单的情况时,可以选择WaitGroup,但是处理比较复杂的逻辑时,应该使用channel。

相关文章

  • Goroutine同步

    在goroutine执行的过程中,需要进行goroutine的同步。Go语言提供了sync包和channel机制来...

  • Channel

    Channel是在goroutine之间同步的方式 定义 ChannelType = ( "chan" | "ch...

  • 使用waitgroup同步goroutine

  • Go 语言极速入门9 - Channel

    当一个资源需要在 goroutine 之间共享时,通道在 goroutine 之间架起了一个管道,并提供了确保同步...

  • 22-Go语言管道(Channel)

    多线程同步问题 互斥锁互斥锁的本质是当一个goroutine访问的时候, 其它goroutine都不能访问这样就能...

  • 21-Channel

    多线程同步问题 互斥锁互斥锁的本质是当一个goroutine访问的时候, 其它goroutine都不能访问这样就能...

  • Go 语言学习笔记-Goroutine和channel

    Goroutine 什么是 Goroutine Goroutine 是 Go 并行设计的核心。Goroutine ...

  • 视频笔记二:RPC 和线程

    为什么用 Go 语法先进。在语言层面支持线程(goroutine)和管道(channel)。对线程间的加锁、同步支...

  • golang中channel的用法

    channel有点类似于管道,它在goroutine同步与通信中,有着起承转合的作用,同时也是Golang实现CS...

  • go语言的channel(管道)

    为什么需要channel? 前面使用全局变量加锁同步来解决goroutine的通讯,但不完美。 (1)主线程在等待...

网友评论

      本文标题:Goroutine同步

      本文链接:https://www.haomeiwen.com/subject/rcwsfftx.html