zoukankan      html  css  js  c++  java
  • golang 线程与通道

    首先我们来看线程,在golang里面也叫goroutine

    在读这篇文章之前,我们需要了解一下并发与并行。golang的线程是一种并发机制,而不是并行。它们之间的区别大家可以上网搜一下,网上有很多的介绍。

    下面我们先来看一个例子吧

    import(

             "fmt"

    )

    funcmain(){

        go fmt.Println("1")
        fmt.Println("2")    
    }

    在golang里面,使用go这个关键字,后面再跟上一个函数就可以创建一个线程。后面的这个函数可以是已经写好的函数,也可以是一个匿名函数

    funcmain(){

            vari=3

        go func(a int) {
            fmt.Println(a)
            fmt.Println("1")
        }(i)
        fmt.Println("2")
      
    }
    
    

    上面的代码就创建了一个匿名函数,并且还传入了一个参数i,下面括号里的i是实参,a是形参。

    那么上面的代码能按照我们预想的打印1、2、3吗?告诉你们吧,不能,程序只能打印出2。下面我把正确的代码贴出来吧

    import(

        "fmt"
        "time"    
    )

    funcmain(){

        var i = 3
        go func(a int) {
            fmt.Println(a)
            fmt.Println("1")
        }(i)
        fmt.Println("2")
        time.Sleep(1 * time.Second)
    
    }
    
    
    我只是在最后加了一行让主线程休眠一秒的代码,程序就会依次打印出2、3、1。

    那为什么会这样呢?因为程序会优先执行主线程,主线程执行完成后,程序会立即退出,没有多余的时间去执行子线程。如果在程序的最后让主线程休眠1秒钟,那程序就会有足够的时间去执行子线程。


    线程先讲到这里,下面我们来看看通道吧。

    通道又叫channel,顾名思义,channel的作用就是在多线程之间传递数据的。

    创建无缓冲channel

    chreadandwrite :=make(chan int)

    chonlyread := make(<-chan int)  //创建只读channel
    chonlywrite := make(chan<- int) //创建只写channel

    下面我们来看一个例子:

            ch :=make(chan int)     

        ch <- 1
    
          go func() {
            <-ch
            fmt.Println("1")
          }()
          fmt.Println("2"
        

    这段代码执行时会出现一个错误: :=make(chan int,1)

        ch <- 1
        go func() {
            v := <-ch
            fmt.Println(v)
        }()
        time.Sleep(1 * time.Second)
        fmt.Println("2")

    这样的话程序就会依次打印出1、2

    2、把ch<-1这一行代码放到子线程代码的后面,代码如下:

           ch :=make(chan int)

        go func() {
            v := <-ch
            fmt.Println(v)
        }()
        ch <- 1
        fmt.Println("2")

    这里就不用让主线程休眠了,因为channel在主线程中被赋值后,主线程就会阻塞,直到channel的值在子线程中被取出。


    最后我们看一个生产者和消费者的例子:

    import (

        "fmt"
        "time"
    )
    func produce(p chan<- int) {
        for i := 0; i < 10; i++ {
            p <- i
            fmt.Println("send:", i)
        }
    }
    func consumer(c <-chan int) {
        for i := 0; i < 10; i++ {
            v := <-c
            fmt.Println("receive:", v)
        }
    }
    func main() {
        ch := make(chan int)
        go produce(ch)
        go consumer(ch)
        time.Sleep(1 * time.Second)
    }

    在这段代码中,因为channel是没有缓冲的,所以当生产者给channel赋值后,生产者这个线程会阻塞,直到消费者线程将channel中的数据取出。消费者第一次将数据取出后,进行下一次循环时,消费者的线程也会阻塞,因为生产者还没有将数据存入,这时程序会去执行生产者的线程。程序就这样在消费者和生产者两个线程间不断切换,直到循环结束。


    下面我们再看一个带缓冲的例子:

    import (

        "fmt"
        "time"
    )
    func produce(p chan<- int) {
        for i := 0; i < 10; i++ {
            p <- i
            fmt.Println("send:", i)
        }
    }
    func consumer(c <-chan int) {
        for i := 0; i < 10; i++ {
            v := <-c
            fmt.Println("receive:", v)
        }
    }
    func main() {
        ch := make(chan int, 10)
        go produce(ch)
        go consumer(ch)
        time.Sleep(1 * time.Second)
    }

    在这个程序中,缓冲区可以存储10个int类型的整数,在执行生产者线程的时候,线程就不会阻塞,一次性将10个整数存入channel,在读取的时候,也是一次性读取。

  • 相关阅读:
    《数据结构》张明瑞 清华大学 计算机科学与技术专业 大二
    Android studio 快捷键
    android下的样式
    Android照片墙应用实现,再多的图片也不怕崩溃
    菜单
    Android下拉刷新完全解析,教你如何一分钟实现下拉刷新功能
    自动文本提示控件
    Android高效加载大图、多图解决方案,有效避免程序OOM
    notification+service实现消息推送
    常见对话框
  • 原文地址:https://www.cnblogs.com/navysummer/p/8458620.html
Copyright © 2011-2022 走看看