登录
首页 >  Golang >  Go教程

Go语言io pipe源码分析详情

来源:脚本之家

时间:2023-01-07 12:08:11 237浏览 收藏

IT行业相对于一般传统行业,发展更新速度更快,一旦停止了学习,很快就会被行业所淘汰。所以我们需要踏踏实实的不断学习,精进自己的技术,尤其是初学者。今天golang学习网给大家整理了《Go语言io pipe源码分析详情》,聊聊pipe、IO、源码、分析,我们一起来看看吧!

pipe.go分析:

  • 这个文件使用到了errors包,也是用到了sync库.
  • 文件说明:pipe是一个适配器,用于连接Reader和Writer.

1.结构分析

对外暴露的是一个构造函数和构造的两个对象. 两个对象分别暴露了方法,同时这两个对象还有一个共同的底层对象. 实际上,这两个对象暴露的方法是直接调用底层对象的, 那么核心还是在底层对象上,只是通过两个对象和一个构造方法将底层对象的细节隐藏了.

2.pipe sruct分析

pipe的方法不多,新的写法却不少.   

 type atomicError struct{ v atomic.Value }

    func (a *atomicError) Store(err error) {
      a.v.Store(struct{ error }{err})
    }
    func (a *atomicError) Load() error {
      err, _ := a.v.Load().(struct{ error })
      return err.error
    }

atomicError提供了error的原子读写. 

  type pipe struct {
      wrMu sync.Mutex // Serializes Write operations
      wrCh chan []byte
      rdCh chan int

      once sync.Once // Protects closing done
      done chan struct{}
      rerr atomicError
      werr atomicError
    }

可以看到pipe结构体中主要分两块:

  • 读写信道
    • 两个无缓冲信道
    • 一个互斥量(保护暴露的写函数)
  • 结束标识
    • once保证done的关闭只执行一次
    • done标志整个读写的结束
    • 剩下两个用于存储读写错误
    • PipeReader/PipeWriter的分析

3.PipeReader对外暴露的是读/关闭

    type PipeReader struct {
      p *pipe
    }

    func (r *PipeReader) Read(data []byte) (n int, err error) {
      return r.p.Read(data)
    }

    func (r *PipeReader) Close() error {
      return r.CloseWithError(nil)
    }

    func (r *PipeReader) CloseWithError(err error) error {
      return r.p.CloseRead(err)
    }

PipeWriter对外暴露的是写/关闭 

    type PipeWriter struct {
       p *pipe
     }

    func (w *PipeWriter) Write(data []byte) (n int, err error) {
      return w.p.Write(data)
    }

    func (w *PipeWriter) Close() error {
      return w.CloseWithError(nil)
    }

    func (w *PipeWriter) CloseWithError(err error) error {
      return w.p.CloseWrite(err)
    }

他们的方法集都是指针接收者.具体方法的实现是通过pipe的方法完成的. pipe的方法更加明确:读/获取读错误/结束读写并设置读错误; 写/获取写错误/结束读写并设置写错误.思路相当明确.

下面主要分析pipe的读写 

   func (p *pipe) Read(b []byte) (n int, err error) {
      select {
      case  0; once = false {
        select {
        case p.wrCh 
<p>读写都是利用两个阶段的<code>select</code>来完成,第一个阶段的select是判断读写有没有结束, 第二阶段处理实际的读写.</p>
<blockquote><p><strong>Read</strong></p>
<ul><li>每次将读的数量写到读信道</li></ul><p><strong>Write</strong></p>
<ul><li>先将缓冲写到写信道,再从读信道中获取读字节数,最后调整缓冲</li><li>如果缓冲太大,一次读没读完,就将写的过程多来几遍,知道缓冲全部写完</li></ul></blockquote>
<h2>4.写法</h2>
<p><code>PipeWriter/PipeReader</code>对外暴露的关闭,其实只可以保留一个<code>CloseWithError</code>, 但是为了方便客户(调用者),还是拆成两个,其实可以做测试比较一下. 性能测试发现拆成两个或写成一个可选参函数,性能上差别不大, 那这种写法的主要作用是让暴露的方法更加清晰易懂.</p>
<p><code>pipe.Write</code>中,for循环带有once参数,可以保证循环至少来一次, 算是do while的一种实现.</p>
<h2>5.总结</h2>
<p>不管是PipeReader/PipeWriter,还是pipe,都对Reader/Writer有(部分)实现.</p>
<p>另外还有一些细节没有说道:读写错误和EOF.</p>
<p><strong>反思:</strong>本次阅读是先理代码后看文档,才发现关于error部分没有留心到, 后面还是先文档后代码,这样效率会高一点.</p>
<p>理论要掌握,实操不能落!以上关于《Go语言io pipe源码分析详情》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注golang学习网公众号吧!</p>
声明:本文转载于:脚本之家 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>
评论列表