一文搞懂Go语言通道【channel】

简介: 一文搞懂Go语言通道【channel】

文章目录



前言


在Go语言中管道类似于一个数据流,每次放入或者取出一部分数据,数据取出后原通道内的数据就删除掉,在linux操作系统中管道会将函数的返回结果作为下一个函数的参数



一、Go语言通道基础概念


1.channel产生背景:


  线程之间进行通信的时候,会因为资源的争夺而产生竟态问题,为了保证数据交换的正确性,
  必须使用互斥量给内存进行加锁,go语言并发的模型是CSP,提倡通过通信共享内存,而不是
  通过共享内存而实现通信,通道恰巧满足这种需求。


2.channel工作方式


  channel类似与一个队列,满足先进先出的规则,严格保证收发数据的顺序,每一个通道只能通
  过固定类型的数据如果通道进行大型结构体、字符串的传输,可以将对应的指针传进去,尽量的节省空间


二、通道使用语法


1.通道的声明与初始化


  //定义一个通道对象使用,其中int可以换为自己需要的类型
  var a chan int  
  //初始化只有一个位置的通道(第一个参数代表通道类型,第二个参数代表通道有几个位置)
  //位置存满后新的数据将存不进来(阻塞)
  a = make(chan int,1)


2.将数据放入通道内


  取出数据使用操作符 <-
  操作符右是输入变量,操作符左是通道
  代表数据流入通道内


代码如下:


  // 声明一个通道
  var a chan int
    a <- 5


3.从通道内取出数据


  取出数据也使用操作符 <-
  操作符右是通道,操作符左是接受变量


代码如下:


  //声明一个通道类型
  var a chan int
  fmt.Println("未初始化的通道", a)
  a = make(chan int)
  // wg.Add(1)
  go func(a chan int) {
    // defer wg.Done()
    for {
      x := <-a
      fmt.Println("接收到了数据:", x)
    }
  }(a)


4.关闭通道close


  如果通道重复关闭或者关闭一个没有初始化的通道就会抛出错误
  close(a)//a为待关闭的通道


在并发函数中一次关闭通道代码如下:


// 互斥锁对象
var once sync.Once
//并发函数
//这个函数的目的是将a通道内数据乘以10发送到通道b内
func f2(a <-chan int, b chan<- int) {
  defer wg.Done()
  for {
    x, ok := <-a
    if !ok {
      break
    }
    fmt.Println(x)
    b <- x * 10
  }
  // 确保b通道只关闭一次
  once.Do(func() {
    close(b)
  })
}


三、单项通道及通道的状态分析


1.单项输出通道


var b <-chan int
• 1


2.单项输入通道


var b chan<- int



示例函数:


//单项通道一般做函数参数,作为一种规范防止通道混用
//此函数完成的功能是将a内的数据乘以10放入通道b内
func f2(a <-chan int, b chan<- int) {
  for {
    x, ok := <-a
    if !ok {
      break
    }
    fmt.Println(x)
    b <- x * 10
  }
}


3.通道的状态


channel nil未初始化 空通道 满通道 非空
接收 阻塞 阻塞 接收值 接收值
发送 阻塞 发送值 阻塞 发送值
关闭 panic 关闭成功 关闭成功 关闭成功
关闭后返回的数据 panic 返回0值 数据读完后返回零值 数据读完返回零值



四、通道死锁原因分析


注意以下情况:

在使用通道的时候,从以上表格可知有时会进入阻塞状态,如果结合我们之前介绍过的waitGroup

,如果在主函数等待使用通道的函数执行结束,而使用通道的函数并且通道陷入阻塞状态,如果有其他函数对其进行唤醒则不会死锁,如果没有其他函数可以对其进行唤醒则会抛出死锁异常。


总结


通道将数据隔离在每一份通道内,在并发的情况下可以很好的使用数据,当然要熟悉通道阻塞的几种情况,避免死锁异常。

目录
相关文章
|
8天前
|
JSON 中间件 Go
go语言后端开发学习(四) —— 在go项目中使用Zap日志库
本文详细介绍了如何在Go项目中集成并配置Zap日志库。首先通过`go get -u go.uber.org/zap`命令安装Zap,接着展示了`Logger`与`Sugared Logger`两种日志记录器的基本用法。随后深入探讨了Zap的高级配置,包括如何将日志输出至文件、调整时间格式、记录调用者信息以及日志分割等。最后,文章演示了如何在gin框架中集成Zap,通过自定义中间件实现了日志记录和异常恢复功能。通过这些步骤,读者可以掌握Zap在实际项目中的应用与定制方法
go语言后端开发学习(四) —— 在go项目中使用Zap日志库
|
1天前
|
安全 Java Go
探索Go语言在高并发环境中的优势
在当今的技术环境中,高并发处理能力成为评估编程语言性能的关键因素之一。Go语言(Golang),作为Google开发的一种编程语言,以其独特的并发处理模型和高效的性能赢得了广泛关注。本文将深入探讨Go语言在高并发环境中的优势,尤其是其goroutine和channel机制如何简化并发编程,提升系统的响应速度和稳定性。通过具体的案例分析和性能对比,本文揭示了Go语言在实际应用中的高效性,并为开发者在选择合适技术栈时提供参考。
|
5天前
|
运维 Kubernetes Go
"解锁K8s二开新姿势!client-go:你不可不知的Go语言神器,让Kubernetes集群管理如虎添翼,秒变运维大神!"
【8月更文挑战第14天】随着云原生技术的发展,Kubernetes (K8s) 成为容器编排的首选。client-go作为K8s的官方Go语言客户端库,通过封装RESTful API,使开发者能便捷地管理集群资源,如Pods和服务。本文介绍client-go基本概念、使用方法及自定义操作。涵盖ClientSet、DynamicClient等客户端实现,以及lister、informer等组件,通过示例展示如何列出集群中的所有Pods。client-go的强大功能助力高效开发和运维。
24 1
|
6天前
|
SQL 关系型数据库 MySQL
Go语言中使用 sqlx 来操作 MySQL
Go语言因其高效的性能和简洁的语法而受到开发者们的欢迎。在开发过程中,数据库操作不可或缺。虽然Go的标准库提供了`database/sql`包支持数据库操作,但使用起来稍显复杂。为此,`sqlx`应运而生,作为`database/sql`的扩展库,它简化了许多常见的数据库任务。本文介绍如何使用`sqlx`包操作MySQL数据库,包括安装所需的包、连接数据库、创建表、插入/查询/更新/删除数据等操作,并展示了如何利用命名参数来进一步简化代码。通过`sqlx`,开发者可以更加高效且简洁地完成数据库交互任务。
13 1
|
11天前
|
XML JSON Go
微服务架构下的配置管理:Go 语言与 yaml 的完美结合
微服务架构下的配置管理:Go 语言与 yaml 的完美结合
|
11天前
|
程序员 Go
Go 语言:面向对象还是非面向对象?揭开编程语言的本质
Go 语言:面向对象还是非面向对象?揭开编程语言的本质
|
5天前
|
算法 NoSQL 中间件
go语言后端开发学习(六) ——基于雪花算法生成用户ID
本文介绍了分布式ID生成中的Snowflake(雪花)算法。为解决用户ID安全性与唯一性问题,Snowflake算法生成的ID具备全局唯一性、递增性、高可用性和高性能性等特点。64位ID由符号位(固定为0)、41位时间戳、10位标识位(含数据中心与机器ID)及12位序列号组成。面对ID重复风险,可通过预分配、动态或统一分配标识位解决。Go语言实现示例展示了如何使用第三方包`sonyflake`生成ID,确保不同节点产生的ID始终唯一。
go语言后端开发学习(六) ——基于雪花算法生成用户ID
|
11天前
|
存储 安全 Go
Go 并发编程精粹:掌握通道(channels)的艺术
Go 并发编程精粹:掌握通道(channels)的艺术
|
6天前
|
JSON 缓存 监控
go语言后端开发学习(五)——如何在项目中使用Viper来配置环境
Viper 是一个强大的 Go 语言配置管理库,适用于各类应用,包括 Twelve-Factor Apps。相比仅支持 `.ini` 格式的 `go-ini`,Viper 支持更多配置格式如 JSON、TOML、YAML
go语言后端开发学习(五)——如何在项目中使用Viper来配置环境
|
11天前
|
存储 编译器 Go
Go语言中的逃逸分析
Go语言中的逃逸分析