mirror of
https://github.com/golang/go.git
synced 2025-12-08 06:10:04 +00:00
Add ReadFrom and WriteTo methods to bytes.Buffer, to enable i/o without buffer allocation.
Use them in Copy and Copyn. Speed up ReadFile by using ReadFrom and avoiding Copy altogether (a minor win). R=rsc, gri CC=golang-dev https://golang.org/cl/166041
This commit is contained in:
parent
7e7008fa5e
commit
bc3e34759c
5 changed files with 194 additions and 5 deletions
|
|
@ -7,6 +7,7 @@ package bytes
|
|||
// Simple byte buffer for marshaling data.
|
||||
|
||||
import (
|
||||
"io";
|
||||
"os";
|
||||
)
|
||||
|
||||
|
|
@ -91,6 +92,60 @@ func (b *Buffer) Write(p []byte) (n int, err os.Error) {
|
|||
return n, nil;
|
||||
}
|
||||
|
||||
// MinRead is the minimum slice size passed to a Read call by
|
||||
// Buffer.ReadFrom. As long as the Buffer has at least MinRead bytes beyond
|
||||
// what is required to hold the contents of r, ReadFrom will not grow the
|
||||
// underlying buffer.
|
||||
const MinRead = 512
|
||||
|
||||
// ReadFrom reads data from r until EOF and appends it to the buffer.
|
||||
// The return value n is the number of bytes read.
|
||||
// Any error except os.EOF encountered during the read
|
||||
// is also returned.
|
||||
func (b *Buffer) ReadFrom(r io.Reader) (n int64, err os.Error) {
|
||||
for {
|
||||
if cap(b.buf)-len(b.buf) < MinRead {
|
||||
var newBuf []byte;
|
||||
// can we get space without allocation?
|
||||
if b.off+cap(b.buf)-len(b.buf) >= MinRead {
|
||||
// reuse beginning of buffer
|
||||
newBuf = b.buf[0 : len(b.buf)-b.off]
|
||||
} else {
|
||||
// not enough space at end; put space on end
|
||||
newBuf = make([]byte, len(b.buf)-b.off, 2*(cap(b.buf)-b.off)+MinRead)
|
||||
}
|
||||
copy(newBuf, b.buf[b.off:]);
|
||||
b.buf = newBuf;
|
||||
b.off = 0;
|
||||
}
|
||||
m, e := r.Read(b.buf[len(b.buf):cap(b.buf)]);
|
||||
b.buf = b.buf[b.off : len(b.buf)+m];
|
||||
n += int64(m);
|
||||
if e == os.EOF {
|
||||
break
|
||||
}
|
||||
if e != nil {
|
||||
return n, e
|
||||
}
|
||||
}
|
||||
return n, nil; // err is EOF, so return nil explicitly
|
||||
}
|
||||
|
||||
// WriteTo writes data to w until the buffer is drained or an error
|
||||
// occurs. The return value n is the number of bytes written.
|
||||
// Any error encountered during the write is also returned.
|
||||
func (b *Buffer) WriteTo(w io.Writer) (n int64, err os.Error) {
|
||||
for b.off < len(b.buf) {
|
||||
m, e := w.Write(b.buf[b.off:]);
|
||||
n += int64(m);
|
||||
b.off += m;
|
||||
if e != nil {
|
||||
return n, e
|
||||
}
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
// WriteString appends the contents of s to the buffer. The return
|
||||
// value n is the length of s; err is always nil.
|
||||
func (b *Buffer) WriteString(s string) (n int, err os.Error) {
|
||||
|
|
|
|||
|
|
@ -240,3 +240,25 @@ func TestNil(t *testing.T) {
|
|||
t.Error("expcted <nil>; got %q", b.String())
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
func TestReadFrom(t *testing.T) {
|
||||
var buf Buffer;
|
||||
for i := 3; i < 30; i += 3 {
|
||||
s := fillBytes(t, "TestReadFrom (1)", &buf, "", 5, bytes[0:len(bytes)/i]);
|
||||
var b Buffer;
|
||||
b.ReadFrom(&buf);
|
||||
empty(t, "TestReadFrom (2)", &b, s, make([]byte, len(data)));
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
func TestWriteTo(t *testing.T) {
|
||||
var buf Buffer;
|
||||
for i := 3; i < 30; i += 3 {
|
||||
s := fillBytes(t, "TestReadFrom (1)", &buf, "", 5, bytes[0:len(bytes)/i]);
|
||||
var b Buffer;
|
||||
buf.WriteTo(&b);
|
||||
empty(t, "TestReadFrom (2)", &b, s, make([]byte, len(data)));
|
||||
}
|
||||
}
|
||||
|
|
|
|||
|
|
@ -112,6 +112,16 @@ type ReadWriteSeeker interface {
|
|||
Seeker;
|
||||
}
|
||||
|
||||
// ReaderFrom is the interface that wraps the ReadFrom method.
|
||||
type ReaderFrom interface {
|
||||
ReadFrom(r Reader) (n int64, err os.Error);
|
||||
}
|
||||
|
||||
// WriterTo is the interface that wraps the WriteTo method.
|
||||
type WriterTo interface {
|
||||
WriteTo(w Writer) (n int64, err os.Error);
|
||||
}
|
||||
|
||||
// ReaderAt is the interface that wraps the basic ReadAt method.
|
||||
//
|
||||
// ReadAt reads len(p) bytes into p starting at offset off in the
|
||||
|
|
@ -178,7 +188,15 @@ func ReadFull(r Reader, buf []byte) (n int, err os.Error) {
|
|||
|
||||
// Copyn copies n bytes (or until an error) from src to dst.
|
||||
// It returns the number of bytes copied and the error, if any.
|
||||
//
|
||||
// If dst implements the ReaderFrom interface,
|
||||
// the copy is implemented by calling dst.ReadFrom(src).
|
||||
func Copyn(dst Writer, src Reader, n int64) (written int64, err os.Error) {
|
||||
// If the writer has a ReadFrom method, use it to to do the copy.
|
||||
// Avoids a buffer allocation and a copy.
|
||||
if rt, ok := dst.(ReaderFrom); ok {
|
||||
return rt.ReadFrom(LimitReader(src, n))
|
||||
}
|
||||
buf := make([]byte, 32*1024);
|
||||
for written < n {
|
||||
l := len(buf);
|
||||
|
|
@ -211,7 +229,21 @@ func Copyn(dst Writer, src Reader, n int64) (written int64, err os.Error) {
|
|||
// Copy copies from src to dst until either EOF is reached
|
||||
// on src or an error occurs. It returns the number of bytes
|
||||
// copied and the error, if any.
|
||||
//
|
||||
// If dst implements the ReaderFrom interface,
|
||||
// the copy is implemented by calling dst.ReadFrom(src).
|
||||
// Otherwise, if src implements the WriterTo interface,
|
||||
// the copy is implemented by calling src.WriteTo(dst).
|
||||
func Copy(dst Writer, src Reader) (written int64, err os.Error) {
|
||||
// If the writer has a ReadFrom method, use it to to do the copy.
|
||||
// Avoids an allocation and a copy.
|
||||
if rt, ok := dst.(ReaderFrom); ok {
|
||||
return rt.ReadFrom(src)
|
||||
}
|
||||
// Similarly, if the reader has a WriteTo method, use it to to do the copy.
|
||||
if wt, ok := src.(WriterTo); ok {
|
||||
return wt.WriteTo(dst)
|
||||
}
|
||||
buf := make([]byte, 32*1024);
|
||||
for {
|
||||
nr, er := src.Read(buf);
|
||||
|
|
|
|||
80
src/pkg/io/io_test.go
Normal file
80
src/pkg/io/io_test.go
Normal file
|
|
@ -0,0 +1,80 @@
|
|||
// Copyright 2009 The Go Authors. All rights reserved.
|
||||
// Use of this source code is governed by a BSD-style
|
||||
// license that can be found in the LICENSE file.
|
||||
|
||||
package io_test
|
||||
|
||||
import (
|
||||
"bytes";
|
||||
. "io";
|
||||
"testing";
|
||||
)
|
||||
|
||||
// An version of bytes.Buffer without ReadFrom and WriteTo
|
||||
type Buffer struct {
|
||||
bytes.Buffer;
|
||||
ReaderFrom; // conflicts with and hides bytes.Buffer's ReaderFrom.
|
||||
WriterTo; // conflicts with and hides bytes.Buffer's WriterTo.
|
||||
}
|
||||
|
||||
// Simple tests, primarily to verify the ReadFrom and WriteTo callouts inside Copy and Copyn.
|
||||
|
||||
func TestCopy(t *testing.T) {
|
||||
rb := new(Buffer);
|
||||
wb := new(Buffer);
|
||||
rb.WriteString("hello, world.");
|
||||
Copy(wb, rb);
|
||||
if wb.String() != "hello, world." {
|
||||
t.Errorf("Copy did not work properly")
|
||||
}
|
||||
}
|
||||
|
||||
func TestCopyReadFrom(t *testing.T) {
|
||||
rb := new(Buffer);
|
||||
wb := new(bytes.Buffer); // implements ReadFrom.
|
||||
rb.WriteString("hello, world.");
|
||||
Copy(wb, rb);
|
||||
if wb.String() != "hello, world." {
|
||||
t.Errorf("Copy did not work properly")
|
||||
}
|
||||
}
|
||||
|
||||
func TestCopyWriteTo(t *testing.T) {
|
||||
rb := new(bytes.Buffer); // implements WriteTo.
|
||||
wb := new(Buffer);
|
||||
rb.WriteString("hello, world.");
|
||||
Copy(wb, rb);
|
||||
if wb.String() != "hello, world." {
|
||||
t.Errorf("Copy did not work properly")
|
||||
}
|
||||
}
|
||||
|
||||
func TestCopyn(t *testing.T) {
|
||||
rb := new(Buffer);
|
||||
wb := new(Buffer);
|
||||
rb.WriteString("hello, world.");
|
||||
Copyn(wb, rb, 5);
|
||||
if wb.String() != "hello" {
|
||||
t.Errorf("Copyn did not work properly")
|
||||
}
|
||||
}
|
||||
|
||||
func TestCopynReadFrom(t *testing.T) {
|
||||
rb := new(Buffer);
|
||||
wb := new(bytes.Buffer); // implements ReadFrom.
|
||||
rb.WriteString("hello");
|
||||
Copyn(wb, rb, 5);
|
||||
if wb.String() != "hello" {
|
||||
t.Errorf("Copyn did not work properly")
|
||||
}
|
||||
}
|
||||
|
||||
func TestCopynWriteTo(t *testing.T) {
|
||||
rb := new(bytes.Buffer); // implements WriteTo.
|
||||
wb := new(Buffer);
|
||||
rb.WriteString("hello, world.");
|
||||
Copyn(wb, rb, 5);
|
||||
if wb.String() != "hello" {
|
||||
t.Errorf("Copyn did not work properly")
|
||||
}
|
||||
}
|
||||
|
|
@ -34,15 +34,15 @@ func ReadFile(filename string) ([]byte, os.Error) {
|
|||
if err != nil && dir.Size < 2e9 { // Don't preallocate a huge buffer, just in case.
|
||||
n = dir.Size
|
||||
}
|
||||
if n == 0 {
|
||||
n = 1024 // No idea what's right, but zero isn't.
|
||||
}
|
||||
// Add a little extra in case Size is zero, and to avoid another allocation after
|
||||
// Read has filled the buffer.
|
||||
n += bytes.MinRead;
|
||||
// Pre-allocate the correct size of buffer, then set its size to zero. The
|
||||
// Buffer will read into the allocated space cheaply. If the size was wrong,
|
||||
// we'll either waste some space off the end or reallocate as needed, but
|
||||
// in the overwhelmingly common case we'll get it just right.
|
||||
buf := bytes.NewBuffer(make([]byte, n)[0:0]);
|
||||
_, err = io.Copy(buf, f);
|
||||
buf := bytes.NewBuffer(make([]byte, 0, n));
|
||||
_, err = buf.ReadFrom(f);
|
||||
return buf.Bytes(), err;
|
||||
}
|
||||
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue