<style>
.reveal, .reveal h1, .reveal h2, .reveal h3, .reveal h4, .reveal h5, .reveal h6 {
font-family:Arial, Microsoft JhengHei;
}
.textleft {
text-align:left;
}
.reveal blockquote, .reveal pre, .reveal section img {
width: 100%
}
.reveal .progress {
height: 14px !important;
}
.progress span {
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>
# Go Slices Slide
###### tags: `Golang`, `簡報`
---
## Agenda
- 起因
- slice in GO
- 回到問題上(起因)
- 延伸閱讀 - slice 上的 "gotcha"
- 延伸閱讀 - slice 當參數傳遞
- 延伸閱讀 - Method receivers
- 延伸閱讀 - slice 相關豆知識
---
## 起因
----
### 看到了一篇 medium
[Go’s append is not always thread safe](https://medium.com/@cep21/gos-append-is-not-always-thread-safe-a3034db7975)
----
<div class="textleft">
這邊舉了兩個例子,在使用 goroutine 對同一個 slice append 會發生 race 的情況
p.s. 使用 `go test -race` 可以檢查是否會發生 race
</div>
----
### 範例一
```go=
package main
import (
"sync"
"testing"
)
func TestAppend(t *testing.T) {
x := []string{"start"}
wg := sync.WaitGroup{}
wg.Add(2)
go func() {
defer wg.Done()
y := append(x, "hello", "world")
t.Log(cap(y), len(y))
}()
go func() {
defer wg.Done()
z := append(x, "goodbye", "bob")
t.Log(cap(z), len(z))
}()
wg.Wait()
}
```
----
### 結果
```shell
> go test -race .
ok _/Users/kais.lin/go/goTest/slice 1.015s
> go test -v
z=> [start goodbye bob]
y=> [start hello world]
[start]
--- PASS: TestAppend (0.00s)
app_test.go:24: 3
app_test.go:18: 3
PASS
ok _/Users/kais.lin/go/goTest/slice 0.006s
```
----
### 範例二
```go=
package main
import (
"testing"
"sync"
)
func TestAppend(t *testing.T) {
x := make([]string, 0, 6)
wg := sync.WaitGroup{}
wg.Add(2)
go func() {
defer wg.Done()
y := append(x, "hello", "world")
t.Log(len(y))
}()
go func() {
defer wg.Done()
z := append(x, "goodbye", "bob")
t.Log(len(z))
}()
wg.Wait()
}
```
----
### 結果
```shell
> go test -race .
==================
WARNING: DATA RACE
Write at 0x00c4200be060 by goroutine 8:
_/tmp.TestAppend.func2()
/tmp/main_test.go:20 +0xcb
Previous write at 0x00c4200be060 by goroutine 7:
_/tmp.TestAppend.func1()
/tmp/main_test.go:15 +0xcb
Goroutine 8 (running) created at:
_/tmp.TestAppend()
/tmp/main_test.go:18 +0x14f
testing.tRunner()
/usr/local/Cellar/go/1.10.2/libexec/src/testing/testing.go:777 +0x16d
Goroutine 7 (running) created at:
_/tmp.TestAppend()
/tmp/main_test.go:13 +0x105
testing.tRunner()
/usr/local/Cellar/go/1.10.2/libexec/src/testing/testing.go:777 +0x16d
==================
==================
WARNING: DATA RACE
Write at 0x00c4200be070 by goroutine 8:
_/tmp.TestAppend.func2()
/tmp/main_test.go:20 +0x11a
Previous write at 0x00c4200be070 by goroutine 7:
_/tmp.TestAppend.func1()
/tmp/main_test.go:15 +0x11a
Goroutine 8 (running) created at:
_/tmp.TestAppend()
/tmp/main_test.go:18 +0x14f
testing.tRunner()
/usr/local/Cellar/go/1.10.2/libexec/src/testing/testing.go:777 +0x16d
Goroutine 7 (finished) created at:
_/tmp.TestAppend()
/tmp/main_test.go:13 +0x105
testing.tRunner()
/usr/local/Cellar/go/1.10.2/libexec/src/testing/testing.go:777 +0x16d
==================
--- FAIL: TestAppend (0.00s)
main_test.go:16: 2
main_test.go:21: 2
testing.go:730: race detected during execution of test
FAIL
FAIL _/tmp 0.901s
> go test -v
=== RUN TestAppend
z=> [hello world]
y=> [hello world]
[]
--- PASS: TestAppend (0.00s)
app_test.go:24: 2
app_test.go:18: 2
PASS
ok _/Users/kais.lin/go/goTest/slice 0.005s
```
----
SO WHY?
---
## slice in GO
----
### 要瞭解 slice 前必須先了解 array
> The slice type is an abstraction built on top of Go’s array type, and so to understand slices we must first understand arrays.
----
### Array in GO
<div class="textleft">
Array 宣告時會指定了長度和元素類型,且不需初始化,預設值為 0
</div>
```go
var a [4]int
a[0] = 1
i := a[0]
// i == 1
a[2] == 0
// true, the zero value of the int type
```
<div class="textleft">
在記憶體中會長這樣
</div>
![](https://blog.golang.org/go-slices-usage-and-internals_slice-array.png)
----
### 那 slice 呢?
<div class="textleft">
為了可以更彈性的利用 Array,所以 slice 長這樣:
</div>
```go
type sliceHeader struct {
Length int
Capacity int
ZerothElement *byte
}
```
<div class="textleft">
- Length: slice 長度<br>
- Capacity: 底層 array 的長度<br>
- ZerothElement: 參考(指向)的底層 array
</div>
----
所以當你從 slice 擷取部分值時:
s = s[2:4]
![](https://blog.golang.org/go-slices-usage-and-internals_slice-2.png)
[]byte 上到下分別為: ZerothElement, len, cap
----
所以對 `slice` `append` 值進去時,其實是進了底層 array,但如過底層 array 不夠大時怎麼辦?
----
```go
func AppendByte(slice []byte, data ...byte) []byte {
m := len(slice)
n := m + len(data)
if n > cap(slice) { // if necessary, reallocate
// allocate double what's needed, for future growth.
newSlice := make([]byte, (n+1)*2)
copy(newSlice, slice)
slice = newSlice
}
slice = slice[0:n]
copy(slice[m:n], data)
return slice
}
```
---
## 回到問題上(起因)
----
回到範例一
```go
...
x := []string{"start"}
...
go func() {
...
y := append(x, "hello", "world")
...
}
go func() {
...
z := append(x, "hello", "world")
...
}
```
----
<!-- .slide: data-background="#777" -->
![](https://cdn-images-1.medium.com/max/1600/1*MoOXXCuFEbYag5ZYIjgx5g.png)
----
<!-- .slide: data-background="#777" -->
![](https://cdn-images-1.medium.com/max/1600/1*VsEQuGIq9YE1__suevGPlg.png)
----
範例二
```go
...
x := make([]string, 0, 6)
...
go func() {
...
y := append(x, "hello", "world")
...
}
go func() {
...
z := append(x, "hello", "world")
...
}
```
----
<!-- .slide: data-background="#777" -->
![](https://cdn-images-1.medium.com/max/1600/1*oOJ9t3KPpbihDnMSF_pLfA.png)
----
<!-- .slide: data-background="#777" -->
![](https://cdn-images-1.medium.com/max/1600/1*jaT9M8OvuYFzyf29Qu8cbA.png)
----
### 作者提供的解決方法
```go
package main
import (
"sync"
"testing"
)
func TestAppend(t *testing.T) {
x := make([]string, 0, 6)
x = append(x, "start")
wg := sync.WaitGroup{}
wg.Add(2)
go func() {
defer wg.Done()
y := make([]string, 0, len(x)+2)
y = append(y, x...)
y = append(y, "hello", "world")
t.Log(cap(y), len(y), y[0])
}()
go func() {
defer wg.Done()
z := make([]string, 0, len(x)+2)
z = append(z, x...)
z = append(z, "goodbye", "bob")
t.Log(cap(z), len(z), z[0])
}()
wg.Wait()
}
```
---
## 延伸閱讀 - slice 上的 "gotcha"
----
### 還記得我們說過<br>slice 指向(參考)著底層的 array
----
<div class="textleft">
所以如果存了一大堆資料進 slice 而最後只為了取其中一小部分值的話,就會發生:
</div>
```go
var digitRegexp = regexp.MustCompile("[0-9]+")
func FindDigits(filename string) []byte {
b, _ := ioutil.ReadFile(filename) // b 為 slice
return digitRegexp.Find(b) // 回傳由 b(slice) 做正規畫後得到的資料
}
```
<div class="textleft">
會導致其他的 data 依然存在底層 array 中~
</div>
----
### 解決方法
```go
func CopyDigits(filename string) []byte {
b, _ := ioutil.ReadFile(filename)
b = digitRegexp.Find(b)
c := make([]byte, len(b))
copy(c, b)
return c
}
```
---
## 延伸閱讀 - slice 當參數傳遞
----
```go
func AddOneToEachElement(slice []int) {
for i := range slice {
slice[i]++
}
}
func main() {
buffer := []int {0,0,0,0,0,0,0,0,0,0,0,1,2,3,4,5,6,7,8,9,10}
slice := buffer[10:20]
fmt.Println("slice before ", slice)
// slice before [0 1 2 3 4 5 6 7 8 9]
fmt.Println("buffer before", buffer)
// buffer before [0 0 0 0 0 0 0 0 0 0 0 1 2 3 4 5 6 7 8 9 10]
AddOneToEachElement(slice)
fmt.Println("slice after ", slice)
// slice after [1 2 3 4 5 6 7 8 9 10]
fmt.Println("buffer after ", buffer)
// buffer after [0 0 0 0 0 0 0 0 0 0 1 2 3 4 5 6 7 8 9 10 10]
}
```
##### 儘管 slice 傳遞看起來像 passed by value,但因為 slice header 內含指標,所以會更新到原始參照的 array
---
## 延伸閱讀 - Method receivers
----
想將這功能改用 receivers 的方式實作
```go
func main() {
pathName := path("/usr/bin/tso") // Conversion from string to path.
i := bytes.LastIndex(pathName, []byte("/"))
if i >= 0 {
pathName = (pathName)[0:i]
}
fmt.Printf("%s\n", pathName)
// /usr/bin
}
```
----
> It is idiomatic to use a pointer receiver for a method that modifies a slice.
>
```go
type path []byte
func (p *path) TruncateAtFinalSlash() {
i := bytes.LastIndex(*p, []byte("/"))
if i >= 0 {
*p = (*p)[0:i]
}
}
func main() {
pathName := path("/usr/bin/tso") // Conversion from string to path.
pathName.TruncateAtFinalSlash()
fmt.Printf("%s\n", pathName)
// /usr/bin
}
```
----
但如果這邊改用 value receiver 的話
```go
type path []byte
func (p path) TruncateAtFinalSlash() {
i := bytes.LastIndex(p, []byte("/"))
if i >= 0 {
p = (p)[0:i]
}
fmt.Printf("%s\n", p)
// /usr/bin
}
func main() {
pathName := path("/usr/bin/tso") // Conversion from string to path.
pathName.TruncateAtFinalSlash()
fmt.Printf("%s\n", pathName)
// /usr/bin/tso
}
```
----
範例二: 要將 path 全轉大寫
```go
type path []byte
func (p path) ToUpper() {
for i, b := range p {
if 'a' <= b && b <= 'z' {
p[i] = b + 'A' - 'a'
}
}
}
func main() {
pathName := path("/usr/bin/tso")
pathName.ToUpper()
fmt.Printf("%s\n", pathName)
// /USR/BIN/TSO
}
```
----
```go
type path []byte
func (p *path) ToUpper() {
// for i, b := range *p {
for _, b := range *p {
if 'a' <= b && b <= 'z' {
// (*p)[i] = b + 'A' - 'a'
b = b + 'A' - 'a'
}
}
fmt.Printf("%s\n", *p)
// /usr/bin/tso
}
func main() {
pathName := path("/usr/bin/tso")
pathName.ToUpper()
fmt.Printf("%s\n", pathName)
// /usr/bin/tso
}
```
----
<div class="textleft">
※ go 在 range 時所遍歷的值是複製的值而不是元素本身,而如果使用`&遍歷值`這樣拿到的也是臨時變數的位置
而範例二的重點我覺得是 range 的小細節,而非將重點放在 slice 上
</div>
---
## 延伸閱讀 - slice 相關豆知識
----
### 空 slice = nil
當你 `make` 一個空 slice 時
```go
sliceHeader{
Length: 0,
Capacity: 0,
ZerothElement: nil,
}
```
----
```go
var a []int
fmt.Println(a == nil)
// true
b := make([]int,0)
fmt.Println(b == nil)
// flase
```
----
### String 也是 slice
> Strings are actually very simple: they are just read-only slices of bytes with a bit of extra syntactic support from the language.
----
所以你也可以直接做互相轉換
```go
str := string(slice)
// reverse
slice := []byte(usr)
```
----
<div class="textleft">
這種 slice-like 的設計,可以輕鬆的建立子字串,並且跟改後不會影響父子串
</div>
```go
a := "1234567890"
b := a[1:5]
fmt.Println(b)
// 12345
b = "54321"
fmt.Println(a)
// 1234567890
fmt.Println(b)
// 54321
```
{"metaMigratedAt":"2023-06-14T17:10:27.304Z","metaMigratedFrom":"Content","title":"Go Slices Slide","breaks":true,"contributors":"[{\"id\":\"69ade472-3ed3-499d-8a69-767243a31621\",\"add\":17682,\"del\":1384}]"}