<small>Discover Insane Possibility of Go Slice Usage from Runtime Source Code</small> === <!-- .slide: data-background-color="pink" --> <!-- .slide: data-transition="zoom" --> > [name=郭學聰 Hsueh-Tsung Kuo] > [time=Sat, 31 Jul 2021] [color=red] ###### CC BY-SA 4.0 --- <!-- .slide: data-transition="convex" --> ## Who am I? ![fieliapm](https://www.gravatar.com/avatar/2aef78f04240a6ac9ccd473ba1cbd1e3?size=2048 =384x384) <small>Someone (who?) said: a game programmer should be able to draw cute anime character(?)</small> ---- <!-- .slide: data-transition="convex" --> * A programmer from game company in Taiwan * Backend (and temporary frontend) engineer * Usually develop something related to my work in Python, Ruby, ECMAScript, Golang, C# * ECMAScript hater since **Netscape** is dead * Built CDN-aware game asset update system * Professional small vehicle driver * Draw cute anime character in spare time --- <!-- .slide: data-transition="convex" --> ## Outline ---- <!-- .slide: data-transition="convex" --> 4. What is Slice? * Array * Slice ---- <!-- .slide: data-transition="convex" --> 5. Source Code of Slice Implementation * Runtime * Slice Structure * Caution * Utility Functions * Where is append()? * Compile Time * Slice Operation * Accessing Slice Elements * Re-slicing * Machine Code Optimization ---- <!-- .slide: data-transition="convex" --> 6. Insane Slice Application * SliceTricks * Application in Algorithm * One More Thing 7. Conclusion 8. Resource 9. Q&A --- <!-- .slide: data-transition="convex" --> ## What is Slice? ---- <!-- .slide: data-transition="convex" --> ### Array ```go= var a [4]int a[0] = 1 i := a[0] // i == 1 // a[2] == 0, the zero value of the int type ``` ---- <!-- .slide: data-transition="convex" --> #### Array Structure ![array](https://blog.golang.org/slices-intro/slice-array.png) ---- <!-- .slide: data-transition="convex" --> #### Array Literal ```go= b := [2]string{"Penn", "Teller"} // make the compiler count the array elements b := [...]string{"Penn", "Teller"} ``` ---- <!-- .slide: data-transition="convex" --> ### Slice ```go= func make([]T, len, cap) []T var s []byte s = make([]byte, 5, 5) // s == []byte{0, 0, 0, 0, 0} ``` ---- <!-- .slide: data-transition="convex" --> ### Slice ```go= s := make([]byte, 5) len(s) == 5 cap(s) == 5 ``` ---- <!-- .slide: data-transition="convex" --> #### Slice Structure ![slice_struct](https://blog.golang.org/slices-intro/slice-struct.png) ![slice_struct](https://blog.golang.org/slices-intro/slice-1.png) ---- <!-- .slide: data-transition="convex" --> #### Slice Literal ```go= letters := []string{"a", "b", "c", "d"} ``` --- <!-- .slide: data-transition="convex" --> ## Source Code of Slice Implementation ---- <!-- .slide: data-transition="convex" --> ### Runtime <small>https://github.com/golang/go/blob/go1.16.5/src/runtime/slice.go</small> ---- <!-- .slide: data-transition="convex" --> #### Slice Structure ```go= type slice struct { array unsafe.Pointer len int cap int } ``` ---- <!-- .slide: data-transition="convex" --> ##### Caution * Underlying arrays of slices are mutable * Slices themselves are immutable * Call by value ---- <!-- .slide: data-transition="convex" --> #### Utility Functions ```go= func makeslice(et *_type, len, cap int) unsafe.Pointer func growslice(et *_type, old slice, cap int) slice func slicecopy(toPtr unsafe.Pointer, toLen int, fromPtr unsafe.Pointer, fromLen int, width uintptr) append() // where? ``` ---- <!-- .slide: data-transition="convex" --> #### Where is append()? <small>https://github.com/golang/go/blob/go1.16.5/src/cmd/compile/internal/gc/ssa.go#L2841</small> ---- <!-- .slide: data-transition="convex" --> ### Compile Time <small>https://github.com/golang/go/blob/go1.16.5/src/cmd/compile/internal/gc/ssa.go#L5458</small> ---- <!-- .slide: data-transition="convex" --> #### Slice Operation * Accessing slice elements * Re-slicing ---- <!-- .slide: data-transition="convex" --> ##### Accessing Slice Elements * slice * string * pointer ```go= val := s[2] s[3] = val ``` ---- <!-- .slide: data-transition="convex" --> ##### Re-slicing * slice * string * pointer ```go= // i : begin index (include) // j : end index (exclude) // k : index of capacity boundary (exclude) s := v[i:j:k] ``` ---- <!-- .slide: data-transition="convex" --> ###### Boundary Checking ```go= // k <= cap(v) // j <= k s := v[i:j:k] ``` * :heavy_check_mark: "len(s) > len(v)" <!-- .element: class="fragment" data-fragment-index="1" --> * if capacity of underlying array is enough <!-- .element: class="fragment" data-fragment-index="2" --> * :x: **string** doesn't have cap property <!-- .element: class="fragment" data-fragment-index="3" --> ---- <!-- .slide: data-transition="convex" --> ###### New Slice Caculation ```go= // Calculate the base pointer (rptr) for the new slice. // // Generate the following code assuming that indexes are in bounds. // The masking is to make sure that we don't generate a slice // that points to the next object in memory. We cannot just set // the pointer to nil because then we would create a nil slice or // string. // // rcap = k - i // rlen = j - i // rptr = ptr + (mask(rcap) & (i * stride)) // // Where mask(x) is 0 if x==0 and -1 if x>0 and stride is the width // of the element type. ``` ---- <!-- .slide: data-transition="convex" --> #### Machine Code Optimization * Re-slicing * Not atomic * Instruction might be interleaved in optimized assembly code * Discard boundary checking operation * If compiler detects that overflow will not happen <!-- .element: class="fragment" data-fragment-index="1" --> --- <!-- .slide: data-transition="convex" --> ## Insane Slice Application ---- <!-- .slide: data-transition="convex" --> ### SliceTricks * Official * <small>https://github.com/golang/go/wiki/SliceTricks</small> * Graphics by @ueokande * <small>https://ueokande.github.io/go-slice-tricks/</small> ---- <!-- .slide: data-transition="convex" --> #### AppendVector ![AppendVector1](https://raw.githubusercontent.com/ueokande/go-slice-tricks/master/src/components/CheatSheet/AppendVector1.svg) ![AppendVector2](https://raw.githubusercontent.com/ueokande/go-slice-tricks/master/src/components/CheatSheet/AppendVector2.svg) ```go= a = append(a, b...) ``` ---- <!-- .slide: data-transition="convex" --> #### Copy ![Copy](https://raw.githubusercontent.com/ueokande/go-slice-tricks/master/src/components/CheatSheet/Copy.svg) ```go= b = make([]T, len(a)) copy(b, a) ``` ```go= b = append([]T(nil), a...) ``` ```go= b = append(a[:0:0], a...) ``` ---- <!-- .slide: data-transition="convex" --> #### Cut ![Cut](https://raw.githubusercontent.com/ueokande/go-slice-tricks/master/src/components/CheatSheet/Cut.svg) ```go= a = append(a[:i], a[j:]...) ``` ---- <!-- .slide: data-transition="convex" --> #### Delete ![Delete1](https://raw.githubusercontent.com/ueokande/go-slice-tricks/master/src/components/CheatSheet/Delete1.svg) ```go= a = append(a[:i], a[i+1:]...) ``` ```go= a = a[:i+copy(a[i:], a[i+1:])] ``` ---- <!-- .slide: data-transition="convex" --> #### Delete without preserving order ![DeleteWithoutPreservingOrder](https://raw.githubusercontent.com/ueokande/go-slice-tricks/master/src/components/CheatSheet/DeleteWithoutPreservingOrder.svg) ```go= a[i] = a[len(a)-1] a = a[:len(a)-1] ``` ---- <!-- .slide: data-transition="convex" --> #### Cut (GC) ![CutGC](https://raw.githubusercontent.com/ueokande/go-slice-tricks/master/src/components/CheatSheet/CutGC.svg) ```go= copy(a[i:], a[j:]) for k, n := len(a)-j+i, len(a); k < n; k++ { a[k] = nil // or the zero value of T } a = a[:len(a)-j+i] ``` ---- <!-- .slide: data-transition="convex" --> #### Delete (GC) ![DeleteGC](https://raw.githubusercontent.com/ueokande/go-slice-tricks/master/src/components/CheatSheet/DeleteGC.svg) ```go= copy(a[i:], a[i+1:]) a[len(a)-1] = nil // or the zero value of T a = a[:len(a)-1] ``` ---- <!-- .slide: data-transition="convex" --> #### Delete without preserving order (GC) ![DeleteWithoutPreservingOrderGC](https://raw.githubusercontent.com/ueokande/go-slice-tricks/master/src/components/CheatSheet/DeleteWithoutPreservingOrderGC.svg) ```go= a[i] = a[len(a)-1] a[len(a)-1] = nil a = a[:len(a)-1] ``` ---- <!-- .slide: data-transition="convex" --> #### Expand ![Expand](https://raw.githubusercontent.com/ueokande/go-slice-tricks/master/src/components/CheatSheet/Expand.svg) ```go= a = append(a[:i], append(make([]T, j), a[i:]...)...) ``` ---- <!-- .slide: data-transition="convex" --> #### Extend ![Extend](https://raw.githubusercontent.com/ueokande/go-slice-tricks/master/src/components/CheatSheet/Extend.svg) ```go= a = append(a, make([]T, j)...) ``` ---- <!-- .slide: data-transition="convex" --> #### Filter (in place) ![Filter](https://raw.githubusercontent.com/ueokande/go-slice-tricks/master/src/components/CheatSheet/Filter.svg) ```go= n := 0 for _, x := range a { if keep(x) { a[n] = x n++ } } a = a[:n] ``` ---- <!-- .slide: data-transition="convex" --> #### Insert ![Insert](https://raw.githubusercontent.com/ueokande/go-slice-tricks/master/src/components/CheatSheet/Insert.svg) ```go= a = append(a[:i], append([]T{x}, a[i:]...)...) ``` ---- <!-- .slide: data-transition="convex" --> #### InsertVector ![InsertVector](https://raw.githubusercontent.com/ueokande/go-slice-tricks/master/src/components/CheatSheet/InsertVector.svg) ```go= a = append(a[:i], append(b, a[i:]...)...) ``` ---- <!-- .slide: data-transition="convex" --> #### Push ![Push](https://raw.githubusercontent.com/ueokande/go-slice-tricks/master/src/components/CheatSheet/Push.svg) ```go= a = append(a, x) ``` ---- <!-- .slide: data-transition="convex" --> #### Pop ![Extend](https://raw.githubusercontent.com/ueokande/go-slice-tricks/master/src/components/CheatSheet/Pop.svg) ```go= x, a = a[len(a)-1], a[:len(a)-1] ``` ---- <!-- .slide: data-transition="convex" --> #### Push Front/Unshift ![Unshift](https://raw.githubusercontent.com/ueokande/go-slice-tricks/master/src/components/CheatSheet/Unshift.svg) ```go= a = append([]T{x}, a...) ``` ---- <!-- .slide: data-transition="convex" --> #### Pop Front/Shift ![Shift](https://raw.githubusercontent.com/ueokande/go-slice-tricks/master/src/components/CheatSheet/Shift.svg) ```go= x, a = a[0], a[1:] ``` ---- <!-- .slide: data-transition="convex" --> ### Application in Algorithm ---- <!-- .slide: data-transition="convex" --> #### QuickSort ```go= func QuickSort(a []int) { if len(a) <= 1 { return } pivot := rand.Int() % len(a) left, right := 0, len(a)-1 a[pivot], a[right] = a[right], a[pivot] for i, _ := range a[:right] { if a[i] < a[right] { a[left], a[i] = a[i], a[left] left++ } } a[left], a[right] = a[right], a[left] QuickSort(a[:left]) QuickSort(a[left+1:]) } ``` ---- <!-- .slide: data-transition="convex" --> ### One More Thing ---- <!-- .slide: data-transition="convex" --> #### Slice Capacity :heavy_plus_sign: append() ```go= s := []int{1, 2, 3, 4, 5, 6, 7, 8} fmt.Printf("s == %v\n\n", s) s2 := s[1:4:5] fmt.Printf("s2 == %v\n\n", s2) s2 = append(s2, 15) fmt.Printf("s == %v\n", s) fmt.Printf("s2 == %v\n\n", s2) s3 := s2[:4:4] fmt.Printf("s3 == %v\n\n", s3) s2 = append(s2, 16) fmt.Printf("s == %v\n", s) fmt.Printf("s2 == %v\n\n", s2) ``` ---- <!-- .slide: data-transition="convex" --> #### Result ```go s == [1 2 3 4 5 6 7 8] s2 == [2 3 4] s == [1 2 3 4 15 6 7 8] s2 == [2 3 4 15] s3 == [2 3 4 15] s == [1 2 3 4 15 6 7 8] s2 == [2 3 4 15 16] ``` --- <!-- .slide: data-transition="convex" --> ## Conclusion ---- <!-- .slide: data-transition="convex" --> ### Slice is Invincible * Accessing slice elements * Re-slicing * append() ---- <!-- .slide: data-transition="convex" --> ### Bless :hash: {成為 slice 大師|<big>Become slice master</big>} > [name=郭學聰 Hsueh-Tsung Kuo] [time=2021_07_31] [color=red] :notebook: --- <!-- .slide: data-transition="convex" --> ## Resource ---- <!-- .slide: data-transition="convex" --> ### Reference * Slice introduction * <small>https://blog.golang.org/slices-intro</small> * Slice source code * <small>https://github.com/golang/go/blob/go1.16.5/src/runtime/slice.go</small> * <small>https://github.com/golang/go/blob/go1.16.5/src/cmd/compile/internal/gc/ssa.go</small> * SliceTricks * Official * <small>https://github.com/golang/go/wiki/SliceTricks</small> * Graphics by @ueokande * <small>https://ueokande.github.io/go-slice-tricks/</small> ---- <!-- .slide: data-transition="convex" --> ### Utility * Slides * Editor: HackMD * Presentation: Mozilla FireFox * Recording * OBS Studio * Panasonic DC-GH5 * LEICA DG SUMMILUX 15mm/F1.7 * Video Editing & Compositing * AviSynth official builds --- <!-- .slide: data-transition="zoom" --> ## Q&A --- <style> .reveal { background: #FFDFEF; color: black; } .reveal h2, .reveal h3, .reveal h4, .reveal h5, .reveal h6 { color: black; } .reveal code { font-size: 18px !important; line-height: 1.2; } .progress div{ height:14px !important; background: hotpink !important; } // original template .rightpart{ float:right; width:50%; } .leftpart{ margin-right: 50% !important; height:50%; } .reveal section img { background:none; border:none; box-shadow:none; } p.blo { font-size: 50px !important; background:#B6BDBB; border:1px solid silver; display:inline-block; padding:0.5em 0.75em; border-radius: 10px; box-shadow: 5px 5px 5px #666; } p.blo1 { background: #c7c2bb; } p.blo2 { background: #b8c0c8; } p.blo3 { background: #c7cedd; } p.bloT { font-size: 60px !important; background:#B6BDD3; border:1px solid silver; display:inline-block; padding:0.5em 0.75em; border-radius: 8px; box-shadow: 1px 2px 5px #333; } p.bloA { background: #B6BDE3; } p.bloB { background: #E3BDB3; } /*.slide-number{ margin-bottom:10px !important; width:100%; text-align:center; font-size:25px !important; background-color:transparent !important; }*/ iframe.myclass{ width:100px; height:100px; bottom:0; left:0; position:fixed; border:none; z-index:99999; } h1.raw { color: #fff; background-image: linear-gradient(90deg,#f35626,#feab3a); -webkit-background-clip: text; -webkit-text-fill-color: transparent; animation: hue 5s infinite linear; } @keyframes hue { from { filter: hue-rotate(0deg); } to { filter: hue-rotate(360deg); } } .progress{ height:14px !important; } .progress span{ height:14px !important; background: url("") repeat-x !important; } .progress span:after, .progress span.nyancat{ content: ""; background: url('') !important; width: 34px !important; height: 21px !important; border: none !important; float:right; margin-top:-7px; margin-right:-10px; } </style>
{"metaMigratedAt":"2023-06-16T03:50:13.586Z","metaMigratedFrom":"YAML","title":"Discover Insane Possibility of Go Slice Usage from Runtime Source Code","breaks":true,"description":"View the slide with \"Slide Mode\".","slideOptions":"{\"spotlight\":{\"enabled\":false},\"allottedMinutes\":25}","contributors":"[{\"id\":\"ea27dcd7-a3f2-47c2-b25e-6760e7936c38\",\"add\":57676,\"del\":36697}]"}
    524 views