vicever / go-advices

List of advices and tricks in the Go's world \ʕ◔ϖ◔ʔ/

Geek Repo:Geek Repo

Github PK Tool:Github PK Tool

Go-advices

Code

  • go fmt your code, make everyone happier
  • multiple if statements can be collapsed into switch
  • use chan struct{} to pass signal, chan bool makes it less clear
  • prefer 30 * time.Second instead of time.Duration(30) * time.Second
  • always wrap for-select idiom to a function
  • group const declarations by type and var by logic and/or type
  • every blocking or IO function call should be cancelable or at least timeoutable
  • implement Stringer interface for integers const values
  • check your defer's error
    defer func() {
        err := ocp.Close()
        if err != nil {
            rerr = err
        }
    }()
  • don't use checkErr function which panics or does os.Exit
  • use panic only in very specific situations, you have to handle error
  • don't use alias for enums 'cause this breaks type safety
    package main
    type Status = int
    type Format = int // remove `=` to have type safety
    
    const A Status = 1
    const B Format = 1
    
    func main() {
      println(A == B)
    }
  • if you're going to omit returning params, do it explicitly
    • so prefer this _ = f() to this f()
  • we've a short form for slice initialization a := []T{}
  • iterate over array or slice using range loop
    • instead of for i := 3; i < 7; i++ {...} prefer for _, c := range a[3:7] {...}
  • use backquote(`) for multiline strings
  • skip unused param with _
    func f(a int, _ string() {}
  • use time.Before and time.After to compare time, avoid time.Sub
  • always pass context as a first param to a func with a ctx name
  • few params of the same type can be defined in a short way
    func f(a int, b int, s string, p string)
    func f(a, b int, s, p string)
  • the zero value of a slice is nil
      var s []int
      fmt.Println(s, len(s), cap(s))
      if s == nil {
        fmt.Println("nil!")
      }
      // Output:
      // [] 0 0
      // nil!
    var a []string
    b := []string{}
    
    fmt.Println(reflect.DeepEqual(a, []string{}))
    fmt.Println(reflect.DeepEqual(b, []string{}))
    // Output:
    // false
    // true
  • do not compare enum types with <, >, <= and >=
    • use explicit values, don't do this:
    value := reflect.ValueOf(object)
    kind := value.Kind()
    if kind >= reflect.Chan && kind <= reflect.Slice {
      // ...
    }
  • use %+v to print data with sufficient details

Concurrency

  • best candidate to make something once in a thread-safe way is sync.Once
    • don't use flags, mutexes, channels or atomics
  • to block forever use select{}, omit channels, waiting for a signal

Performance

  • do not omit defer
    • 200ns speedup is negligible in most cases
  • always close http body aka defer r.Body.Close()
    • unless you need leaked goroutine
  • filtering without allocating
      b := a[:0]
      for _, x := range a {
      	if f(x) {
      	    b = append(b, x)
      	}
      }
  • time.Time has pointer field time.Location and this is bad for go GC
    • it's relevant only for big number of time.Time, use timestamp instead
  • prefer regexp.MustCompile instead of regexp.Compile
    • in most cases your regex is immutable, so init it in func init
  • do not overuse fmt.Sprintf in your hot path. It is costly due to maintaining the buffer pool and dynamic dispatches for interfaces.
    • if you are doing fmt.Sprintf("%s%s", var1, var2), consider simple string concatenation.
    • if you are doing fmt.Sprintf("%x", var), consider using hex.EncodeToString or strconv.FormatInt(var, 16)
  • always discard body e.g. io.Copy(ioutil.Discard, resp.Body) if you don't use it
    • HTTP client's Transport will not reuse connections unless the body is read to completion and closed
      res, _ := client.Do(req)
      io.Copy(ioutil.Discard, res.Body)
      defer res.Body.Close()
  • don't use defer in a loop or you'll get a small memory leak
    • 'cause defers will grow your stack without the reason
  • don't forget to stop ticker, unless you need a leaked channel
    ticker := time.NewTicker(1 * time.Second)
    defer ticker.Stop()
  • use custom marshaler to speed up marshaling
    func (entry Entry) MarshalJSON() ([]byte, error) {
      buffer := bytes.NewBufferString("{")
      first := true
      for key, value := range entry {
      	jsonValue, err := json.Marshal(value)
      	if err != nil {
      		return nil, err
      	}
      	if !first {
      		buffer.WriteString(",")
      	}
      	first = false
      	buffer.WriteString(key + ":" + string(jsonValue))
      }
      buffer.WriteString("}")
      return buffer.Bytes(), nil
    }
  • sync.Map isn't a silver bullet, do not use it without a strong reasons
  • storing non-pointer values in sync.Pool allocates memory
  • regular expressions are mutexed
    • to avoid performance degradation in concurrent programs make a copy:
    re, err := regexp.Compile(pattern)
    re2 := re.Copy()

Build

Testing

  • prefer package_test name for tests, rather than package
  • go test -short allows to reduce set of tests to be runned
    func TestSomething(t *testing.T) {
      if testing.Short() {
        t.Skip("skipping test in short mode.")
      }
    }
  • skip test deppending on architecture
    if runtime.GOARM == "arm" {
      t.Skip("this doesn't work under ARM")
    }
  • track your allocations with testing.AllocsPerRun

Tools

  • quick replace gofmt -w -l -r "panic(err) -> log.Error(err)" .
  • go list allows to find all direct and transitive dependencies
    • go list -f '{{ .Imports }}' package
    • go list -f '{{ .Deps }}' package
  • for fast benchmark comparison we've a benchcmp tool

Misc

  • dump goroutines https://stackoverflow.com/a/27398062/433041
    go func() {
      sigs := make(chan os.Signal, 1)
      signal.Notify(sigs, syscall.SIGQUIT)
      buf := make([]byte, 1<<20)
      for {
        <-sigs
        stacklen := runtime.Stack(buf, true)
        log.Printf("=== received SIGQUIT ===\n*** goroutine dump...\n%s\n*** end\n", buf[:stacklen])
      }
    }()
  • check interface implementation during compilation
    var _ io.Reader = (*MyFastReader)(nil)
  • if a param of len is nil then it's zero
  • anonymous structs are cool
    var hits struct {
      sync.Mutex
      n int
    }
    hits.Lock()
    hits.n++
    hits.Unlock()
  • httputil.DumpRequest is very useful thing, don't create your own
  • to get call stack we've runtime.Caller https://golang.org/pkg/runtime/#Caller
  • to marshal arbitrary JSON you can marshal to map[string]interface{}{}

About

List of advices and tricks in the Go's world \ʕ◔ϖ◔ʔ/

License:BSD 3-Clause "New" or "Revised" License


Languages

Language:Go 100.0%