0

0

Golangio.Pipe管道读写与数据传递实践

P粉602998670

P粉602998670

发布时间:2025-09-19 20:00:02

|

797人浏览过

|

来源于php中文网

原创

io.Pipe是Go中用于goroutine间同步数据传输的管道,实现io.Reader和io.Writer接口,支持单向通信、阻塞读写及错误传递,常用于内存流处理。

golangio.pipe管道读写与数据传递实践

在 Go 语言中,io.Pipe 是一种用于在 goroutine 之间进行同步数据传输的管道机制。它实现了 io.Readerio.Writer 接口,适合在并发场景下实现一个协程写入、另一个协程读取的数据流控制。与操作系统级别的管道不同,io.Pipe 完全由 Go 运行时管理,常用于内存中的数据传递,比如处理 HTTP 响应、压缩流、日志转发等场景。

io.Pipe 基本原理

io.Pipe 返回一对关联的 *io.PipeReader 和 *io.PipeWriter。写入 PipeWriter 的数据可以从 PipeReader 中读取。它的核心特点是:

  • 同步阻塞:读写操作是同步的,写入方在没有读取方消费时会阻塞。
  • 单向通信:PipeReader 只能读,PipeWriter 只能写。
  • 非线程安全:多个 goroutine 同时写或读同一个端点需自行加锁或通过 channel 协调。
  • 支持关闭:关闭读端或写端可通知对方结束操作。

创建方式非常简单:

r, w := io.Pipe()
// r 是 *io.PipeReader,实现 io.Reader
// w 是 *io.PipeWriter,实现 io.Writer

基本读写示例

下面是一个简单的例子,演示如何在一个 goroutine 中写入数据,在另一个中读取:

立即学习go语言免费学习笔记(深入)”;

package main

import (
  "fmt"
  "io"
  "log"
)

func main() {
  r, w := io.Pipe()

  go func() {
    defer w.Close()
    _, err := w.Write([]byte("hello from writer"))
    if err != nil {
      log.Fatal(err)
    }
  }()

  buf := make([]byte, 100)
  n, err := r.Read(buf)
  if err != nil {
    log.Fatal(err)
  }
  fmt.Printf("read: %s\n", buf[:n])
  r.Close()
}

运行结果输出:
read: hello from writer

注意:必须在写入完成后调用 w.Close(),否则读取方在数据读完后会一直等待更多数据。如果写入失败或提前中断,也应使用 w.CloseWithError(err) 通知读取方错误原因。

Clickable
Clickable

用AI在几秒钟内生成广告

下载

结合 bufio.Scanner 实现行读取

实际开发中,我们常需要逐行处理数据流。可以将 io.Pipebufio.Scanner 结合使用:

r, w := io.Pipe()
scanner := bufio.NewScanner(r)

go func() {
  defer w.Close()
  w.Write([]byte("line 1\n"))
  w.Write([]byte("line 2\n"))
  w.Write([]byte("line 3\n"))
}()

for scanner.Scan() {
  fmt.Println("got:", scanner.Text())
}
if err := scanner.Err(); err != nil {
  log.Fatal(err)
}

这种方式非常适合模拟日志输出、命令行输出捕获等场景。

错误处理与资源释放

使用 io.Pipe 时,良好的错误处理至关重要。推荐使用 CloseWithError 显式传递错误信息,避免读取方无限等待:

go func() {
  _, err := w.Write(someData)
  if err != nil {
    w.CloseWithError(fmt.Errorf("write failed: %v", err))
    return
  }
  w.Close()
}()

读取方接收到错误后会终止读取:

_, err := r.Read(buf)
if err != nil {
  // 可能是 EOF,也可能是 CloseWithError 传入的错误
  fmt.Println("read error:", err)
}

务必确保两端都关闭,防止资源泄漏。

基本上就这些。io.Pipe 虽然简单,但在流式数据处理中非常实用,关键是理解其同步特性和生命周期管理。

相关专题

更多
if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

713

2023.08.22

scripterror怎么解决
scripterror怎么解决

scripterror的解决办法有检查语法、文件路径、检查网络连接、浏览器兼容性、使用try-catch语句、使用开发者工具进行调试、更新浏览器和JavaScript库或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

184

2023.10.18

500error怎么解决
500error怎么解决

500error的解决办法有检查服务器日志、检查代码、检查服务器配置、更新软件版本、重新启动服务、调试代码和寻求帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

268

2023.10.25

printf用法大全
printf用法大全

php中文网为大家提供printf用法大全,以及其他printf函数的相关文章、相关下载资源以及各种相关课程,供大家免费下载体验。

72

2023.06.20

fprintf和printf的区别
fprintf和printf的区别

fprintf和printf的区别在于输出的目标不同,printf输出到标准输出流,而fprintf输出到指定的文件流。根据需要选择合适的函数来进行输出操作。更多关于fprintf和printf的相关文章详情请看本专题下面的文章。php中文网欢迎大家前来学习。

277

2023.11.28

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

994

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

53

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

243

2025.12.29

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

150

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 3.2万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号