Prealloc هي أداة تحليل ثابت GO لإيجاد إعلانات شريحة يمكن أن تكون محددة.
go install github.com/alexkohler/prealloc@latest
على غرار أدوات التحليل الثابتة الأخرى (مثل Golint ، Go Vet) ، يمكن استدعاء Prealloc بأسماء أو أكثر من أسماء الملفات أو الدلائل أو الحزم التي يسمى مسار الاستيراد. Prealloc يدعم أيضا ... Wildcard.
prealloc [flags] files/directories/packages
على الرغم من أن GO يحاول تجنب إعادة التخصيص من خلال زراعة القدرة مقدمًا ، فإن هذا لا يكفي في بعض الأحيان لشرائح أطول. إذا كان حجم الشريحة معروفًا في وقت إنشائها ، فيجب تحديده.
النظر في المعيار التالي: (يمكن العثور على هذا في prealloc_test.go في هذا الريبو)
import "testing"
func BenchmarkNoPreallocate ( b * testing. B ) {
existing := make ([] int64 , 10 , 10 )
b . ResetTimer ()
for i := 0 ; i < b . N ; i ++ {
// Don't preallocate our initial slice
var init [] int64
for _ , element := range existing {
init = append ( init , element )
}
}
}
func BenchmarkPreallocate ( b * testing. B ) {
existing := make ([] int64 , 10 , 10 )
b . ResetTimer ()
for i := 0 ; i < b . N ; i ++ {
// Preallocate our initial slice
init := make ([] int64 , 0 , len ( existing ))
for _ , element := range existing {
init = append ( init , element )
}
}
}$ go test -bench=. -benchmem
goos: linux
goarch: amd64
BenchmarkNoPreallocate-4 3000000 510 ns/op 248 B/op 5 allocs/op
BenchmarkPreallocate-4 20000000 111 ns/op 80 B/op 1 allocs/op كما ترون ، لا يمكن أن يسبب التلاشي قبل الأداء نجاحًا كبيرًا ، ويرجع ذلك في المقام الأول إلى إلغاء إعادة تخصيص الصفيف الأساسي. يعد النمط المعياري أعلاه أمرًا شائعًا في GO: أعلن شريحة ، ثم اكتب نوعًا من النطاق أو للحلقة التي تُؤسس أو الفهارس فيها. والغرض من هذه الأداة هو إعلان إعلانات شريحة/حلقة مثل الإعلام في BenchmarkNoPreallocate .
بعض الأمثلة من GO 1.9.2 المصدر:
$ prealloc go/src/....
archive/tar/reader_test.go:854 Consider preallocating ss
archive/zip/zip_test.go:201 Consider preallocating all
cmd/api/goapi.go:301 Consider preallocating missing
cmd/api/goapi.go:476 Consider preallocating files
cmd/asm/internal/asm/endtoend_test.go:345 Consider preallocating extra
cmd/cgo/main.go:60 Consider preallocating ks
cmd/cgo/ast.go:149 Consider preallocating pieces
cmd/compile/internal/ssa/flagalloc.go:64 Consider preallocating oldSched
cmd/compile/internal/ssa/regalloc.go:719 Consider preallocating phis
cmd/compile/internal/ssa/regalloc.go:718 Consider preallocating oldSched
cmd/compile/internal/ssa/regalloc.go:1674 Consider preallocating oldSched
cmd/compile/internal/ssa/gen/rulegen.go:145 Consider preallocating ops
cmd/compile/internal/ssa/gen/rulegen.go:145 Consider preallocating ops
cmd/dist/build.go:893 Consider preallocating all
cmd/dist/build.go:1246 Consider preallocating plats
cmd/dist/build.go:1264 Consider preallocating results
cmd/dist/buildgo.go:59 Consider preallocating list
cmd/doc/pkg.go:363 Consider preallocating names
cmd/fix/typecheck.go:219 Consider preallocating b
cmd/go/internal/base/path.go:34 Consider preallocating out
cmd/go/internal/get/get.go:175 Consider preallocating out
cmd/go/internal/load/pkg.go:1894 Consider preallocating dirent
cmd/go/internal/work/build.go:2402 Consider preallocating absOfiles
cmd/go/internal/work/build.go:2731 Consider preallocating absOfiles
cmd/internal/objfile/pe.go:48 Consider preallocating syms
cmd/internal/objfile/pe.go:38 Consider preallocating addrs
cmd/internal/objfile/goobj.go:43 Consider preallocating syms
cmd/internal/objfile/elf.go:35 Consider preallocating syms
cmd/link/internal/ld/lib.go:1070 Consider preallocating argv
cmd/vet/all/main.go:91 Consider preallocating pp
database/sql/sql.go:66 Consider preallocating list
debug/macho/file.go:506 Consider preallocating all
internal/trace/order.go:55 Consider preallocating batches
mime/quotedprintable/reader_test.go:191 Consider preallocating outcomes
net/dnsclient_unix_test.go:954 Consider preallocating confLines
net/interface_solaris.go:85 Consider preallocating ifat
net/interface_linux_test.go:91 Consider preallocating ifmat4
net/interface_linux_test.go:100 Consider preallocating ifmat6
net/internal/socktest/switch.go:34 Consider preallocating st
os/os_windows_test.go:766 Consider preallocating args
runtime/pprof/internal/profile/filter.go:77 Consider preallocating lines
runtime/pprof/internal/profile/profile.go:554 Consider preallocating names
text/template/parse/node.go:189 Consider preallocating decl // cmd/api/goapi.go:301
var missing [] string
for feature := range optionalSet {
missing = append ( missing , feature )
}
// cmd/fix/typecheck.go:219
var b []ast. Expr
for _ , x := range a {
b = append ( b , x )
}
// net/internal/socktest/switch.go:34
var st [] Stat
sw . smu . RLock ()
for _ , s := range sw . stats {
ns := * s
st = append ( st , ns )
}
sw . smu . RUnlock ()
// cmd/api/goapi.go:301
var missing [] string
for feature := range optionalSet {
missing = append ( missing , feature )
} حتى إذا كان الحجم الذي يتم شريحة الشريحة صغيراً ، فلا يزال هناك زيادة في الأداء في تحديد السعة بشكل صريح بدلاً من تركها append لاكتشاف أنها تحتاج إلى محول. بالطبع ، لا يلزم القيام بتوصيل Preallocation في كل مكان . مهمة هذه الأداة هي فقط للمساعدة في اقتراح الأماكن التي ينبغي للمرء أن يفكر فيها في التلاشي.
أثناء إعلان شريحة الشريحة الخاصة بك ، بدلاً من استخدام القيمة الصفرية للشريحة مع var ، قم بتهيئتها باستخدام وظيفة GO المدمجة في make النوع والطول المناسبين. سيكون هذا الطول عمومًا كل ما تقوم به. سيبدو إصلاح الأمثلة من الأعلى هكذا:
// cmd/api/goapi.go:301
missing := make ([] string , 0 , len ( optionalSet ))
for feature := range optionalSet {
missing = append ( missing , feature )
}
// cmd/fix/typecheck.go:219
b := make ([]ast. Expr , 0 , len ( a ))
for _ , x := range a {
b = append ( b , x )
}
// net/internal/socktest/switch.go:34
st := make ([] Stat , 0 , len ( sw . stats ))
sw . smu . RLock ()
for _ , s := range sw . stats {
ns := * s
st = append ( st , ns )
}
sw . smu . RUnlock ()
// cmd/api/goapi.go:301
missing := make ([] string , 0 , len ( optionalSet ))
for feature := range optionalSet {
missing = append ( missing , feature )
} ملاحظة: إذا كان الأداء أمرًا بالغ الأهمية ، فقد يكون استخدام copy أكثر كفاءة بدلاً من append شرائح أكبر. كمرجع ، راجع المعيار التالي:
func BenchmarkSize200PreallocateCopy ( b * testing. B ) {
existing := make ([] int64 , 200 , 200 )
b . ResetTimer ()
for i := 0 ; i < b . N ; i ++ {
// Preallocate our initial slice
init := make ([] int64 , len ( existing ))
copy ( init , existing )
}
} $ go test -bench=. -benchmem
goos: linux
goarch: amd64
BenchmarkSize200NoPreallocate-4 500000 3080 ns/op 4088 B/op 9 allocs/op
BenchmarkSize200Preallocate-4 1000000 1163 ns/op 1792 B/op 1 allocs/op
BenchmarkSize200PreallocateCopy-4 2000000 807 ns/op 1792 B/op 1 allocs/op
سحب طلبات الترحيب!
إذا كنت قد استمتعت Prealloc ، فقم بإلقاء نظرة على أدوات التحليل الثابتة الأخرى!