登录
首页 >  Golang >  Go问答

为什么传入通道的值没有被 fmt.Println 打印出来?

来源:stackoverflow

时间:2024-03-30 13:18:32 419浏览 收藏

编程并不是一个机械性的工作,而是需要有思考,有创新的工作,语法是固定的,但解决问题的思路则是依靠人的思维,这就需要我们坚持学习和更新自己的知识。今天golang学习网就整理分享《为什么传入通道的值没有被 fmt.Println 打印出来?》,文章讲解的知识点主要包括,如果你对Golang方面的知识点感兴趣,就不要错过golang学习网,在这可以对大家的知识积累有所帮助,助力开发能力的提升。

问题内容

func main() {
    links := []string{
        "http://google.com",
        "http://amazon.com",
        "http://golang.org",
        "http://yahoo.com",
        "http://ebay.com",
    }

    c := make(chan string)

    for _, link := range links {
        testRequest(link, c)
    }
    msg := <-c
    fmt.Println(msg)
}
func testRequest(s string, c chan string) {
    _, err := http.Get(s)
    if err != nil {
        fmt.Println(s, "Is down presently")
        c <- "Something might be down"
        return
    }
    fmt.Println(s, "Is working perfectly")
    c <- "Working great!"
}

通道应该是引用类型,由于某种原因,该函数没有向通道传递任何内容,因为每次我运行代码时,它都会按预期打印出第一行,但程序不会停止运行,而且不执行 fmt.println(channel) 所以我的假设是没有值按预期传递到通道,这有什么原因吗?


解决方案


你的程序在第一个循环后卡住了,发送者尝试发送到频道,但接收者只能在循环完成后才能接收

//this loop will blocks until finished
for _, link := range links {
    testrequest(link, c) // try to send, receiver can't receive
}

//until above loop finishs, this never excutes
msg := <-c
mt.println(msg)

使用 go 运行发送者循环,以便接收者循环可以执行,但您需要防止主程序退出,直到所有工作完成。

func main() {
    links := []string{
        "http://google.com",
        "http://amazon.com",
        "http://yahoo.com",
        "http://ebay.com",
    }

    c := make(chan string)

    // this code will not blocks
    go func() {
        for _, link := range links {
            testrequest(link, c)
        }
    }()

    // above code doesn't block, this code can excutes
    for {
        msg := <-c
        fmt.println(msg)
    }
}

func testrequest(s string, c chan string) {
    _, err := http.get(s)
    if err != nil {
        fmt.println(s, "is down presently")
        c <- "something might be down"
        return
    }
    fmt.println(s, "is working perfectly")
    c <- "working great!"
}

对于无缓冲通道,在 main() 中从通道读取数据之前,testrequests() 中的写入将被阻塞。你陷入了僵局。一般来说,你应该得到一个错误,通常要找出所有 goroutine 何时被阻塞。不知道为什么你不这样做。

您可能想在不同的 goroutine 中运行 testrequests():

package main

import (
    "fmt"
    "net/http"
)

func main() {
    links := []string{
            "http://google.com",
            "http://amazon.com",
            "http://golang.org",
            "http://yahoo.com",
            "http://ebay.com",
    }

    c := make(chan string)

    go func() {
            for _, link := range links {
                    testRequest(link, c)
            }
            close(c)
    }()
    for msg := range c {
            fmt.Println(msg)
    }
    fmt.Println("Done handling request")
}

func testRequest(s string, c chan string) {
    _, err := http.Get(s)
    if err != nil {
            fmt.Println(s, "Is down presently")
            c <- "Something might be down"
            return
    }
    fmt.Println(s, "Is working perfectly")
    c <- "Working great!"
}

https://play.golang.org/p/g9X1h_NNJAB

文中关于的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《为什么传入通道的值没有被 fmt.Println 打印出来?》文章吧,也可关注golang学习网公众号了解相关技术文章。

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>