我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Golang的select怎么使用

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Golang的select怎么使用

本文小编为大家详细介绍“Golang的select怎么使用”,内容详细,步骤清晰,细节处理妥当,希望这篇“Golang的select怎么使用”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。

背景

golang 中主推 channel 通信。单个 channel 的通信可以通过一个goroutinechannel 发数据,另外一个从channel取数据进行。这是阻塞的,因为要想顺利执行完这个步骤,需要 channel 准备好才行,准备好的条件如下:

发送

  • 缓存有空间(如果是有缓存的 channel)

  • 有等待接收的 goroutine

接收

  • 缓存有数据(如果是有缓存的 channel)

  • 有等待发送的 goroutine

channel实际使用中还有如下两个需求,这个时候就需要select了。

  • 同时监听多个channel

  • 在没有channel准备好的时候,也可以往下执行。

select 流程

select。作用是阻塞当前goroutine。不要用for{}来阻塞goroutine,因为会占用cpu。而select{}不会,因为当前goroutine不会再被调度。

 if len(cases) == 0 {         block() }

配置好poll的顺序。由于是同时监听多个channel的发送或者接收,所以需要按照一定的顺序查看哪个channel准备好了。如果每次采用select中的顺序查看channel是否准备好了,那么只要在前面的channel准备好的足够快,那么会造成后面的channel即使准备好了,也永远不会被执行。打乱顺序的逻辑如下,采用了洗牌算法\color{red}{洗牌算法}洗牌算法,注意此过程中会过滤掉channel为nil的case。\color{red}{注意此过程中会过滤掉 channel 为 nil 的 case。}注意此过程中会过滤掉channel为nil的case。

 // generate permuted order norder := 0 for i := range scases {         cas := &scases[i]         // Omit cases without channels from the poll and lock orders.         if cas.c == nil {                 cas.elem = nil // allow GC                 continue         }         j := fastrandn(uint32(norder + 1))         pollorder[norder] = pollorder[j]         pollorder[j] = uint16(i)         norder++ }

配置好lock的顺序。由于可能会修改channel中的数据,所以在打算往channel中发送数据或者从channel接收数据的时候,需要锁住 channel。而一个channel可能被多个select监听,如果两个select对两个channel A和B,分别按照顺序A, B和B,A上锁,是可能会造成死锁的,导致两个select都执行不下去。

Golang的select怎么使用

所以select中锁住channel的顺序至关重要,解决方案是按照channel的地址的顺序锁住channel。因为在两个selectchannel有交集的时候,都是按照交集中channel的地址顺序锁channel

实际排序代码如下,采用堆排序算法\color{red}{堆排序算法}堆排序算法按照channel的地址从小到大对channel进行排序。

 // sort the cases by Hchan address to get the locking order. // simple heap sort, to guarantee n log n time and constant stack footprint. for i := range lockorder {         j := i         // Start with the pollorder to permute cases on the same channel.         c := scases[pollorder[i]].c         for j > 0 && scases[lockorder[(j-1)/2]].c.sortkey() < c.sortkey() {                 k := (j - 1) / 2                 lockorder[j] = lockorder[k]                 j = k         }         lockorder[j] = pollorder[i] } for i := len(lockorder) - 1; i >= 0; i-- {         o := lockorder[i]         c := scases[o].c         lockorder[i] = lockorder[0]         j := 0         for {                 k := j*2 + 1                 if k >= i {                         break                 }                 if k+1 < i && scases[lockorder[k]].c.sortkey() < scases[lockorder[k+1]].c.sortkey() {                         k++                 }                 if c.sortkey() < scases[lockorder[k]].c.sortkey() {                         lockorder[j] = lockorder[k]                         j = k                         continue                 }                 break         }         lockorder[j] = o }

锁住select中的所有channel。要查看channel中的数据了。

 // lock all the channels involved in the select sellock(scases, lockorder)

第一轮查看是否已有准备好的channel。如果有直接发送数据到channel或者从channel接收数据。注意selectchannel切片中,前面部分是从channel接收数据的case,后半部分是往channel发送数据的case。

Golang的select怎么使用

按照pollorder顺序查看是否有channel准备好了。

 for _, casei := range pollorder {         casi = int(casei)         cas = &scases[casi]         c = cas.c         if casi >= nsends {                 sg = c.sendq.dequeue()                 if sg != nil {                         goto recv                 }                 if c.qcount > 0 {                         goto bufrecv                 }                 if c.closed != 0 {                         goto rclose                 }         } else {                 if raceenabled {                         racereadpc(c.raceaddr(), casePC(casi), chansendpc)                 }                 if c.closed != 0 {                         goto sclose                 }                 sg = c.recvq.dequeue()                 if sg != nil {                         goto send                 }                 if c.qcount < c.dataqsiz {                         goto bufsend                 }         } }

直接执行default分支

 if !block {         selunlock(scases, lockorder)         casi = -1         goto retc }

第二轮遍历channel。创建sudog把当前goroutine放到每个channel的等待列表中去,等待channel准备好时被唤醒。

 // pass 2 - enqueue on all chans gp = getg() if gp.waiting != nil {         throw("gp.waiting != nil") } nextp = &gp.waiting for _, casei := range lockorder {         casi = int(casei)         cas = &scases[casi]         c = cas.c         sg := acquireSudog()         sg.g = gp         sg.isSelect = true         // No stack splits between assigning elem and enqueuing         // sg on gp.waiting where copystack can find it.         sg.elem = cas.elem         sg.releasetime = 0         if t0 != 0 {                 sg.releasetime = -1         }         sg.c = c         // Construct waiting list in lock order.         *nextp = sg         nextp = &sg.waitlink         if casi < nsends {                 c.sendq.enqueue(sg)         } else {                 c.recvq.enqueue(sg)         } }

等待被唤醒。其中gopark的时候会释放对所有channel占用的锁。

 // wait for someone to wake us up gp.param = nil // Signal to anyone trying to shrink our stack that we're about // to park on a channel. The window between when this G's status // changes and when we set gp.activeStackChans is not safe for // stack shrinking. atomic.Store8(&gp.parkingOnChan, 1) gopark(selparkcommit, nil, waitReasonSelect, traceEvGoBlockSelect, 1) gp.activeStackChans = false

被唤醒

  • 锁住所有channel

  • 清理当前goroutine的等待sudog

  • 找到是被哪个channel唤醒的,并清理每个channel上当前的goroutine对应的sudog

 sellock(scases, lockorder) gp.selectDone = 0 sg = (*sudog)(gp.param) gp.param = nil // pass 3 - dequeue from unsuccessful chans // otherwise they stack up on quiet channels // record the successful case, if any. // We singly-linked up the SudoGs in lock order. casi = -1 cas = nil caseSuccess = false sglist = gp.waiting // Clear all elem before unlinking from gp.waiting. for sg1 := gp.waiting; sg1 != nil; sg1 = sg1.waitlink {         sg1.isSelect = false         sg1.elem = nil         sg1.c = nil } gp.waiting = nil for _, casei := range lockorder {         k = &scases[casei]         if sg == sglist {                 // sg has already been dequeued by the G that woke us up.                 casi = int(casei)                 cas = k                 caseSuccess = sglist.success                 if sglist.releasetime > 0 {                         caseReleaseTime = sglist.releasetime                 }         } else {                 c = k.c                 if int(casei) < nsends {                         c.sendq.dequeueSudoG(sglist)                 } else {                         c.recvq.dequeueSudoG(sglist)                 }         }         sgnext = sglist.waitlink         sglist.waitlink = nil         releaseSudog(sglist)         sglist = sgnext }

读到这里,这篇“Golang的select怎么使用”文章已经介绍完毕,想要掌握这篇文章的知识点还需要大家自己动手实践使用过才能领会,如果想了解更多相关内容的文章,欢迎关注编程网行业资讯频道。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Golang的select怎么使用

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Golang的select怎么使用

本文小编为大家详细介绍“Golang的select怎么使用”,内容详细,步骤清晰,细节处理妥当,希望这篇“Golang的select怎么使用”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。背景golang 中主推
2023-07-04

golang select语句怎么使用

在Go语言中,select语句用于在多个通信操作中选择一个进行执行。select语句的基本语法如下:goselect {case <- channel1: // 执行channel1的操作case data := <- channel
2023-10-21

golang中select语句怎么使用

这篇文章主要介绍“golang中select语句怎么使用”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“golang中select语句怎么使用”文章能帮助大家解决问题。前言在golang语言中,sel
2023-07-02

golang中select怎么用

这篇文章将为大家详细讲解有关golang中select怎么用,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。什么是golanggolang 是Google开发的一种静态强类型、编译型、并发型,并具有垃圾回收
2023-06-15

SQL select distinct怎么使用

使用SELECT DISTINCT语句可以返回表中唯一不重复的值。语法如下:```sqlSELECT DISTINCT 列名FROM 表名;```示例:假设有一个名为"customers"的表,其中包含了一个"country"列,我们想要获
2023-08-14

java的select语句怎么使用

在Java中,可以使用JDBC(Java Database Connectivity)来执行SQL查询语句。以下是一个使用JDBC执行SELECT语句的示例:1. 导入必要的包:```javaimport java.sql.*;```2.
2023-09-20

MySQL的select语句怎么使用

这篇“MySQL的select语句怎么使用”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“MySQL的select语句怎么使用
2023-06-27

sql中的select distinct怎么使用

在SQL中,SELECT DISTINCT用于返回唯一不重复的值。通过在SELECT语句中添加DISTINCT关键字,可以筛选出唯一的值。例如,查询一个表中的不重复的城市名称:SELECT DISTINCT cityFROM cust
sql中的select distinct怎么使用
2024-04-16

Golang怎么实现带优先级的select

这篇“Golang怎么实现带优先级的select”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Golang怎么实现带优先级的
2023-07-05

怎么使用MySQL中的select、distinct、limit

这篇文章主要讲解了“怎么使用MySQL中的select、distinct、limit”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么使用MySQL中的select、distinct、lim
2023-06-25

MyBatis @Select注解怎么使用

这篇文章主要介绍“MyBatis @Select注解怎么使用”,在日常操作中,相信很多人在MyBatis @Select注解怎么使用问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”MyBatis @Select
2023-07-02

select多选multiple怎么使用

在使用SELECT语句时,如果希望实现多选(multiple)的选择功能,可以使用以下方法:1. 使用IN关键字:可以在WHERE子句中使用IN关键字来指定多个选项。例如,如果想选择ID为1、2和3的行,可以这样写:```SELECT *
2023-08-11

access select语句怎么使用

SELECT语句用于从数据库中选择数据。它的基本语法如下:```sqlSELECT 列名1, 列名2, ... FROM 表名 WHERE 条件```其中,列名是要选择的数据列,可以是单个列或多个列,用逗号分隔。表名是要从中选择数据的表名称
2023-09-22

sqlserver中select into怎么使用

SELECT INTO语句可以将查询的结果集插入到新的表中。语法如下:SELECT column1, column2, ...INTO new_tableFROM source_tableWHERE condition;其中,col
sqlserver中select into怎么使用
2024-04-09

MySQL中select limit怎么使用

在MySQL中,SELECT语句的LIMIT子句用于限制查询结果的数量。使用LIMIT子句的基本语法如下:SELECT 列名称FROM 表名称LIMIT 数量;其中,数量指定了要返回的行数。例如,要从名为"customers"的表中选择前5
2023-10-19

Linq Select子句怎么使用

本篇内容主要讲解“Linq Select子句怎么使用”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Linq Select子句怎么使用”吧!选择数据(SELECT)Linq Select子句指定所
2023-06-17

redis select命令怎么使用

SELECT命令用于切换到指定的数据库,数据库索引号从0开始,最大可以达到16384。该命令会清空当前数据库的数据,然后切换到指定的数据库。语法如下:SELECT index例如,要切换到第5个数据库,可以使用以下命令:SELECT
redis select命令怎么使用
2024-04-28

go语言中怎么使用select

这篇文章主要介绍“go语言中怎么使用select”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“go语言中怎么使用select”文章能帮助大家解决问题。在golang语言中,select语句 就是用来
2023-06-30

asp中select语句怎么使用

在ASP中使用SELECT语句查询数据库可以通过以下步骤来实现:1. 创建数据库连接对象:```Set conn = Server.CreateObject("ADODB.Connection")conn.Open "Provider=Mi
2023-09-25

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录