当前位置:首页 > 文章列表 > Golang > Go问答 > 通道传输值并读取输出的准备就绪

通道传输值并读取输出的准备就绪

来源:stackoverflow 2024-02-06 11:18:21 0浏览 收藏

学习Golang要努力,但是不要急!今天的这篇文章《通道传输值并读取输出的准备就绪》将会介绍到等等知识点,如果你想深入学习Golang,可以关注我!我会持续更新相关文章的,希望对大家都能有所帮助!

问题内容

我正在尝试使用 golang 中的两个通道构建接收者和发送者模式。我正在执行一项任务(api 调用),并接收回 response 结构。我的目标是,当收到响应时,我想将其发送到另一个通道(writechan)进行额外处理。

我想连续读取/监听该接收通道(respchan)并处理通过的任何内容(例如response)。然后我想启动一个线程,并在另一个 goroutine 中对该 response 进行进一步的操作。

我想了解如何将此模式链接在一起,以允许数据从 api 调用流出并同时写入(每个响应将被写入 write() 函数处理的单独文件目标。

基本上我当前的模式如下:

package main

import (
    "fmt"
    "sync"
)

func main() {

    var wg sync.WaitGroup
    respChan := make(chan Response) // Response is a struct that contains API response metadata
    defer close(respChan)
    // requests is just a slice of requests to be made to an API
    // This part is working well
    for _, req := range requests {
        wg.Add(1)
        go func(r Request) {
            defer wg.Done()
            resp, _ := r.Get() // Make the API call and receive back a Response struct
            respChan <- resp // Put the response into our channel
        }(req)
    }

    // Now, I want to extract the responses as they become available and send them to another function to do some processing. This I am unsure of how to handle properly
    writeChan := make(chan string)
    defer close(writeChan)
    select {
        case resp := <-respChan: // receive from response channel
            go func(response Response) {
                signal, _ := Write(response) // Separate func to write the response to a file. Not important here in this context.
                writeChan <- signal // Put the signal data into the channel which is a string file path of where the file was written (will be used for a later process)

            }(resp)
        case <-time.After(15 *time.Second):
            fmt.Println("15 seconds have passed without receiving anything...")

    }
    wg.Wait()
}

正确答案


让我与您分享一个您可以从中受益的工作示例。首先,我将展示代码,然后,我将引导您完成所有相关部分。

package main

import (
    "fmt"
    "net/http"
    "os"
    "strings"
    "time"
)

type request struct {
    url            string
    delayinseconds time.duration
}

type response struct {
    url        string
    statuscode int
}

func main() {
    requests := []request{
        {"https://www.google.com", 0},
        {"https://stackoverflow.com", 1},
        {"https://www.wikipedia.com", 4},
    }

    respchan := make(chan response)
    defer close(respchan)

    for _, req := range requests {
        go func(r request) {
            fmt.printf("%q - %v\n", r.url, strings.repeat("#", 30))
            // simulate heavy work
            time.sleep(time.second * r.delayinseconds)
            resp, _ := http.get(r.url)
            res := response{r.url, resp.statuscode}
            fmt.println(time.now())
            respchan <- res
        }(req)
    }

    writechan := make(chan struct{})
    defer close(writechan)

    for i := 0; i < len(requests); i++ {
        select {
        case res := <-respchan:
            go func(r response) {
                f, err := os.create(fmt.sprintf("%v.txt", strings.replace(r.url, "https://", "", 1)))
                if err != nil {
                    panic(err)
                }
                defer f.close()
                f.write([]byte(fmt.sprintf("%q ok with %d\n", r.url, r.statuscode)))
                writechan <- struct{}{}
            }(res)
        case <-time.after(time.second * 2):
            fmt.println("timeout")
        }
    }
}

设置

首先,我定义了示例中将使用的两个结构:requestresponse。在前者中,我放置了一个 delayinseconds 来模拟一些重负载和耗时的操作。然后,我定义了 requests 变量,其中包含必须完成的所有请求。

写作部分

在这里,我检查了 requests 变量。对于每个请求,我将向目标 url 发出 http 请求。 time.sleep 模拟重负载。然后,我将响应写入未缓冲的 respchan 通道。

阅读部分

这里的主要变化是将 select 构造包装到 for 循环中。因此,我们将确保迭代正确的次数(基于 requests 变量的长度)。

最后的注释

首先,请记住,代码过于简单只是为了展示相关部分。因此,缺少许多错误处理,并且一些内联函数可以外推到命名函数中。你不需要使用sync.waitgroup来实现你所需要的,通道的使用就足够了。
随意尝试延迟并检查写入了哪些文件!

如果这对您有帮助,请告诉我!

编辑

根据您的要求,我将根据您的需求为您提供更准确的解决方案。新的阅读部分将类似于以下内容:

count := 0
for {
    // this check is need to exit the for loop and not wait indefinitely
    // it can be removed based on your needs
    if count == 3 {
        fmt.Println("all responses arrived...")
        return
    }
    res := <-respChan
    count++
    go func(r Response) {
        f, err := os.Create(fmt.Sprintf("%v.txt", strings.Replace(r.Url, "https://", "", 1)))
        if err != nil {
            panic(err)
        }
        defer f.Close()
        f.Write([]byte(fmt.Sprintf("%q OK with %d\n", r.Url, r.StatusCode)))
        writeChan <- struct{}{}
    }(res)
}

这里,执行在 for 循环内无限期等待。无论每个请求需要多长时间才能完成,它一到达就会被获取。我在 for 循环的顶部放置了一个 if ,以便在处理完我们需要的请求后退出。但是,您可以避免它并让代码运行直到收到取消信号(这取决于您)。

请告诉我这是否更能满足您的要求,谢谢!

以上就是本文的全部内容了,是否有顺利帮助你解决问题?若是能给你带来学习上的帮助,请大家多多支持golang学习网!更多关于Golang的相关知识,也可关注golang学习网公众号。

版本声明
本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
Go服务器:如何嵌套静态文件提供?Go服务器:如何嵌套静态文件提供?
上一篇
Go服务器:如何嵌套静态文件提供?
如何确定程序是以交互方式运行还是以后台方式运行?
下一篇
如何确定程序是以交互方式运行还是以后台方式运行?
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    542次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    511次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    498次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • 千音漫语:智能声音创作助手,AI配音、音视频翻译一站搞定!
    千音漫语
    千音漫语,北京熠声科技倾力打造的智能声音创作助手,提供AI配音、音视频翻译、语音识别、声音克隆等强大功能,助力有声书制作、视频创作、教育培训等领域,官网:https://qianyin123.com
    138次使用
  • MiniWork:智能高效AI工具平台,一站式工作学习效率解决方案
    MiniWork
    MiniWork是一款智能高效的AI工具平台,专为提升工作与学习效率而设计。整合文本处理、图像生成、营销策划及运营管理等多元AI工具,提供精准智能解决方案,让复杂工作简单高效。
    131次使用
  • NoCode (nocode.cn):零代码构建应用、网站、管理系统,降低开发门槛
    NoCode
    NoCode (nocode.cn)是领先的无代码开发平台,通过拖放、AI对话等简单操作,助您快速创建各类应用、网站与管理系统。无需编程知识,轻松实现个人生活、商业经营、企业管理多场景需求,大幅降低开发门槛,高效低成本。
    147次使用
  • 达医智影:阿里巴巴达摩院医疗AI影像早筛平台,CT一扫多筛癌症急慢病
    达医智影
    达医智影,阿里巴巴达摩院医疗AI创新力作。全球率先利用平扫CT实现“一扫多筛”,仅一次CT扫描即可高效识别多种癌症、急症及慢病,为疾病早期发现提供智能、精准的AI影像早筛解决方案。
    140次使用
  • 智慧芽Eureka:更懂技术创新的AI Agent平台,助力研发效率飞跃
    智慧芽Eureka
    智慧芽Eureka,专为技术创新打造的AI Agent平台。深度理解专利、研发、生物医药、材料、科创等复杂场景,通过专家级AI Agent精准执行任务,智能化工作流解放70%生产力,让您专注核心创新。
    147次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码