一文搞懂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月前
|
运维 监控 算法
监控局域网其他电脑:Go 语言迪杰斯特拉算法的高效应用
在信息化时代,监控局域网成为网络管理与安全防护的关键需求。本文探讨了迪杰斯特拉(Dijkstra)算法在监控局域网中的应用,通过计算最短路径优化数据传输和故障检测。文中提供了使用Go语言实现的代码例程,展示了如何高效地进行网络监控,确保局域网的稳定运行和数据安全。迪杰斯特拉算法能减少传输延迟和带宽消耗,及时发现并处理网络故障,适用于复杂网络环境下的管理和维护。
|
2月前
|
数据采集 Go API
Go语言实战案例:多协程并发下载网页内容
本文是《Go语言100个实战案例 · 网络与并发篇》第6篇,讲解如何使用 Goroutine 和 Channel 实现多协程并发抓取网页内容,提升网络请求效率。通过实战掌握高并发编程技巧,构建爬虫、内容聚合器等工具,涵盖 WaitGroup、超时控制、错误处理等核心知识点。
|
2月前
|
Go 开发者
Go语言实战案例:使用select监听多个channel
本文为《Go语言100个实战案例 · 网络与并发篇》第5篇,详解Go并发核心工具`select`的使用。通过实际案例讲解如何监听多个Channel、实现多任务处理、超时控制和非阻塞通信,帮助开发者掌握Go并发编程中的多路异步事件处理技巧。
|
2月前
|
数据采集 编解码 监控
Go语言实战案例:使用channel实现生产者消费者模型
本文是「Go语言100个实战案例 · 网络与并发篇」第4篇,通过实战案例详解使用 Channel 实现生产者-消费者模型,涵盖并发控制、任务调度及Go语言并发哲学,助你掌握优雅的并发编程技巧。
|
2月前
|
数据采集 JSON Go
Go语言实战案例:实现HTTP客户端请求并解析响应
本文是 Go 网络与并发实战系列的第 2 篇,详细介绍如何使用 Go 构建 HTTP 客户端,涵盖请求发送、响应解析、错误处理、Header 与 Body 提取等流程,并通过实战代码演示如何并发请求多个 URL,适合希望掌握 Go 网络编程基础的开发者。
|
3月前
|
JSON 前端开发 Go
Go语言实战:创建一个简单的 HTTP 服务器
本篇是《Go语言101实战》系列之一,讲解如何使用Go构建基础HTTP服务器。涵盖Go语言并发优势、HTTP服务搭建、路由处理、日志记录及测试方法,助你掌握高性能Web服务开发核心技能。
|
3月前
|
Go
如何在Go语言的HTTP请求中设置使用代理服务器
当使用特定的代理时,在某些情况下可能需要认证信息,认证信息可以在代理URL中提供,格式通常是:
248 0
|
4月前
|
JSON 编解码 API
Go语言网络编程:使用 net/http 构建 RESTful API
本章介绍如何使用 Go 语言的 `net/http` 标准库构建 RESTful API。内容涵盖 RESTful API 的基本概念及规范,包括 GET、POST、PUT 和 DELETE 方法的实现。通过定义用户数据结构和模拟数据库,逐步实现获取用户列表、创建用户、更新用户、删除用户的 HTTP 路由处理函数。同时提供辅助函数用于路径参数解析,并展示如何设置路由器启动服务。最后通过 curl 或 Postman 测试接口功能。章节总结了路由分发、JSON 编解码、方法区分、并发安全管理和路径参数解析等关键点,为更复杂需求推荐第三方框架如 Gin、Echo 和 Chi。
|
4月前
|
Go 开发者
Go 并发编程基础:无缓冲与有缓冲通道
本章深入探讨Go语言中通道(Channel)的两种类型:无缓冲通道与有缓冲通道。无缓冲通道要求发送和接收必须同步配对,适用于精确同步和信号通知;有缓冲通道通过内部队列实现异步通信,适合高吞吐量和生产者-消费者模型。文章通过示例对比两者的行为差异,并分析死锁风险及使用原则,帮助开发者根据场景选择合适的通道类型以实现高效并发编程。