Multiple return values
func (file *File) Write(b []byte) (n int, err error)
Named result parameters
The names are not mandatory but they can make code shorter and clearer: they’re documentation.
When named, they are initialized to the zero values for their types when the function begins.func nextInt(b []byte, pos int) (int, int)
func nextInt(b []byte, pos int) (value, nextPos int) {
Defer
Go’s defer
statement schedules a function call (the deferred function) to be run immediately before the calling function returns.
// Contents returns the file's contents as a string.
func Contents(filename string) (string, error) {
f, err := os.Open(filename)
if err != nil {
return "", err
}
defer f.Close() // f.Close will run when we're finished.
var result []byte
buf := make([]byte, 100)
for {
n, err := f.Read(buf[0:])
result = append(result, buf[0:n]...) // append is discussed later.
if err != nil {
if err == io.EOF {
break
}
return "", err // f will be closed if we return here.
}
}
return string(result), nil // f will be closed if we return here.
}
two advantages:
- First, it guarantees that you will never forget to close the file;
- Second, it means that the close sits near the open, which is much clearer than placing it at the end of the function.
The arguments to the deferred function are evaluated when the defer executes, not when the call executes.
Deferred functions are executed in LIFO order.
for i := 0; i < 5; i++ {
defer fmt.Printf("%d ", i)
}
// 4 3 2 1 0 to be printed