goroutine/channel(1)

goroutine/channel(1)
goroutine/channel(1)
goroutine/channel(1)
goroutine/channel(1)
goroutine/channel(1)
goroutine/channel(1)
goroutine/channel(1)

创新互联是专业的网站建设公司,提供网站建设,网站制作,网站设计等网站开发一体化解决方案;包括成都h5网站建设,小程序开发,网站定制,企业网站建设,成都商城网站开发,成都响应式网站建设,建网站,PHP网站建设,软件开发,软文发稿,网站营销。欢迎做网站的企业前来合作洽谈,创新互联将竭诚为您服务!

goroutine/channel(1)

channel可以是各种数据类型:
goroutine/channel(1)
(map类型)
goroutine/channel(1)
(struct类型)
goroutine/channel(1)
(接口类型)
goroutine/channel(1)
goroutine/channel(1)

package main

import "fmt"

type student struct {
    name string
}

func main() {
    var stuChan chan interface{}
    //空interface类型可以是任意类型
    stuChan = make(chan interface{}, 10)
    stu := student{name:"stu01"}
    stuChan <- &stu

    var stu01 interface{}
    stu01 = <- stuChan

    fmt.Println(stu01)

    var stu02 *student
    //将stu01由interface转换为结构体类型指针
    stu02, ok := stu01.(*student)
    if !ok {
        fmt.Println("不可以转换")
        return
    }
    fmt.Println(stu02)

}

输出:
&{stu01}
&{stu01}

Process finished with exit code 0

goroutine/channel(1)
goroutine/channel(1)

goroutine/channel(1)

channel关闭,判断:

package main

import "fmt"

func main() {
    var ch chan int
    ch = make(chan int, 10)

    for i := 0; i < 10; i++ {
        ch <- i
    }

    close(ch)

    for {
        var b int
        //从chan取数据的时候第二个参数用来判断是否成功
        b, ok := <- ch
        if ok == false {
            fmt.Println("chan is closed")
            return
        }
        fmt.Println(b)
    }

}

输出:
0
1
2
3
4
5
6
7
8
9
chan is closed
range遍历chan:

package main

import "fmt"

func main() {
    var ch chan int
    ch = make(chan int, 1000)

    for i := 0; i < 1000; i++ {
        ch <- i
    }

    close(ch)
    for v := range ch {
        fmt.Println(v)
    }
}

关闭chan,goroutine同步案例:

package main

import (
    "fmt"
)

func calc(taskChan chan int, resChan chan int,  exitChan chan bool) {
    for v := range taskChan {
        flag := true
        for i := 2; i < v; i++ {
            if v%i == 0 {
                flag = false
                break
            }
        }

        if flag {
            resChan <- v
        }
    }

    fmt.Println("exit")
    //每个calc的协程(goroutine)结束后写入true到名为exitChan的chan里。代表该goroutine结束了
    exitChan <- true
}

func main() {
    intChan := make(chan int, 1000)
    resultChan := make(chan int, 1000)
    exitChan := make(chan bool, 8)

    go func() {
        for i := 0; i < 10000; i++ {
            intChan <- i
        }

        close(intChan)
    }()

    for i := 0; i < 8; i++ {
        go calc(intChan, resultChan, exitChan)
    }

    //等待所有计算的goroutine全部退出
    go func() {
        for i := 0; i < 8; i++ {
            <-exitChan
            fmt.Println("wait goroute ", i, " exited")
        }
        //8个goroutine结束后关闭resultChan管道
        close(resultChan)
    }()

    //resultChan管道关闭后,range才能在遍历完后结束
    for v := range resultChan {
        fmt.Println(v)
    }
}

再来个同步channel的例子:

package main

import (
    "fmt"
)

func send(ch chan int, exitChan chan bool) {
    for i := 0; i < 10; i++ {
        ch <- i
    }
    close(ch)
    exitChan <- true
}

func recv(ch chan int, exitChan chan bool) {
    for {
        v, ok := <- ch
        if !ok {
            break
        }
        fmt.Println(v)
    }
    exitChan <- true
}

func main() {
    ch := make(chan int, 10)
    exitChan := make(chan bool, 2)

    go send(ch, exitChan)
    go recv(ch, exitChan)

    var total = 0
    for _ = range exitChan {
        total++
        if total == 2 {
            break
        }
    }

}

输出:
0
1
2
3
4
5
6
7
8
9

Process finished with exit code 0
channel的只读,只写(读函数里只准读,写函数里只准写。来进行规范,防止出错):

package main

import "fmt"

func send(ch chan<- int, exitChan chan struct{}) {

    for i := 0; i < 10; i++ {
        ch <- i
    }

    close(ch)
    var a struct{}
    exitChan <- a
}

func recv(ch <-chan int, exitChan chan struct{}) {
    for {
        v, ok := <-ch
        if !ok {
            break
        }

        fmt.Println(v)
    }

    var a struct{}
    exitChan <- a
}

func main() {
    var ch chan int
    ch = make(chan int, 10)
    exitChan := make(chan struct{}, 2)

    go send(ch, exitChan)
    go recv(ch, exitChan)

    var total = 0
    for _ = range exitChan {
        total++
        if total == 2 {
            break
        }
    }
}

当前文章:goroutine/channel(1)
当前路径:http://scyanting.com/article/jdcddo.html