リークの理由は: 1. time.After() の使用。time.After(duration x) は NewTimer() を生成します。duration x が期限切れになる前に、新しく作成されたタイマーは生成されません。 GC 、GC は期限切れ後にのみ発生します; 2. time.NewTicker リソースが時間内に解放されません; 3. 選択のブロック; 4. チャネルのブロック; 5. 多すぎるゴルーチンの適用、ゴルーチンのブロック; 6. スライスなどが原因です。
このチュートリアルの動作環境: Windows 7 システム、GO バージョン 1.18、Dell G3 コンピューター。
デフォルトの time.After() では、毎回 time.After(duration x) が NewTimer を生成するため、メモリ リークの問題が発生します。 ( )、新しく作成されたタイマーは、期間 x が期限切れになる前に GC されず、期限切れ後にのみ GC されます。
時間が経つにつれて、特に期間が長くなると、このメソッドはリソースを積極的に解放します。2 つの違いは自分で確認するか、以前の記事 https://blog.csdn.net/weixin_38299404/article/details を読んでください。 /119352884
for true { select { case <-time.After(time.Minute * 3): // do something default: time.Sleep(time.Duration(1) * time.Second) } }
timer := time.NewTicker(time.Duration(2) * time.Second) defer timer.Stop() for true { select { case <-timer.C: // do something default: time.Sleep(time.Duration(1) * time.Second) } }
timer := time.NewTicker(time.Duration(2) * time.Second) // defer timer.Stop() for true { select { case <-timer.C: // do something default: time.Sleep(time.Duration(1) * time.Second) } }上記の状況により ch3 の消費がブロックされ、メモリ リークが発生します
func main() { ch1 := make(chan int) ch2 := make(chan int) ch3 := make(chan int) go Getdata("https://www.baidu.com",ch1) go Getdata("https://www.baidu.com",ch2) go Getdata("https://www.baidu.com",ch3) select{ case v:=<- ch1: fmt.Println(v) case v:=<- ch2: fmt.Println(v) } }上記の for ループ条件がデフォルトに達すると、ループ アイドリングが発生し、最終的に CPU サージにつながります
func main() { fmt.Println("main start") msgList := make(chan int, 100) go func() { for { select { case <-msgList: default: } } }() c := make(chan os.Signal, 1) signal.Notify(c, os.Interrupt, os.Kill) s := <-c fmt.Println("main exit.get signal:", s) }書き込みブロッキング
バッファリングされていないチャネルのブロッキングは、通常、読み取りがないため、書き込み操作がブロックされます。
func channelTest() { //声明未初始化的channel读写都会阻塞 var c chan int //向channel中写数据 go func() { c <- 1 fmt.Println("g1 send succeed") time.Sleep(1 * time.Second) }() //从channel中读数据 go func() { <-c fmt.Println("g2 receive succeed") time.Sleep(1 * time.Second) }() time.Sleep(10 * time.Second) }
バッファがいっぱいであるため、バッファされたチャネルがブロックされます。、書き込み操作がブロックされました
func channelTest() { var c = make(chan int) //10个协程向channel中写数据 for i := 0; i < 10; i++ { go func() { <- c fmt.Println("g1 receive succeed") time.Sleep(1 * time.Second) }() } //1个协程丛channel读数据 go func() { c <- 1 fmt.Println("g2 send succeed") time.Sleep(1 * time.Second) }() //会有写的9个协程阻塞得不到释放 time.Sleep(10 * time.Second) }
チャネルからのデータの読み取りを待機していますが、データを書き込むゴルーチンがありませんでした
func channelTest() { var c = make(chan int, 8) //10个协程向channel中写数据 for i := 0; i < 10; i++ { go func() { <- c fmt.Println("g1 receive succeed") time.Sleep(1 * time.Second) }() } //1个协程丛channel读数据 go func() { c <- 1 fmt.Println("g2 send succeed") time.Sleep(1 * time.Second) }() //会有写的几个协程阻塞写不进去 time.Sleep(10 * time.Second) }
func channelTest() { var c = make(chan int) //1个协程向channel中写数据 go func() { <- c fmt.Println("g1 receive succeed") time.Sleep(1 * time.Second) }() //10个协程丛channel读数据 for i := 0; i < 10; i++ { go func() { c <- 1 fmt.Println("g2 send succeed") time.Sleep(1 * time.Second) }() } //会有读的9个协程阻塞得不到释放 time.Sleep(10 * time.Second) }
//协程拿到锁未释放,其他协程获取锁会阻塞 func mutexTest() { mutex := sync.Mutex{} for i := 0; i < 10; i++ { go func() { mutex.Lock() fmt.Printf("%d goroutine get mutex", i) //模拟实际开发中的操作耗时 time.Sleep(100 * time.Millisecond) }() } time.Sleep(10 * time.Second) }
func mutexTest() { m1, m2 := sync.Mutex{}, sync.RWMutex{} //g1得到锁1去获取锁2 go func() { m1.Lock() fmt.Println("g1 get m1") time.Sleep(1 * time.Second) m2.Lock() fmt.Println("g1 get m2") }() //g2得到锁2去获取锁1 go func() { m2.Lock() fmt.Println("g2 get m2") time.Sleep(1 * time.Second) m1.Lock() fmt.Println("g2 get m1") }() //其余协程获取锁都会失败 go func() { m1.Lock() fmt.Println("g3 get m1") }() time.Sleep(10 * time.Second) }
#6. 配列値の転送
var a []int func test(b []int) { a = b[:3] return }したがって、大きな配列が仮パラメータのシナリオに配置される場合、メモリ使用量の短期的な急増を避けるために、通常はスライスまたはポインタを使用して配列を渡します。 ##Go ビデオ チュートリアル、
]
以上がgolang のメモリリークの原因は何ですか?の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。