RWBY === <!-- .slide: data-background="pink" --> <!-- .slide: data-transition="zoom" --> Reader/Writer/Close :dizzy: > [name=郭學聰 Hsueh-Tsung Kuo] [time=Thu, 27 Dec 2018] [color=red] --- <!-- .slide: data-transition="convex" --> ## who am I? ![fieliapm](https://pbs.twimg.com/profile_images/591670980021387264/aZAYLRUe_400x400.png) ---- <!-- .slide: data-transition="convex" --> * programmer from Rayark, a game company in Taiwan * backend engineer, focus on common service * usually develop something related to my work in Python, Ruby, Golang, C# * built almost entire VOEZ game server by myself only * supported Sdorica backend development --- <!-- .slide: data-transition="convex" --> ## outline ---- <!-- .slide: data-transition="convex" --> * Reader/Writer spec & definition * Reader/Writer interface variation * why ioutil? * defer & Close() * customize Reader/Writer/Closer * from block cipher to streaming --- <!-- .slide: data-transition="convex" --> ## RWBY? ---- <!-- .slide: data-transition="convex" --> ![RWBY: Volume 5](https://m.media-amazon.com/images/M/MV5BNmE4MzE1OTgtMGRjZC00MmVhLWFhMjgtMWU4MTU0YjZlNzY1XkEyXkFqcGdeQXVyNDI4NjI0MDM@._V1_SY999_CR0,0,726,999_AL_.jpg =363x499) --- <!-- .slide: data-transition="convex" --> ## Reader/Writer spec & definition * Reader https://golang.org/pkg/io/#Reader * Writer https://golang.org/pkg/io/#Writer * Closer https://golang.org/pkg/io/#Closer ---- <!-- .slide: data-transition="convex" --> ### Reader ```go= type Reader interface { Read(p []byte) (n int, err error) } ``` ---- <!-- .slide: data-transition="convex" --> ### Reader Key point: * Callers should always process the `n > 0` bytes returned before considering the error err. ---- <!-- .slide: data-transition="convex" --> :o: ```go= func ReadAtLeast(r Reader, buf []byte, min int) (n int, err error) { if len(buf) < min { return 0, ErrShortBuffer } for n < min && err == nil { var nn int nn, err = r.Read(buf[n:]) n += nn } if n >= min { err = nil } else if n > 0 && err == EOF { err = ErrUnexpectedEOF } return } ``` https://golang.org/src/io/io.go?s=10817:10883#L294 ---- <!-- .slide: data-transition="convex" --> :o: ```go= func (b *Buffer) ReadFrom(r io.Reader) (n int64, err error) { b.lastRead = opInvalid for { i := b.grow(MinRead) b.buf = b.buf[:i] m, e := r.Read(b.buf[i:cap(b.buf)]) if m < 0 { panic(errNegativeRead) } b.buf = b.buf[:i+m] n += int64(m) if e == io.EOF { return n, nil // e is EOF, so return nil explicitly } if e != nil { return n, e } } } ``` https://golang.org/src/bytes/buffer.go?s=6926:6985#L191 ---- <!-- .slide: data-transition="convex" --> ### Writer ```go= type Writer interface { Write(p []byte) (n int, err error) } ``` ---- <!-- .slide: data-transition="convex" --> ### Writer Key point: * Write must return a non-nil error if it returns `n < len(p)`. ---- <!-- .slide: data-transition="convex" --> ### Closer ```go= type Closer interface { Close() error } ``` ---- <!-- .slide: data-transition="convex" --> ### Closer Key point: * The behavior of Close after the first call is undefined. --- <!-- .slide: data-transition="convex" --> ## Reader/Writer interface variation See: https://golang.org/pkg/io/ ---- <!-- .slide: data-transition="convex" --> ### ReaderAt ```go= type ReaderAt interface { ReadAt(p []byte, off int64) (n int, err error) } ``` ---- <!-- .slide: data-transition="convex" --> ### ReaderAt Key point: * When ReadAt returns `n < len(p)`, it returns a non-nil error explaining why more bytes were not returned. https://golang.org/pkg/io/#ReaderAt --- <!-- .slide: data-transition="convex" --> ## why io & io/ioutil? * Key point of Reader ---- <!-- .slide: data-transition="convex" --> to ensure Read() fill buf ```go= func ReadAtLeast(r Reader, buf []byte, min int) (n int, err error) func ReadFull(r Reader, buf []byte) (n int, err error) ``` https://golang.org/pkg/io/#ReadAtLeast https://golang.org/pkg/io/#ReadFull ---- <!-- .slide: data-transition="convex" --> or use utility function to read/write all ```go= var Discard io.Writer = devNull(0) func NopCloser(r io.Reader) io.ReadCloser func ReadAll(r io.Reader) ([]byte, error) func ReadDir(dirname string) ([]os.FileInfo, error) func ReadFile(filename string) ([]byte, error) func TempDir(dir, prefix string) (name string, err error) func TempFile(dir, pattern string) (f *os.File, err error) func WriteFile(filename string, data []byte, perm os.FileMode) error ``` https://golang.org/pkg/io/ioutil/ --- <!-- .slide: data-transition="convex" --> ## defer & Close() ---- <!-- .slide: data-transition="convex" --> ### may miss error ```go= func RWBY() (err error) { defer f.Close() } ``` ---- <!-- .slide: data-transition="convex" --> ### don't miss any error ```go= func RWBY() (err error) { defer func() { cerr := f.Close() if err == nil { err = cerr } }() } ``` ---- <!-- .slide: data-transition="convex" --> ### don't miss any error ```go= func safeClose(c io.Closer, err *error) { cerr := c.Close() if *err == nil { *err = cerr } } func RWBY() (err error) { defer safeClose(f, &err) } ``` --- <!-- .slide: data-transition="convex" --> ## customize Reader/Writer/Closer ---- <!-- .slide: data-transition="convex" --> ### WriterCounter ```go= type WriterCounter struct { writer io.Writer count int64 } func (wc *WriterCounter) Write(p []byte) (n int, err error) { n, err = wc.writer.Write(p) atomic.AddInt64(&wc.count, int64(n)) return } func (wc *WriterCounter) Count() int64 { return atomic.LoadInt64(&wc.count) } func NewWriterCounter(w io.Writer) *WriterCounter { return &WriterCounter{writer: w} } ``` ---- <!-- .slide: data-transition="convex" --> ### WriteNopCloser ```go= type writeNopCloser struct { io.Writer } func (_ writeNopCloser) Close() error { return nil } func WriteNopCloser(w io.Writer) io.WriteCloser { return writeNopCloser{Writer: w} } ``` ---- <!-- .slide: data-transition="convex" --> ### ChainCloseReadCloser ```go= type ChainCloseReadCloser struct { io.ReadCloser underlyingReadClosers []io.ReadCloser } func (chainCloseReadCloser *ChainCloseReadCloser) Close() error { err := chainCloseReadCloser.ReadCloser.Close() for _, underlyingReadCloser := range chainCloseReadCloser.underlyingReadClosers { cerr := underlyingReadCloser.Close() // always return first close error if err == nil { err = cerr } } return err } func NewChainCloseReadCloser(readCloser io.ReadCloser, underlyingReadClosers ...io.ReadCloser) io.ReadCloser { return &ChainCloseReadCloser{ReadCloser: readCloser, underlyingReadClosers: underlyingReadClosers} } ``` --- <!-- .slide: data-transition="convex" --> ## from block cipher to streaming ---- <!-- .slide: data-transition="convex" --> * ??? --- <!-- .slide: data-transition="convex" --> ## conclusion ---- <!-- .slide: data-transition="convex" --> > :hash: Reader/Writer/Closer 博大精深 > [name=Hsueh-Tsung Kuo] [time=Thu, 26 Apr 2018] [color=red] ---- <!-- .slide: data-transition="convex" --> ### special thanks * komod --- <!-- .slide: data-transition="zoom" --> ## Q&A --- <style> .reveal { background: #FFDFEF; color: black; } .reveal h2, .reveal h3, .reveal h4 { color: black; } .reveal code { font-size: 14px !important; line-height: 1.2; } .rightpart{ float:right; width:50%; } .leftpart{ margin-right: 50% !important; height:50%; } .reveal section img { background:none; border:none; box-shadow:none; } p.blo { font-size: 50px !important; background:#B6BDBB; border:1px solid silver; display:inline-block; padding:0.5em 0.75em; border-radius: 10px; box-shadow: 5px 5px 5px #666; } p.blo1 { background: #c7c2bb; } p.blo2 { background: #b8c0c8; } p.blo3 { background: #c7cedd; } p.bloT { font-size: 60px !important; background:#B6BDD3; border:1px solid silver; display:inline-block; padding:0.5em 0.75em; border-radius: 8px; box-shadow: 1px 2px 5px #333; } p.bloA { background: #B6BDE3; } p.bloB { background: #E3BDB3; } .slide-number{ margin-bottom:10px !important; width:100%; text-align:center; font-size:25px !important; background-color:transparent !important; } iframe.myclass{ width:100px; height:100px; bottom:0; left:0; position:fixed; border:none; z-index:99999; } h1.raw { color: #fff; background-image: linear-gradient(90deg,#f35626,#feab3a); -webkit-background-clip: text; -webkit-text-fill-color: transparent; animation: hue 5s infinite linear; } @keyframes hue { from { filter: hue-rotate(0deg); } to { filter: hue-rotate(360deg); } } .progress{ height:14px !important; } .progress span{ height:14px !important; background: url("") repeat-x !important; } .progress span:after, .progress span.nyancat{ content: ""; background: url('') !important; width: 34px !important; height: 21px !important; border: none !important; float:right; margin-top:-7px; margin-right:-10px; } </style>
{"metaMigratedAt":"2023-06-14T19:33:42.492Z","metaMigratedFrom":"Content","title":"RWBY","breaks":true,"contributors":"[{\"id\":\"ea27dcd7-a3f2-47c2-b25e-6760e7936c38\",\"add\":29366,\"del\":14264}]"}
    618 views