목차
golang이 메모리 누수로 쉽게 이어질 수 있는 여러 상황
1.1 time.After() 사용
1.2 time.NewTicker 리소스 시간 내에 해제되지 않습니다time.NewTicker를 사용할 때 리소스를 해제하려면 Stop() 메서드를 수동으로 호출해야 합니다. 그렇지 않으면 영구적인 메모리 누수가 발생합니다" >1.2 time.NewTicker 리소스 시간 내에 해제되지 않습니다time.NewTicker를 사용할 때 리소스를 해제하려면 Stop() 메서드를 수동으로 호출해야 합니다. 그렇지 않으면 영구적인 메모리 누수가 발생합니다
2. 선택 차단사용 시 select, 완전히 커버되지 않은 경우와 처리할 기본 브랜치가 없으면 결국 메모리 누수로 이어지게 됩니다" >2. 선택 차단사용 시 select, 완전히 커버되지 않은 경우와 처리할 기본 브랜치가 없으면 결국 메모리 누수로 이어지게 됩니다
3. 채널 차단은 주로 쓰기 차단과 읽기 차단의 두 가지 상황으로 나뉩니다.빈 채널" >3. 채널 차단은 주로 쓰기 차단과 읽기 차단의 두 가지 상황으로 나뉩니다.빈 채널
4.1 너무 많은 고루틴을 적용함
예를 들어 for 루프에서 너무 많은 고루틴이 제때 해제되지 않아 메모리 누수가 발생함
4.2.1 I/O 문제" >4.2.1 I/O 문제
I/O 연결이 시간 초과를 설정하지 않아 고루틴이 계속 대기하고 코드가 계속 차단됩니다.
waitgroup의 Add, Done 및 wait 수의 불일치로 인해 wait가 계속 대기하게 됩니다." >waitgroup의 Add, Done 및 wait 수의 불일치로 인해 wait가 계속 대기하게 됩니다.
두 개의 슬라이스 공유 주소 중 하나는 전역 변수이고 다른 하나는 GC일 수 없습니다. 슬라이스를 추가한 후 사용되었으며 정리되지 않았습니다. " >두 개의 슬라이스 공유 주소 중 하나는 전역 변수이고 다른 하나는 GC일 수 없습니다. 슬라이스를 추가한 후 사용되었으며 정리되지 않았습니다.
백엔드 개발 Golang golang 메모리 누수의 원인은 무엇입니까?

golang 메모리 누수의 원인은 무엇입니까?

Jan 10, 2023 pm 05:45 PM
golang 언어로 가다 메모리 누수

누출 이유는 다음과 같습니다. 1. time.After()를 사용하면 time.After(duration x)가 NewTimer()를 생성합니다. 기간 x가 만료되기 전에 새로 생성된 타이머는 만료 후 GC가 되지 않습니다. , GC, 2. time.NewTicker 리소스가 제 시간에 해제되지 않음, 3. 차단 선택, 5. 너무 많은 고루틴 적용, 고루틴 차단 등

golang 메모리 누수의 원인은 무엇입니까?

이 튜토리얼의 운영 환경: Windows 7 시스템, GO 버전 1.18, Dell G3 컴퓨터.

golang이 메모리 누수로 쉽게 이어질 수 있는 여러 상황

1. 타이머의 부적절한 사용

1.1 time.After() 사용

기본 time.After()에는 메모리가 있습니다. NewTimer()가 매번 생성되기 때문에 문제가 유출됩니다.After(duration

시간이 지날수록 특히 기간이 길어지면 차이를 직접 확인하시거나 이전 글을 읽어주세요 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)
	}
}
로그인 후 복사

1.2 time.NewTicker 리소스 시간 내에 해제되지 않습니다time.NewTicker를 사용할 때 리소스를 해제하려면 Stop() 메서드를 수동으로 호출해야 합니다. 그렇지 않으면 영구적인 메모리 누수가 발생합니다

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, 완전히 커버되지 않은 경우와 처리할 기본 브랜치가 없으면 결국 메모리 누수로 이어지게 됩니다

2.1 고루틴 차단을 유발합니다
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의 소비가 차단됩니다. 및 메모리 누수 유발

2.2 루프 유휴 상태로 인해 CPU 급증이 발생함
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가 급증하게 됩니다

3. 채널 차단은 주로 쓰기 차단과 읽기 차단의 두 가지 상황으로 나뉩니다.빈 채널

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)
}
로그인 후 복사

4. 고루틴으로 인한 메모리 누수

4.1 너무 많은 고루틴을 적용함

예를 들어 for 루프에서 너무 많은 고루틴이 제때 해제되지 않아 메모리 누수가 발생함

4.2 고루틴 차단

4.2.1 I/O 문제

I/O 연결이 시간 초과를 설정하지 않아 고루틴이 계속 대기하고 코드가 계속 차단됩니다.

4.2.2 뮤텍스 잠금이 해제되지 않았습니다
고루틴이 잠금 리소스를 얻을 수 없어 고루틴이 차단됩니다.
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)
}
로그인 후 복사

4.2.3 교착 상태
프로그램 교착 상태가 발생하면 다른 고루틴이
//协程拿到锁未释放,其他协程获取锁会阻塞
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)
}
로그인 후 복사

4.2.4 waitgroup의 부적절한 사용
waitgroup의 Add, Done 및 wait 수의 불일치로 인해 wait가 계속 대기하게 됩니다.

5.

두 개의 슬라이스 공유 주소 중 하나는 전역 변수이고 다른 하나는 GC일 수 없습니다. 슬라이스를 추가한 후 사용되었으며 정리되지 않았습니다.

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. 배열의 값 전송

배열은 Golang의 기본 데이터 유형이므로 각 배열은 서로 다른 메모리 공간을 차지하고 수명주기가 서로 간섭하지 않습니다. 누출이 있지만 배열로 형식 매개변수를 전송할 때 시간 값의 복사본을 따르십시오. 여러 고루틴에서 함수를 호출하고 배열이 너무 크면 메모리 사용량이 급증합니다.
var a []int
 
func test(b []int) {
        a = b[:3]
        return
}
로그인 후 복사
따라서 슬라이스나 포인터는 일반적으로 단기적인 메모리 사용량 급증을 피하기 위해 공식 매개변수 시나리오에서 대규모 배열을 전송하는 데 사용됩니다.

위 내용은 golang 메모리 누수의 원인은 무엇입니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

AI Hentai Generator

AI Hentai Generator

AI Hentai를 무료로 생성하십시오.

인기 기사

R.E.P.O. 에너지 결정과 그들이하는 일 (노란색 크리스탈)
4 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. 최고의 그래픽 설정
4 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. 아무도들을 수없는 경우 오디오를 수정하는 방법
4 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. 채팅 명령 및 사용 방법
4 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌

뜨거운 도구

메모장++7.3.1

메모장++7.3.1

사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전

SublimeText3 중국어 버전

중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

신 수준의 코드 편집 소프트웨어(SublimeText3)

GO에서 플로팅 포인트 번호 작업에 어떤 라이브러리가 사용됩니까? GO에서 플로팅 포인트 번호 작업에 어떤 라이브러리가 사용됩니까? Apr 02, 2025 pm 02:06 PM

Go Language의 부동 소수점 번호 작동에 사용되는 라이브러리는 정확도를 보장하는 방법을 소개합니다.

Go 's Crawler Colly의 큐 스레드의 문제는 무엇입니까? Go 's Crawler Colly의 큐 스레드의 문제는 무엇입니까? Apr 02, 2025 pm 02:09 PM

Go Crawler Colly의 대기열 스레딩 문제는 Colly Crawler 라이브러리를 GO 언어로 사용하는 문제를 탐구합니다. � ...

이동 중에 왜 println 및 string () 함수로 문자열이 다른 효과를 갖는 이유는 무엇입니까? 이동 중에 왜 println 및 string () 함수로 문자열이 다른 효과를 갖는 이유는 무엇입니까? Apr 02, 2025 pm 02:03 PM

Go Language의 문자열 인쇄의 차이 : println 및 String () 함수 사용 효과의 차이가 진행 중입니다 ...

GO의 어떤 라이브러리가 대기업에서 개발하거나 잘 알려진 오픈 소스 프로젝트에서 제공합니까? GO의 어떤 라이브러리가 대기업에서 개발하거나 잘 알려진 오픈 소스 프로젝트에서 제공합니까? Apr 02, 2025 pm 04:12 PM

GO의 어떤 라이브러리가 대기업이나 잘 알려진 오픈 소스 프로젝트에서 개발 했습니까? GO에 프로그래밍 할 때 개발자는 종종 몇 가지 일반적인 요구를 만납니다.

Redis Stream을 사용하여 GO Language에서 메시지 대기열을 구현할 때 User_ID 유형 변환 문제를 해결하는 방법은 무엇입니까? Redis Stream을 사용하여 GO Language에서 메시지 대기열을 구현할 때 User_ID 유형 변환 문제를 해결하는 방법은 무엇입니까? Apr 02, 2025 pm 04:54 PM

Go Language에서 메시지 대기열을 구현하기 위해 Redisstream을 사용하는 문제는 Go Language와 Redis를 사용하는 것입니다 ...

골란드의 사용자 정의 구조 레이블이 표시되지 않으면 어떻게해야합니까? 골란드의 사용자 정의 구조 레이블이 표시되지 않으면 어떻게해야합니까? Apr 02, 2025 pm 05:09 PM

골란드의 사용자 정의 구조 레이블이 표시되지 않으면 어떻게해야합니까? Go Language 개발을 위해 Goland를 사용할 때 많은 개발자가 사용자 정의 구조 태그를 만날 것입니다 ...

GO 언어에서 'var'와 'type` 키워드 정의 구조의 차이점은 무엇입니까? GO 언어에서 'var'와 'type` 키워드 정의 구조의 차이점은 무엇입니까? Apr 02, 2025 pm 12:57 PM

GO 언어에서 구조를 정의하는 두 가지 방법 : VAR과 유형 키워드의 차이. 구조를 정의 할 때 Go Language는 종종 두 가지 다른 글쓰기 방법을 본다 : 첫째 ...

VSCODE에서 자동으로 삭제되는 Golang 제네릭 기능 유형 제약 조건을 해결하는 방법은 무엇입니까? VSCODE에서 자동으로 삭제되는 Golang 제네릭 기능 유형 제약 조건을 해결하는 방법은 무엇입니까? Apr 02, 2025 pm 02:15 PM

VSCODE 사용자의 GOLANG 제네릭 기능 유형 제약 조건을 자동으로 삭제하면 VSCODE를 사용하여 Golang 코드를 작성할 때 이상한 문제가 발생할 수 있습니다. 언제...

See all articles