Go CodeReviewComments
https://github.com/golang/go/wiki/CodeReviewComments
To avoid unexpected aliasing, be careful when copying a struct from another package. For example, the bytes.Buffer type contains a
[]byte
slice. If you copy aBuffer
, the slice in the copy may alias the array in the original, causing subsequent method calls to have surprising effects.In general, do not copy a value of type
T
if its methods are associated with the pointer type,*T
.
When you spawn goroutines, make it clear when - or whether - they exit.
The import . form can be useful in tests that, due to circular dependencies, cannot be made part of the package being tested.
Don’t pass pointers as function arguments just to save a few bytes. If a function refers to its argument
x
only as*x
throughout, then the argument shouldn’t be a pointer. Common instances of this include passing a pointer to a string (*string
) or a pointer to an interface value (*io.Reader
). In both cases the value itself is a fixed size and can be passed directly. This advice does not apply to large structs, or even small structs that might grow.
Choosing whether to use a value or pointer receiver on methods can be difficult, especially to new Go programmers. Some useful guidelines:
- If the receiver is a map, func or chan, don’t use a pointer to them. If the receiver is a slice and the method doesn’t reslice or reallocate the slice, don’t use a pointer to it.
- If the method needs to mutate the receiver, the receiver must be a pointer.
- If the receiver is a struct that contains a sync.Mutex or similar synchronizing field, the receiver must be a pointer to avoid copying.
- If the receiver is a large struct or array, a pointer receiver is more efficient. How large is large? Assume it’s equivalent to passing all its elements as arguments to the method. If that feels too large, it’s also too large for the receiver.
- If the receiver is a struct, array or slice and any of its elements is a pointer to something that might be mutating, prefer a pointer receiver, as it will make the intention clearer to the reader.
- If the receiver is a small array or struct that is naturally a value type (for instance, something like the time.Time type), with no mutable fields and no pointers, or is just a simple basic type such as int or string, a value receiver makes sense. A value receiver can reduce the amount of garbage that can be generated; if a value is passed to a value method, an on-stack copy can be used instead of allocating on the heap. (The compiler tries to be smart about avoiding this allocation, but it can’t always succeed.) Don’t choose a value receiver type for this reason without profiling first.
- Don’t mix receiver types. Choose either pointers or struct types for all available methods.
- Finally, when in doubt, use a pointer receiver.