go 语言中 channel 的简单介绍

简介: go 语言中 channel 的简单介绍

介绍

channel 是 Go 中被用来实现并行之间通信的类型,其功能是允许线程间通过发送和接收来传输指定类型的数据。初始值为 nil。

创建channel

创建 channel 如下:

var c1 chan [value type]
c1 = make([channel type] [value type], [capacity])
  • [value type] 定义的是 channel 中所传输数据的类型
  • [channel type] 定义的是 channel 的类型,其类型有以下三种:
  • "chan" 可读可写 —— "chan int" 则表示可读写 int 数据类型的 channel
  • "chan<-" 仅可写 —— "chan<- float64" 则表示仅可写64位 float 数据类型的 channel
  • "<-chan" 仅可读 —— "<-chan int" 则表示仅可读 int 数据类型的 channel
  • [capacity] 是一个可选参数,其定义的是 channel 中的缓存区 (buffer)。如果不填则默认该 channel 没有缓冲区 (unbuffered)。对于没有缓冲区的 channel,消息的发送和收取必须能同时完成,否则会造成阻塞并提示死锁错误。对于 channel 的阻塞和非阻塞将在后面详细介绍。

比如我们想创建了一个读写 int 类型,buffer 长度 100 的 channel c1,则如下:

var c1 chan int
c1 = make(chan int, 100)

通过 channel 发送和接收消息

示例代码:

package main
 
import "fmt"
 
func main(){
  //定义变量
  var c1 chan int
  var i1 int
  //初始化 channel
  c1 = make(chan int, 100)
  //向 channel c1 发送(写入)一个 int 20
  c1 <- 20
  //从 channel c1 接收(读取)一个 int 并赋值给 i1
  i1 = <- c1
  //将 i1 打印输出
  fmt.Println("received: ", i1, " from c1")
}

运行结果:

received:  20  from c1

使用 Channel 发生死锁

如下代码会出现死锁:

package main
 
import "fmt"
import "time"
 
func main(){
  var c1 chan string
  c1 = make(chan string)
  func() {
    time.Sleep(time.Second * 2)
    c1 <- "result 1"
  }()
  fmt.Println("received: '", <- c1,"' from c1")
}

因为对 channel 的发送和接收动作永远不会同时发生,所以从以下这行代码就有问题了,不能直接从一个空通道中获取值,从而阻塞造成死锁:

c1 <- "result 1"

解决该问题的方式有两种。

避免死锁方法一:使用 goroutine 进行并行计算

package main
 
import "fmt"
import "time"
 
func main(){
  var c1 chan string
  c1 = make(chan string)
  go func() {
    time.Sleep(time.Second * 2)
    c1 <- "result 1"
  }()
  fmt.Println("received: '", <- c1,"' from c1")
}

通过 go 语句定义发送操作的方程在另一个线程并行运行,这样发送和接收操作就可以同时发生,从而能够解决死锁问题。

避免死锁方法二:使用 buffer

package main
 
import "fmt"
import "time"
 
func main(){
  var c1 chan string
  c1 = make(chan string,1) //这里我们设置了一个长度为 1 的 buffer
  func() {
    time.Sleep(time.Second * 2)
    c1 <- "result 1"
  }()
  fmt.Println("received: '", <- c1,"' from c1")
}

为 channel 添加一个缓冲区(buffer),这样只要 buffer 没有用尽,阻塞就不会发生,死锁也不会发生。

相关文章
|
6天前
|
存储 JSON 监控
Viper,一个Go语言配置管理神器!
Viper 是一个功能强大的 Go 语言配置管理库,支持从多种来源读取配置,包括文件、环境变量、远程配置中心等。本文详细介绍了 Viper 的核心特性和使用方法,包括从本地 YAML 文件和 Consul 远程配置中心读取配置的示例。Viper 的多来源配置、动态配置和轻松集成特性使其成为管理复杂应用配置的理想选择。
23 2
|
10天前
|
JavaScript Java Go
探索Go语言在微服务架构中的优势
在微服务架构的浪潮中,Go语言以其简洁、高效和并发处理能力脱颖而出。本文将深入探讨Go语言在构建微服务时的性能优势,包括其在内存管理、网络编程、并发模型以及工具链支持方面的特点。通过对比其他流行语言,我们将揭示Go语言如何成为微服务架构中的一股清流。
102 53
|
4天前
|
Go 索引
go语言中的循环语句
【11月更文挑战第4天】
13 2
|
4天前
|
Go C++
go语言中的条件语句
【11月更文挑战第4天】
16 2
|
9天前
|
Ubuntu 编译器 Linux
go语言中SQLite3驱动安装
【11月更文挑战第2天】
31 7
|
9天前
|
关系型数据库 Go 网络安全
go语言中PostgreSQL驱动安装
【11月更文挑战第2天】
38 5
|
9天前
|
安全 Go
用 Zap 轻松搞定 Go 语言中的结构化日志
在现代应用程序开发中,日志记录至关重要。Go 语言中有许多日志库,而 Zap 因其高性能和灵活性脱颖而出。本文详细介绍如何在 Go 项目中使用 Zap 进行结构化日志记录,并展示如何定制日志输出,满足生产环境需求。通过基础示例、SugaredLogger 的便捷使用以及自定义日志配置,帮助你在实际开发中高效管理日志。
25 1
|
8天前
|
程序员 Go
go语言中的控制结构
【11月更文挑战第3天】
85 58
|
7天前
|
监控 Go API
Go语言在微服务架构中的应用实践
在微服务架构的浪潮中,Go语言以其简洁、高效和并发处理能力脱颖而出,成为构建微服务的理想选择。本文将探讨Go语言在微服务架构中的应用实践,包括Go语言的特性如何适应微服务架构的需求,以及在实际开发中如何利用Go语言的特性来提高服务的性能和可维护性。我们将通过一个具体的案例分析,展示Go语言在微服务开发中的优势,并讨论在实际应用中可能遇到的挑战和解决方案。
|
8天前
|
存储 编译器 Go
go语言中的变量、常量、数据类型
【11月更文挑战第3天】
25 9