当前位置:首页 > 文章列表 > Golang > Go问答 > goroutine通道中的同步问题

goroutine通道中的同步问题

来源:stackoverflow 2024-04-14 12:36:23 0浏览 收藏

在Golang实战开发的过程中,我们经常会遇到一些这样那样的问题,然后要卡好半天,等问题解决了才发现原来一些细节知识点还是没有掌握好。今天golang学习网就整理分享《goroutine通道中的同步问题》,聊聊,希望可以帮助到正在努力赚钱的你。

问题内容

我正在尝试编写一个代理服务器,在其中将视频文件转换为实时流。一个视频点播文件由多个不同比特率的子清单文件组成。每个子清单由多个 ts 段组成,每个 ts 段为 4 秒。为简单起见,我创建了 2 个子清单的虚拟映射,每个子清单包含 4 个 ts 段。我的任务是在 for 循环中无限创建直播流。所以我在 go 例程中 4 秒后发送每个 ts 段并写入通道,并且我还有一个 go 例程,我在其中读取此 go 例程并写入输出字符串。一旦所有片段结束,我将更新到 loopend 频道并重新开始。有 2 个针对不同比特率的 api,每 4 秒后给我一个新的 ts。

package main

import (
    "fmt"
    "github.com/gin-contrib/gzip"
    "github.com/gin-gonic/gin"
    "sync"
    "time"
)

func main() {
    vodToLiveMap := convertVodToLive()
    engine := gin.Default()
    engine.Use(gzip.Gzip(gzip.DefaultCompression))
    engine.GET("/134200", func(gctx *gin.Context) {
        prepareResponse(gctx, "134200", vodToLiveMap)
    })

    engine.GET("/290400", func(gctx *gin.Context) {
        prepareResponse(gctx, "290400", vodToLiveMap)
    })
    engine.Run() // listen and serve on 0.0.0.0:8080
}

// this represents one ulr for http live stream
type hlsEntry struct {
    timeCode int    // timecode in microseconds
    body     string // string including transport segment URL, timecode
}

type VodToLive struct {
    SequenceNumber int
    Out            chan string
    CurrentStream  string
}

func convertVodToLive() sync.Map {
    var hlsEntriesBitrate1 = []hlsEntry{
        {
            timeCode: 4000000,
            body:     "#EXTINF:4.00000,\n/master_Layer1_001.ts\n",
        },
        {
            timeCode: 4000000,
            body:     "#EXTINF:4.00000,\n/master_Layer1_002.ts\n",
        },
        {
            timeCode: 4000000,
            body:     "#EXTINF:4.00000,\n/master_Layer1_003.ts\n",
        },
        {
            timeCode: 4000000,
            body:     "#EXTINF:4.00000,\n/master_Layer1_004.ts\n",
        },
    }

    var hlsEntriesBitrate2 = []hlsEntry{
        {
            timeCode: 4000000,
            body:     "#EXTINF:4.00000,\n/master_Layer2_001.ts\n",
        },
        {
            timeCode: 4000000,
            body:     "#EXTINF:4.00000,\n/master_Layer2_002.ts\n",
        },
        {
            timeCode: 4000000,
            body:     "#EXTINF:4.00000,\n/master_Layer2_003.ts\n",
        },
        {
            timeCode: 4000000,
            body:     "#EXTINF:4.00000,\n/master_Layer2_004.ts\n",
        },
    }
    fmt.Println(hlsEntriesBitrate2)
    var hlsEntriesMap = make(map[string][]hlsEntry)
    hlsEntriesMap["134200"] = hlsEntriesBitrate1
    hlsEntriesMap["290400"] = hlsEntriesBitrate2
    outputString := ""
    vodToLiveMap := sync.Map{}
    i := 0
    //Iterate for all child manifest files of different bit rates
    for childURLSlug := range hlsEntriesMap {
        vodToLive := &VodToLive{}
        vodToLive.SequenceNumber = 1
        vodToLive.Out = make(chan string)
        vodToLiveMap.Store(childURLSlug, vodToLive)
        loopEnd := make(chan bool)

        //Writing to vodToLive for a childURLSlug
        go func(childURLSlug string) {
            for {
                for _, entry := range hlsEntriesMap[childURLSlug] {
                    select {
                    case <-time.After(time.Duration(entry.timeCode) * time.Microsecond):
                        vodToLive.SequenceNumber++
                        vodToLive.Out <- entry.body
                    }
                }
                loopEnd <- true
                fmt.Println(childURLSlug)
            }
        }(childURLSlug)

        //Reading from vodToLive's Out channel for a childURLSlug to update CurrentStream
        go func(childURLSlug string) {
            for {
                for {
                    select {
                    //If video reaches end of stream then restart stream from Sequence #1
                    case <-loopEnd:
                        vodToLive.SequenceNumber = 1
                        //time.Sleep(10 * time.Second)
                        vodToLive.CurrentStream = ""
                        break

                    default:
                        outputString = <-vodToLive.Out
                        fmt.Println(outputString)
                        vodToLive.CurrentStream = vodToLive.CurrentStream + outputString

                    }
                }
            }
        }(childURLSlug)
        i++
    }
    return vodToLiveMap
}

func prepareResponse(gctx *gin.Context, childURLSlug string, vodToLiveMap sync.Map) {

    vodToLive, _ := vodToLiveMap.Load(childURLSlug)

    gctx.Header("Content-Type", "application/x-mpegURL")
    response := `#EXTM3U
#EXT-X-VERSION:3
#EXT-X-TARGETDURATION:4
#EXT-X-MEDIA-SEQUENCE:` + "1" + "\n" + vodToLive.(*VodToLive).CurrentStream

    size := len(response)
    if size > 0 && response[size-1] == '\n' {
        response = response[:size-1]
    }
    gctx.String(200, response)
}
//curl localhost:8080/134200
//curl localhost:8080/290400

但这里的问题是我没有在curl请求中获取所有ts段,有时甚至ts在循环结束后也没有更新。我在每个调用中显示了所有 4 个 ts 段,但这些应该在每个 endloop 后重置。


解决方案


只需将默认值替换为通道上的选择 (vodtolive.out) 并在 case <-loopend 中添加 4 秒等待即可解决此问题

go func(childURLSlug string) {
    for {
        for {
            select {
            //If video reaches end of stream then restart stream from Sequence #1
            case <-loopEnd:
                vodToLive.SequenceNumber = 1
                // so this will ensure that next stream starts only when first is complete.
                time.Sleep(4 * time.Second) //Usually a ts is 4 seconds length.
                vodToLive.CurrentStream = ""
                break

            case res := <-vodToLive.Out:
                outputString := res
                fmt.Println(outputString)
                vodToLive.CurrentStream = vodToLive.CurrentStream + outputString

            }
        }
    }
}(childURLSlug)

理论要掌握,实操不能落!以上关于《goroutine通道中的同步问题》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注golang学习网公众号吧!

版本声明
本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
如何格式化包含动态数量元素的字符串?如何格式化包含动态数量元素的字符串?
上一篇
如何格式化包含动态数量元素的字符串?
在结构体内部切片
下一篇
在结构体内部切片
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之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检测,湖南茅茅虫科技有限公司倾力打造,运用NLP技术精准识别AI生成文本,提供论文、专著等学术文本的AIGC检测服务。支持多种格式,生成可视化报告,保障您的学术诚信和内容质量。
    31次使用
  • 赛林匹克平台:科技赛事聚合,赋能AI、算力、量子计算创新
    赛林匹克平台(Challympics)
    探索赛林匹克平台Challympics,一个聚焦人工智能、算力算法、量子计算等前沿技术的赛事聚合平台。连接产学研用,助力科技创新与产业升级。
    53次使用
  • SEO  笔格AIPPT:AI智能PPT制作,免费生成,高效演示
    笔格AIPPT
    SEO 笔格AIPPT是135编辑器推出的AI智能PPT制作平台,依托DeepSeek大模型,实现智能大纲生成、一键PPT生成、AI文字优化、图像生成等功能。免费试用,提升PPT制作效率,适用于商务演示、教育培训等多种场景。
    64次使用
  • 稿定PPT:在线AI演示设计,高效PPT制作工具
    稿定PPT
    告别PPT制作难题!稿定PPT提供海量模板、AI智能生成、在线协作,助您轻松制作专业演示文稿。职场办公、教育学习、企业服务全覆盖,降本增效,释放创意!
    59次使用
  • Suno苏诺中文版:AI音乐创作平台,人人都是音乐家
    Suno苏诺中文版
    探索Suno苏诺中文版,一款颠覆传统音乐创作的AI平台。无需专业技能,轻松创作个性化音乐。智能词曲生成、风格迁移、海量音效,释放您的音乐灵感!
    63次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码