-
Notifications
You must be signed in to change notification settings - Fork 17.9k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
runtime: append is slower than make+assignments for small slices with known length #26734
Comments
please run with go test -bench=. -benchmem |
Note that it could be that append allocates and zeroes more memory than make+assign as append rounds up to the next best allocation bucket size. In addition s = append(s, ...) will copy the contents of the previous iteration in addition to the new work if s needs to be resized. Please also run benchmarks on a quiet system and with -count=20 as noise in the ns range when allocation are involved can be quiet high. |
With
With
benchstat:
* I had to make the test names the same. |
I modified the benchmarks a bit to make more similar: |
@martisch should the next step be comparing the assembly of the two implementations? (I don't think most people expect append to be that close to make+assign since make+assign seems optimal.) |
Yes looking at the assemblies could reveal if make+assign does an optimization (e.g. combining writes, avoiding some write barriers) that append does not. But it could also come down to runtime.growslice just having more overhead due to being a more general function and not being specialized for compile time known length of the appended slice (if that is the case maybe it could be detected and optimized away by the compiler). When increasing the size of the string array append/created and number of values copied by 8x append seems to become faster: Some thoughts:
|
As @martisch suggests, this might be related to write barriers. https://github.com/go101/go-benchmarks/tree/master/append-vs-make related issue: #23306 |
The gap is closing in now (go version devel +acbfbc634e Mon Nov 11 13:53:02 2019 +0530 linux/amd64):
This is with the benchmarks at #26734 (comment). |
What version of Go are you using (
go version
)?go1.10.3 linux/amd64
Does this issue reproduce with the latest release?
Yes.
What operating system and processor architecture are you using (
go env
)?GOARCH="amd64"
GOBIN=""
GOCACHE="/home/opennota/.cache/go-build"
GOEXE=""
GOHOSTARCH="amd64"
GOHOSTOS="linux"
GOOS="linux"
GOPATH="/home/opennota/gocode"
GORACE=""
GOROOT="/home/opennota/go"
GOTMPDIR=""
GOTOOLDIR="/home/opennota/go/pkg/tool/linux_amd64"
GCCGO="gccgo"
CC="gcc"
CXX="g++"
CGO_ENABLED="1"
CGO_CFLAGS="-g -O2"
CGO_CPPFLAGS=""
CGO_CXXFLAGS="-g -O2"
CGO_FFLAGS="-g -O2"
CGO_LDFLAGS="-g -O2"
PKG_CONFIG="pkg-config"
GOGCCFLAGS="-fPIC -m64 -pthread -fmessage-length=0 -fdebug-prefix-map=/tmp/go-build289966161=/tmp/go-build -gno-record-gcc-switches"
What did you do?
https://play.golang.org/p/e34FVX-sqQx
go test -bench=.
What did you expect to see?
The ns/op numbers in the two tests are close.
What did you see instead?
append is 24% slower:
The text was updated successfully, but these errors were encountered: