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.
|
// Simple byte buffer for marshaling data.
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"io";
|
||||||
"os";
|
"os";
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
@ -91,6 +92,60 @@ func (b *Buffer) Write(p []byte) (n int, err os.Error) {
|
||||||
return n, nil;
|
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
|
// WriteString appends the contents of s to the buffer. The return
|
||||||
// value n is the length of s; err is always nil.
|
// value n is the length of s; err is always nil.
|
||||||
func (b *Buffer) WriteString(s string) (n int, err os.Error) {
|
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())
|
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;
|
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.
|
// ReaderAt is the interface that wraps the basic ReadAt method.
|
||||||
//
|
//
|
||||||
// ReadAt reads len(p) bytes into p starting at offset off in the
|
// 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.
|
// Copyn copies n bytes (or until an error) from src to dst.
|
||||||
// It returns the number of bytes copied and the error, if any.
|
// 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) {
|
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);
|
buf := make([]byte, 32*1024);
|
||||||
for written < n {
|
for written < n {
|
||||||
l := len(buf);
|
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
|
// Copy copies from src to dst until either EOF is reached
|
||||||
// on src or an error occurs. It returns the number of bytes
|
// on src or an error occurs. It returns the number of bytes
|
||||||
// copied and the error, if any.
|
// 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) {
|
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);
|
buf := make([]byte, 32*1024);
|
||||||
for {
|
for {
|
||||||
nr, er := src.Read(buf);
|
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.
|
if err != nil && dir.Size < 2e9 { // Don't preallocate a huge buffer, just in case.
|
||||||
n = dir.Size
|
n = dir.Size
|
||||||
}
|
}
|
||||||
if n == 0 {
|
// Add a little extra in case Size is zero, and to avoid another allocation after
|
||||||
n = 1024 // No idea what's right, but zero isn't.
|
// Read has filled the buffer.
|
||||||
}
|
n += bytes.MinRead;
|
||||||
// Pre-allocate the correct size of buffer, then set its size to zero. The
|
// 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,
|
// 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
|
// 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.
|
// in the overwhelmingly common case we'll get it just right.
|
||||||
buf := bytes.NewBuffer(make([]byte, n)[0:0]);
|
buf := bytes.NewBuffer(make([]byte, 0, n));
|
||||||
_, err = io.Copy(buf, f);
|
_, err = buf.ReadFrom(f);
|
||||||
return buf.Bytes(), err;
|
return buf.Bytes(), err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue