golang slice使用不慎導致的問題
原文連結 : http://www.bugclosed.com/post/16
背景
go語言中切片slice是方便且好用的強大資料結構,但是使用的時候需要注意,不然容易出問題,最近因為遇到了一個slice的使用問題,比較典型。 有一個功能需求,使用者需要獲取1-20的不重複隨機序列。
邏輯實現
由於是需要固定的1-20共20個不同數字,所以直接定義好了唯一序列如下:
var(
originalNumbers = []uint32{1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12,13,14,15,16, 17, 18, 19, 20}
)
因為每個使用者獲取的資料的序列都需要隨機打亂,實現的邏輯如下:
func shuffle(list []uint32) []uint32 {
n := len(list)
for i := n - 1; i > 0; i-- {
j := dist.Int63n(int64(i + 1))
list[i], list[j] = list[j], list[i]
}
return list
}
func getOriginalNumbers() []uint32{
return originalNumbers
}
func GetRandomNumbers(cardType int) []uint32 {
return shuffle(getOriginalNumbers())
}
問題暴露
通過仔細分析,從以上的邏輯其實是可以發現問題的,只是寫程式碼的時候疏忽導致沒有主要到潛在問題。執行的時候發現邏輯不正確,偶爾有使用者得到的序列是有重複的數字。
從原始資料的初始化來看,數字是1-20初始化到slice裡面的,絕對不會出現重複。仔細看了GetRandoNumbers和shuffle打亂邏輯是存在併發訪問問題的。
首先originalNumbers是一個slice,引數傳遞slice時僅僅是傳遞的切片的指標,並非複製一份切片。所以在併發的情況下,每個使用者的GetRandomNumbers都會獲取到同一個slice地址。而shuffle函式會對得到切片資料進行寫操作(資料打亂),當出現併發寫問題的時候,資料發生錯亂就不足為奇了。
問題解決
這個問題本質就是併發寫問題,只需要將資料分離即可解決問題。
func getOriginalNumbers() []uint32{
tmp := make([]uint32, len(originalNumbers))
copy(tmp, shortDeck)
return tmp
}
總結
這是一個很典型的slice誤用問題,slice是一個資料結構,他會指向底層真正的記憶體資料塊,可以認為slice傳遞的是記憶體的指標。
相關文章
- golang中切片slice的引用問題Golang
- 建立index 使用Online導致的問題Index
- 未使用 `deleteLater` 而直接使用 `delete` 導致問題delete
- ANALYZE導致的阻塞問題分析
- MySQL Flush導致的等待問題MySql
- Golang Slice技巧Golang
- 使用impdp不當導致的資料丟失問題
- C++中泛型使用導致的膨脹問題C++泛型
- 克隆ORACLE軟體的導致的問題Oracle
- 【epoll問題】EPOLLRDHUP使用導致無法接受資料
- 使用資料庫處理併發可能導致的問題資料庫
- golang slice效能分析Golang
- CAS導致的ABA問題及解決
- MySQL8.0 view導致的效能問題MySqlView
- 分散式鎖導致的超賣問題分散式
- Golang 陣列和切片 Slice 和 Map 使用Golang陣列
- react-router4:解決使用browserRouter模式導致的404問題React模式
- Elasticsearch 使用不同分詞器導致搜尋排名的問題Elasticsearch分詞
- 字元校驗集問題導致索引無法正常使用字元索引
- 關於 Laravel mix 導致 Bootstrap 失效的問題Laravelboot
- ORACLE資料檔名導致的奇怪問題Oracle
- SCHEDULER呼叫XDB程式導致效能問題
- 執行計劃的偏差導致的效能問題
- 【RAC】處理因ons導致CPU使用率過高的問題
- golang使用sqlx報錯:unsupported type []interface {}, a slice of interfaceGolangSQL
- vue的scope導致樣式修改不了問題Vue
- EfCore3的OwnedType會導致Sql效率問題SQL
- javascript小數乘法運算導致的精度問題JavaScript
- 完美的執行計劃導致的效能問題
- Oracle資料庫導致效能問題的可能原因Oracle資料庫
- 一次oracle行級鎖導致的問題Oracle
- Oracle監聽日誌過大導致的問題Oracle
- Fastclick 導致click事件觸發兩次的問題AST事件
- memlock過低導致的資料庫效能問題資料庫
- oracle 序列值導致的主鍵衝突問題Oracle
- 【爬坑】.Net編譯環境導致的問題編譯
- Spring中非同步註解@Async的使用、原理及使用時可能導致的問題Spring非同步
- redis AOF落地策略rewrite導致阻塞問題Redis