当前位置:首页 > 文章列表 > Golang > Go问答 > 正确监测长时间运行任务的进展

正确监测长时间运行任务的进展

来源:stackoverflow 2024-02-16 08:09:23 0浏览 收藏

对于一个Golang开发者来说,牢固扎实的基础是十分重要的,golang学习网就来带大家一点点的掌握基础知识点。今天本篇文章带大家了解《正确监测长时间运行任务的进展》,主要介绍了,希望对大家的知识积累有所帮助,快点收藏起来吧,否则需要时就找不到了!

问题内容

我想跟踪一些长时间运行的进程的执行情况并显示用户完成百分比和错误(如果有)。如果它是一个长时间运行的流程,那么很简单 - 您可以创建进度(百分比)和错误的通道。当我们有 x 个长时间运行的进程时,实现此类逻辑的正确方法是什么?

下面是一段有效的代码片段,但我不太喜欢它的实现方式。 我创建了一个结构 progresstracker ,它保留 url (作为字段)、errorprogress 作为渠道。我将这样的 progresstracker 保留在一个切片中,一旦提交所有任务,我就会通过 progresstracker 的切片进行迭代,并监听 progresstracker 中每个 tracker 的通道。一旦提交的请求数 == 接收到的响应数 - 退出循环。

这是 go 惯用的解决方案吗?将 progresstracker 作为通道传递给函数会更容易,但我不知道在这种情况下如何正确发送“进度”、“错误”和“完成”事件。

代码如下,go演示中也有同样的代码:https://go.dev/play/p/f3hxjszr9wv

package main

import (
    "errors"
    "fmt"
    "strings"
    "sync"
    "time"
)

type progresstracker struct {
    progress chan int
    error    chan error
    complete chan bool
    url      string
}

/**
this method sleeps for 1 second and sends progress (in %) in each iteration to progress channel
for .net sites on 3rd iteration fail with error
when everything is completed, send a message to complete channel
*/
func work(url string, tracker *progresstracker) {
    tracker.url = url
    fmt.printf("processing url %s\n", url)
    for i := 1; i <= 5; i++ {
        time.sleep(time.second)
        if i == 3 && strings.hassuffix(url, ".net") {
            tracker.error <- errors.new("emulating error for .net sites")
            tracker.complete <- true
        }
        progress := 20 * i
        tracker.progress <- progress
    }
    tracker.complete <- true
}

func main() {
    var trackers []*progresstracker
    var urls = []string{"google.com", "youtube.com", "someurl.net"}
    var wg sync.waitgroup
    
    wg.add(len(urls))
    for _, url := range urls {
        tracker := &progresstracker{
            progress: make(chan int),
            error:    make(chan error),
            complete: make(chan bool),
        }
        trackers = append(trackers, tracker)
        go func(workurl string, progresstracker *progresstracker) {
            work(workurl, progresstracker)
        }(url, tracker)
    }

    go func() {
        wg.wait()
    }()

    var processed = 0

    //iterate through all trackers and select each channel.
    //exit from this loop when number of processed requests equals the number of trackers
    for {
        for _, t := range trackers {
            select {
            case pr := <-t.progress:
                fmt.printf("url = %s, progress = %d\n", t.url, pr)
            case err := <-t.error:
                fmt.printf("url = %s, error = %s\n", t.url, err.error())
            case <-t.complete:
                fmt.printf("url = %s is completed\n", t.url)
                processed = processed + 1
                if processed == len(trackers) {
                    fmt.printf("everything is completed, exit")
                    return
                }
            }
        }
    }
}

更新: 如果我为其中一个任务添加延迟,那么我选择所有通道的 for 循环也将等待每次迭代中最慢的工作线程。 go 演示:https://go.dev/play/p/9fvde7zgirp 更新的工作功能:

func work(url string, tracker *ProgressTracker) {
    tracker.Url = url
    fmt.Printf("processing url %s\n", url)
    for i := 1; i <= 5; i++ {
        if url == "google.com" {
            time.Sleep(time.Second * 3)
        }
        time.Sleep(time.Second)
        if i == 3 && strings.HasSuffix(url, ".net") {
            tracker.Error <- errors.New("emulating error for .net sites")
            tracker.Complete <- true
            return
        }
        progress := 20 * i
        tracker.Progress <- progress
    }
    tracker.Complete <- true
}

正确答案


谢谢您的回答!我想出了这种方法,它适合我的需求:

package main

import (
    "errors"
    "fmt"
    "strings"
    "sync"
    "time"
)

type ProgressTracker struct {
    Progress  int
    Error     error
    Completed bool
    Url       string
}

/**
This method sleeps for 1 second and sends progress (in %) in each iteration to Progress channel
For .net sites on 3rd iteration fail with error
When everything is completed, send a message to Complete channel
*/
func work(url string, tracker chan ProgressTracker) {
    var internalTracker = ProgressTracker{
        Url: url,
    }
    tracker <- internalTracker
    fmt.Printf("processing url %s\n", url)
    for i := 1; i <= 5; i++ {
        if url == "google.com" {
            time.Sleep(time.Second * 3)
        }
        time.Sleep(time.Second)
        if i == 3 && strings.HasSuffix(url, ".net") {
            internalTracker.Error = errors.New("error for .net sites")
            internalTracker.Completed = true
            tracker <- internalTracker
            return
        }
        progress := 20 * i
        internalTracker.Progress = progress
        internalTracker.Completed = false
        tracker <- internalTracker
    }
    internalTracker.Completed = true
    tracker <- internalTracker
}

func main() {
    var urls = []string{"google.com", "youtube.com", "someurl.net"}
    var tracker = make(chan ProgressTracker, len(urls))
    var wg sync.WaitGroup
    wg.Add(len(urls))

    for _, url := range urls {
        go func(workUrl string) {
            defer wg.Done()
            work(workUrl, tracker)
        }(url)
    }

    go func() {
        wg.Wait()
        close(tracker)
        fmt.Printf("After wg wait")
    }()

    var completed = 0

    for completed < len(urls) {
        select {
        case t := <-tracker:
            if t.Completed {
                fmt.Printf("Processing for %s is completed!\n", t.Url)
                completed = completed + 1
            } else {
                fmt.Printf("Processing for %s is in progress: %d\n", t.Url, t.Progress)
            }
            if t.Error != nil {
                fmt.Printf("Url %s has errors %s\n", t.Url, t.Error)
            }
        }

    }
}

在这里,我将 progresstracker 作为通道传递(progresstracker 中的字段被声明为简单字段,而不是通道),并且在工作函数中的每个事件上返回正在发生的情况的完整状态(如果进度增加 - 设置新值并返回一个结构 到通道,如果发生错误 - 设置错误并返回结构等)。

您陷入僵局,因为您继续对已完成但没有默认值的跟踪器进行 select 。您的内部 for 循环每次都会迭代所有跟踪器,其中包括已完成并且永远不会发送其他消息的跟踪器。解决这个问题的最简单的方法是一个空的 default ,这也将使这些在现实生活中表现得更好,因为它们并不都以相同的速度进行,但它确实会将其变成一个紧密的循环,从而消耗更多的 cpu。 p>

您的 waitgroup 根本不执行任何操作;你在 goroutine 中调用 wait 但当它返回时什么也不做,并且你从未在它跟踪的 goroutine 中调用 done,所以它永远不会返回。相反,您单独跟踪收到的 complete 消息的数量,并使用它而不是 waitgroup;目前还不清楚为什么要这样实现。

修复这两个问题可以解决所述问题:https://go.dev/play/p/do0g9jrX0mY

但是,这可能不是正确的方法。不可能用一个人为的例子来说明什么是正确的方法;如果示例是它需要做的全部,那么您不需要任何逻辑,您可以将打印语句放在工作人员中,只需使用等待组而不使用通道即可完成。假设您实际上正在对结果执行某些操作,您可能需要一个由所有工作人员共享的单个 completed 通道和一个 error 通道,并且可能需要一个完全不同的机制来跟踪进度,例如您可以从中读取的原子 int/float当您想了解当前进度时。那么您就不需要嵌套循环的东西,您只需一个带有 select 的循环来从共享通道读取消息。这完全取决于此代码的使用上下文。

本篇关于《正确监测长时间运行任务的进展》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于Golang的相关知识,请关注golang学习网公众号!

版本声明
本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
解决Go模块安装错误:undefined:any解决Go模块安装错误:undefined:any
上一篇
解决Go模块安装错误:undefined:any
两个具有相同表现但产生不同结果的 Go 函数的原因解析
下一篇
两个具有相同表现但产生不同结果的 Go 函数的原因解析
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    542次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    508次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    497次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • 毕业宝AIGC检测:AI生成内容检测工具,助力学术诚信
    毕业宝AIGC检测
    毕业宝AIGC检测是“毕业宝”平台的AI生成内容检测工具,专为学术场景设计,帮助用户初步判断文本的原创性和AI参与度。通过与知网、维普数据库联动,提供全面检测结果,适用于学生、研究者、教育工作者及内容创作者。
    14次使用
  • AI Make Song:零门槛AI音乐创作平台,助你轻松制作个性化音乐
    AI Make Song
    AI Make Song是一款革命性的AI音乐生成平台,提供文本和歌词转音乐的双模式输入,支持多语言及商业友好版权体系。无论你是音乐爱好者、内容创作者还是广告从业者,都能在这里实现“用文字创造音乐”的梦想。平台已生成超百万首原创音乐,覆盖全球20个国家,用户满意度高达95%。
    26次使用
  • SongGenerator.io:零门槛AI音乐生成器,快速创作高质量音乐
    SongGenerator
    探索SongGenerator.io,零门槛、全免费的AI音乐生成器。无需注册,通过简单文本输入即可生成多风格音乐,适用于内容创作者、音乐爱好者和教育工作者。日均生成量超10万次,全球50国家用户信赖。
    24次使用
  •  BeArt AI换脸:免费在线工具,轻松实现照片、视频、GIF换脸
    BeArt AI换脸
    探索BeArt AI换脸工具,免费在线使用,无需下载软件,即可对照片、视频和GIF进行高质量换脸。体验快速、流畅、无水印的换脸效果,适用于娱乐创作、影视制作、广告营销等多种场景。
    26次使用
  • SEO标题协启动:AI驱动的智能对话与内容生成平台 - 提升创作效率
    协启动
    SEO摘要协启动(XieQiDong Chatbot)是由深圳协启动传媒有限公司运营的AI智能服务平台,提供多模型支持的对话服务、文档处理和图像生成工具,旨在提升用户内容创作与信息处理效率。平台支持订阅制付费,适合个人及企业用户,满足日常聊天、文案生成、学习辅助等需求。
    28次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码