欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

golang容易導(dǎo)致內(nèi)存泄漏的6種情況匯總

 更新時間:2023年01月09日 09:44:01   作者:Yuan_sr  
內(nèi)存泄漏是我們在生產(chǎn)環(huán)境中必須面臨的問題,下面這篇文章主要給大家介紹了關(guān)于golang容易導(dǎo)致內(nèi)存泄漏的6種情況,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下

1. 定時器使用不當(dāng)

1.1 time.After()的使用

默認(rèn)的time.After()是會有內(nèi)存泄露問題的,因?yàn)槊看蝨ime.After(duration x)會產(chǎn)生NewTimer(),在duration x到期之前,新創(chuàng)建的timer不會被GC,到期之后才會GC。

隨著時間推移,尤其是duration x很大的話,會產(chǎn)生內(nèi)存泄露的問題,應(yīng)特別注意

for true {
	select {
	case <-time.After(time.Minute * 3):
    // do something
  default:
		time.Sleep(time.Duration(1) * time.Second)
	}
}

為了保險起見,使用NewTimer()或者NewTicker()代替的方式主動釋放資源,兩者的區(qū)別請自行查閱或看我往期文章https://blog.csdn.net/weixin_38299404/article/details/119352884

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)
	}
}

1.2 time.NewTicker資源未及時釋放

在使用time.NewTicker時需要手動調(diào)用Stop()方法釋放資源,否則將會造成永久性的內(nèi)存泄漏

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)
	}
}

2. select阻塞

使用select時如果有case沒有覆蓋完全的情況且沒有default分支進(jìn)行處理,最終會導(dǎo)致內(nèi)存泄漏

2.1 導(dǎo)致goroutine阻塞的情況

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)
    }
}

上述這種情況會阻塞在ch3的消費(fèi)處導(dǎo)致內(nèi)存泄漏

2.2 循環(huán)空轉(zhuǎn)導(dǎo)致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)
}

上述for循環(huán)條件一旦命中default則會出現(xiàn)循環(huán)空轉(zhuǎn)的情況,并最終導(dǎo)致CPU暴漲

3. channel阻塞

channel阻塞主要分為寫阻塞和讀阻塞兩種情況

空channel

func channelTest() {
  	//聲明未初始化的channel讀寫都會阻塞
    var c chan int
  	//向channel中寫數(shù)據(jù)
    go func() {
        c <- 1
        fmt.Println("g1 send succeed")
        time.Sleep(1 * time.Second)
    }()
  	//從channel中讀數(shù)據(jù)
    go func() {
        <-c
        fmt.Println("g2 receive succeed")
        time.Sleep(1 * time.Second)
    }()
    time.Sleep(10 * time.Second)
}

寫阻塞

無緩沖channel的阻塞通常是寫操作因?yàn)闆]有讀而阻塞

func channelTest() {
    var c = make(chan int)
  	//10個協(xié)程向channel中寫數(shù)據(jù)
    for i := 0; i < 10; i++ {
        go func() {
            <- c
            fmt.Println("g1 receive succeed")
            time.Sleep(1 * time.Second)
        }()
    }
  	//1個協(xié)程叢channel讀數(shù)據(jù)
    go func() {
        c <- 1
        fmt.Println("g2 send succeed")
        time.Sleep(1 * time.Second)
    }()
  	//會有寫的9個協(xié)程阻塞得不到釋放
    time.Sleep(10 * time.Second)
}

有緩沖的channel因?yàn)榫彌_區(qū)滿了,寫操作阻塞

func channelTest() {
    var c = make(chan int, 8)
  	//10個協(xié)程向channel中寫數(shù)據(jù)
    for i := 0; i < 10; i++ {
        go func() {
            <- c
            fmt.Println("g1 receive succeed")
            time.Sleep(1 * time.Second)
        }()
    }
  	//1個協(xié)程叢channel讀數(shù)據(jù)
    go func() {
        c <- 1
        fmt.Println("g2 send succeed")
        time.Sleep(1 * time.Second)
    }()
  	//會有寫的幾個協(xié)程阻塞寫不進(jìn)去
    time.Sleep(10 * time.Second)
}

讀阻塞

期待從channel讀數(shù)據(jù),結(jié)果沒有g(shù)oroutine往進(jìn)寫數(shù)據(jù)

func channelTest() {
   var c = make(chan int)
  //1個協(xié)程向channel中寫數(shù)據(jù)
  go func() {
    <- c
    fmt.Println("g1 receive succeed")
    time.Sleep(1 * time.Second)
  }()
  //10個協(xié)程叢channel讀數(shù)據(jù)
  for i := 0; i < 10; i++ {
    go func() {
        c <- 1
        fmt.Println("g2 send succeed")
        time.Sleep(1 * time.Second)
    }()
  }
  //會有讀的9個協(xié)程阻塞得不到釋放
  time.Sleep(10 * time.Second)
}

4. goroutine導(dǎo)致的內(nèi)存泄漏

4.1 申請過多的goroutine

例如在for循環(huán)中申請過多的goroutine來不及釋放導(dǎo)致內(nèi)存泄漏

4.2 goroutine阻塞

4.2.1 I/O問題

I/O連接未設(shè)置超時時間,導(dǎo)致goroutine一直在等待,代碼會一直阻塞。

4.2.2 互斥鎖未釋放

goroutine無法獲取到鎖資源,導(dǎo)致goroutine阻塞

//協(xié)程拿到鎖未釋放,其他協(xié)程獲取鎖會阻塞
func mutexTest() {
    mutex := sync.Mutex{}
    for i := 0; i < 10; i++ {
        go func() {
            mutex.Lock()
            fmt.Printf("%d goroutine get mutex", i)
      			//模擬實(shí)際開發(fā)中的操作耗時
            time.Sleep(100 * time.Millisecond)
        }()
    }
    time.Sleep(10 * time.Second)
}

4.2.3 死鎖

當(dāng)程序死鎖時其他goroutine也會阻塞

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")
    }()
  	//其余協(xié)程獲取鎖都會失敗
    go func() {
        m1.Lock()
        fmt.Println("g3 get m1")
    }()
    time.Sleep(10 * time.Second)
}

4.2.4 waitgroup使用不當(dāng)

waitgroup的Add、Done和wait數(shù)量不匹配會導(dǎo)致wait一直在等待

5. slice 引起的內(nèi)存泄漏

當(dāng)兩個slice 共享地址,其中一個為全局變量,另一個也無法被GC;

append slice 后一直使用,沒有進(jìn)行清理。

var a []int
 
func test(b []int) {
        a = b[:3]
        return
}

6. 數(shù)組的值傳遞

由于數(shù)組時Golang的基本數(shù)據(jù)類型,每個數(shù)組占用不通的內(nèi)存空間,生命周期互不干擾,很難出現(xiàn)內(nèi)存泄漏的情況,但是數(shù)組作為形參傳輸時,遵循的時值拷貝,如果函數(shù)被多個goroutine調(diào)用且數(shù)組過大時,則會導(dǎo)致內(nèi)存使用激增。

//統(tǒng)計nums中target出現(xiàn)的次數(shù)
func countTarget(nums [1000000]int, target int) int {
    num := 0
    for i := 0; i < len(nums) && nums[i] == target; i++ {
        num++
    }
    return num
}

因此對于大數(shù)組放在形參場景下通常使用切片或者指針進(jìn)行傳遞,避免短時間的內(nèi)存使用激增。

總結(jié)

到此這篇關(guān)于golang容易導(dǎo)致內(nèi)存泄漏的6種情況的文章就介紹到這了,更多相關(guān)golang內(nèi)存泄漏內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Go語言命令行參數(shù)及cobra使用方法

    Go語言命令行參數(shù)及cobra使用方法

    Cobra是關(guān)于golang的一個命令行解析庫,用它能夠快速創(chuàng)建功能強(qiáng)大的 cli應(yīng)用程序和命令行工具,本文主要介紹了Go語言命令行參數(shù)及cobra使用方法,感興趣的可以了解一下
    2024-01-01
  • Golang?gin跨域解決方案示例

    Golang?gin跨域解決方案示例

    這篇文章主要為大家介紹了Golang?gin跨域解決方案,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步早日升職加薪
    2022-04-04
  • 關(guān)于golang中死鎖的思考與學(xué)習(xí)

    關(guān)于golang中死鎖的思考與學(xué)習(xí)

    本文主要介紹了關(guān)于golang中死鎖的思考與學(xué)習(xí),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-03-03
  • 詳解golang中Context超時控制與原理

    詳解golang中Context超時控制與原理

    Context本身的含義是上下文,我們可以理解為它內(nèi)部攜帶了超時信息、退出信號,以及其他一些上下文相關(guān)的值,本文給大家詳細(xì)介紹了golang中Context超時控制與原理,文中有相關(guān)的代碼示例供大家參考,需要的朋友可以參考下
    2024-01-01
  • Go語言高效I/O并發(fā)處理雙緩沖和Exchanger模式實(shí)例探索

    Go語言高效I/O并發(fā)處理雙緩沖和Exchanger模式實(shí)例探索

    這篇文章主要介紹了Go語言高效I/O并發(fā)處理雙緩沖和Exchanger模式實(shí)例探索,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2024-01-01
  • 淺談golang 的高效編碼細(xì)節(jié)

    淺談golang 的高效編碼細(xì)節(jié)

    本文主要介紹了golang 的高效編碼細(xì)節(jié),文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-10-10
  • docker中部署golang項(xiàng)目的步驟詳解

    docker中部署golang項(xiàng)目的步驟詳解

    這篇文章主要給大家介紹了關(guān)于在docker中部署golang項(xiàng)目的步驟,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。
    2017-11-11
  • 一文搞懂Go語言標(biāo)準(zhǔn)庫strconv

    一文搞懂Go語言標(biāo)準(zhǔn)庫strconv

    strconv包實(shí)現(xiàn)了基本數(shù)據(jù)類型和其字符串表示的相互轉(zhuǎn)換,本文主要介紹Go語言標(biāo)準(zhǔn)庫strconv,想要學(xué)習(xí)strconv標(biāo)準(zhǔn)庫的可以了解一下
    2023-04-04
  • 深入理解Golang?make和new的區(qū)別及實(shí)現(xiàn)原理

    深入理解Golang?make和new的區(qū)別及實(shí)現(xiàn)原理

    在Go語言中,有兩個比較雷同的內(nèi)置函數(shù),分別是new和make方法,二者都可以用來分配內(nèi)存,那他們有什么區(qū)別呢?下面我們就從底層來分析一下二者的不同。感興趣的小伙伴們可以參考借鑒,希望對大家能有所幫助
    2022-10-10
  • Go語言擴(kuò)展原語之Semaphore的用法詳解

    Go語言擴(kuò)展原語之Semaphore的用法詳解

    Go語言的擴(kuò)展包中提供了帶權(quán)重的信號量?semaphore.Weighted,讓我們可以按照不同的權(quán)重管理資源的訪問,下面小編就來和大家聊聊它們的具體用法吧
    2023-07-07

最新評論