Go에 GMP 일정 모델이 있는 이유는 무엇인가요? 다음 글에서는 Go 언어에 GMP 스케줄링 모델이 있는 이유를 소개하겠습니다. 이것이 여러분에게 도움이 되기를 바랍니다.
GMP 스케줄링 모델은 Go의 핵심입니다. 이는 다중 스레드 동시 스케줄링 코루틴의 효율성 문제를 합리적으로 해결합니다.
GMP란 무엇인가
우선 각 GMP가 무엇을 가리키는지 이해해야 합니다.
- G: 고루틴(Goroutine)의 약어는 스레드에서 실행되는 코루틴을 의미합니다.
- M: Machine의 약어, 즉 thead, thread, 순환 스케줄링 코루틴 및 실행입니다.
- P: 프로세서의 약어로, 로컬 대기열에 코루틴을 저장하고 잠자지 않는 사용 가능한 코루틴을 스레드에 제공하는 프로세서를 나타냅니다.
스레드 M은 각각 프로세서 P를 보유합니다. 코루틴을 얻으려는 경우 P에서 먼저 얻으므로 GMP 모델 다이어그램은 다음과 같습니다.
일반적인 프로세스는 스레드 M이 P의 큐에서 코루틴을 얻는 것입니다. 코루틴을 얻을 수 없으면 잠금을 놓고 경쟁하게 됩니다. 전역 대기열에서 가져옵니다.
프로세서 P
코루틴 G와 스레드 M의 구조는 이전 글에서 설명했습니다. 프로세서 P에 대한 분석은 다음과 같습니다.
Function
프로세서 P는 코루틴 배치를 저장하므로 스레드 M은 전역 대기열의 코루틴을 놓고 다른 스레드와 경쟁할 필요 없이 잠금 없이 코루틴에서 코루틴을 얻을 수 있으므로 코루틴 예약의 효율성이 향상됩니다.
소스 코드 분석
p 구조 소스 코드는 srcruntimeruntime2.go
에 있으며, 여기에 몇 가지 중요한 필드가 표시됩니다. srcruntimeruntime2.go
中,这里展示部分重要字段。
type p struct { ... m muintptr // back-link to associated m (nil if idle) // Queue of runnable goroutines. Accessed without lock. runqhead uint32 runqtail uint32 runq [256]guintptr runnext guintptr ... }
-
m
为处理器p
所属的线程 -
runq
是一个储存协程的队列 -
runqhead
,runqtail
表示队列的头尾指针 -
runnext
指向下一个可运行的协程
线程M与处理器P是如何协作的?
在srcruntimeproc.go
中,有一个schedule
方法,这是线程运行的第一个函数。这函数中,线程需要获取到可运行的协程,代码如下:
func schedule() { ... // 寻找一个可运行的协程 gp, inheritTime, tryWakeP := findRunnable() ... }
func findRunnable() (gp *g, inheritTime, tryWakeP bool) { // 从本地队列中获取协程 if gp, inheritTime := runqget(pp); gp != nil { return gp, inheritTime, false } // 本地队列拿不到则从全局队列中获取协程 if sched.runqsize != 0 { lock(&sched.lock) gp := globrunqget(pp, 0) unlock(&sched.lock) if gp != nil { return gp, false, false } } }
从本地队列中获取协程
func runqget(pp *p) (gp *g, inheritTime bool) { next := pp.runnext // 队列中下一个可运行的协程 if next != 0 && pp.runnext.cas(next, 0) { return next.ptr(), true } ... }
那如果本地队列和全局队列中都没有协程了怎么办呢,难道就让线程这么闲着?
这时候处理器P就会任务窃取,从其他线程的本地队列中窃取一些任务,美其名曰分担其他线程的压力,还提高了自己线程的利用率。
源码在srcruntimeproc.gostealWork
中,感兴趣可以看看。
新建的协程该分配到哪?
新建的协程该分配到本地还是全局队列呢,得分情况:
- Go认为新协程的优先级高,于是先寻找本地队列放入,而且还插队。
- 本队队列满了才放入全局队列。
实际流程为:
- 随机寻找P
- 将新协程放入P的
runnext
中,意味着下一个就运行该协程,插队了 - 若P的协程满了,则放入全局队列
源码在srcruntimeproc.gonewproc
// Create a new g running fn. // Put it on the queue of g's waiting to run. // The compiler turns a go statement into a call to this. func newproc(fn *funcval) { gp := getg() pc := getcallerpc() systemstack(func() { newg := newproc1(fn, gp, pc) // 创建新协程 pp := getg().m.p.ptr() runqput(pp, newg, true) // 寻找本地队列放入 if mainStarted { wakep() } }) }
m
은 프로세서 p
가 속한 스레드입니다. runq
는 코루틴을 저장하는 대기열입니다.
runqhead, <code>runqtail
은 대기열의 헤드 및 테일 포인터를 나타냅니다.
runnext
는 실행 가능한 다음 코루틴을 가리킵니다.
스레드 남 프로세서 P에서는 어떻게 작동하나요? 🎜🎜🎜srcruntimeproc.go
에는 스레드가 실행하는 첫 번째 함수인 schedule
메서드가 있습니다. 이 함수에서 스레드는 실행 가능한 코루틴을 가져와야 합니다. 코드는 다음과 같습니다. 🎜rrreeerrreee🎜로컬 큐에서 코루틴 가져오기🎜rrreee🎜로컬 큐와 글로벌 큐에 코루틴이 없으면 어떻게 되나요? let 스레드가 너무 유휴 상태입니까? 🎜🎜이때 프로세서 P는 작업을 훔치고 다른 스레드의 로컬 대기열에서 일부 작업을 훔칩니다. 이를 다른 스레드의 압력을 공유하고 자체 스레드의 활용도를 향상시킨다고 합니다. 🎜🎜소스코드는 srcruntimeproc.gostealWork
에 있으니 관심있으신 분들은 한번 살펴보시면 됩니다. 🎜🎜새로 생성된 코루틴을 어디에 할당해야 하나요? 🎜🎜🎜새로 생성된 코루틴을 로컬 큐에 할당해야 할까요, 아니면 글로벌 큐에 할당해야 할까요? 점수: 🎜🎜🎜Go는 새 코루틴의 우선순위가 높다고 생각하여 이를 넣을 로컬 큐를 먼저 찾고 점프합니다. 대기열. 🎜🎜팀 대기열이 가득 찬 경우에만 글로벌 대기열에 추가됩니다. 🎜🎜🎜실제 프로세스는 다음과 같습니다. 🎜🎜P를 무작위로 찾습니다🎜🎜새 코루틴을 P의 runnext
에 넣습니다. 이는 코루틴이 다음에 실행되고 대기열이 점프된다는 의미입니다🎜 🎜P의 코루틴이 가득 차면 전역 대기열에 추가됩니다🎜
🎜소스 코드는 srcruntimeproc.gonewproc
함수에 있습니다. 🎜rrreee🎜🎜결론🎜🎜🎜이 문서에서는 처음에 GMP 스케줄링 모델을 소개하고, 프로세서 P와 스레드 M이 코루틴을 얻는 방법을 구체적으로 소개합니다. 🎜🎜프로세서 P는 코루틴을 얻기 위해 다중 스레드 상호 배제 문제를 해결하고 코루틴 예약의 효율성을 향상시킵니다. 그러나 코루틴이 로컬 큐에 있든 글로벌 큐에 있든 상관없이 순차적으로만 실행되는 것 같습니다. 코루틴을 비동기식으로 구현하면 동시 실행이 어떻게 되나요? 다음 기사에서 분석을 계속하겠습니다(아무도 읽지 않을지라도...). 🎜🎜추천 학습: 🎜Golang 튜토리얼🎜🎜
- 🎜P를 무작위로 찾습니다🎜🎜새 코루틴을 P의
runnext
에 넣습니다. 이는 코루틴이 다음에 실행되고 대기열이 점프된다는 의미입니다🎜 🎜P의 코루틴이 가득 차면 전역 대기열에 추가됩니다🎜srcruntimeproc.gonewproc
함수에 있습니다. 🎜rrreee🎜🎜결론🎜🎜🎜이 문서에서는 처음에 GMP 스케줄링 모델을 소개하고, 프로세서 P와 스레드 M이 코루틴을 얻는 방법을 구체적으로 소개합니다. 🎜🎜프로세서 P는 코루틴을 얻기 위해 다중 스레드 상호 배제 문제를 해결하고 코루틴 예약의 효율성을 향상시킵니다. 그러나 코루틴이 로컬 큐에 있든 글로벌 큐에 있든 상관없이 순차적으로만 실행되는 것 같습니다. 코루틴을 비동기식으로 구현하면 동시 실행이 어떻게 되나요? 다음 기사에서 분석을 계속하겠습니다(아무도 읽지 않을지라도...). 🎜🎜추천 학습: 🎜Golang 튜토리얼🎜🎜위 내용은 Go 언어에 GMP 스케줄링 모델이 존재하는 이유에 대한 심층 분석의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

goistrongchoiceforprojectsneedingsimplicity, performance, and concurrency, butitmaylackinadvancedfeaturesandecosystemmaturity.1) go'ssyntaxissimpleandeasytolearn, go'ssyntaxissimpleandeasytolearn, theadtofewerbugsandmoremaintainablecode, theitlacksfeaturecomecemememecememememecememememememememememecemememememecemememecemememecemememecemecemecode

go'sinitfunctionandjava'sstaticinitializersbothservetosetupenvironmentmentsbotheforethemainfunction, buttheydifferinexecutionandcontrol.go'sinitissimpleandautomatic, 적합한 적절한 소제 세트 비트 캔 리드 토프 렉스 오버러스

theinitfunctioningoare에 대한 thecommonusecases : 1) loadingConfigurationFiles는 eprogramStarts, 2) 초기화 GlobalVaribles, 및 3) runningpre-checksorvalidationsforeprogramProeceeds

channelsarecrucialingoforenablingsafeandeforcommunicationbetwengoroutines.theyfacilitatesynchronizationandmanagegoroutineLifeCycle, EssentialForConcurrentProgramming.ChannelSAldAllowSendingEndAvivingValues, ActAssignalsForsyNchronization 및 관리

GO에서 오류를 줄이고 오류를 통해 컨텍스트를 추가 할 수 있습니다. 1) 오류 패키지의 새로운 기능을 사용하면 오류 전파 중에 컨텍스트 정보를 추가 할 수 있습니다. 2) fmt.errorf 및 %w를 통해 오류를 포장하여 문제를 찾는 데 도움이됩니다. 3) 사용자 정의 오류 유형은 더 많은 의미 론적 오류를 만들고 오류 처리의 표현 능력을 향상시킬 수 있습니다.

GOOFFERSROBUSTFEATURESECURECODING, BUTDEVENCERMUSTIMPLESSECURITYBESTPRACTICEFICEATIVERY.1) USEGO'SCRYPTOPACKAGEFORUEDATAHANDLING.2) MANICECONCURNCY WHYSYNCHRONICIATIONSPRIMITIVESTOPRECONDITIONS.3) SANITIZEEXTERNALINPOPODOIDSQLINJ

go의 오류 인터페이스는 typeerRorInterface {error () string}로 정의되며, 오류로 간주 될 오류 () 메소드를 구현하는 모든 유형이 허용됩니다. 사용 단계는 다음과 같습니다. 1. 기본적으로 iferr! = nil {log.printf ( "anerroroccurred :%v", err) return}과 같은 오류를 확인하고 로그 오류합니다. 2. typeMyErrorStruct {msgstringDetailString}과 같은 자세한 정보를 제공하기 위해 사용자 정의 오류 유형을 만듭니다. 3. 오류 랩퍼를 사용하여 (GO1.13 이후) 원래 오류 메시지를 잃지 않고 컨텍스트를 추가하고,

Toeffectially HandleErrorsinconCurrentGoprograms, UsechannelStocommunicateErrors, ubstractErrorWatchers, ConsiderTimeOuts, UseBufferedChannels 및 ProvideClearerRormessages.1) UsechannelStopassErrorsfromgoroutinestothemainfunction


핫 AI 도구

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

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

Undress AI Tool
무료로 이미지를 벗다

Clothoff.io
AI 옷 제거제

Video Face Swap
완전히 무료인 AI 얼굴 교환 도구를 사용하여 모든 비디오의 얼굴을 쉽게 바꾸세요!

인기 기사

뜨거운 도구

안전한 시험 브라우저
안전한 시험 브라우저는 온라인 시험을 안전하게 치르기 위한 보안 브라우저 환경입니다. 이 소프트웨어는 모든 컴퓨터를 안전한 워크스테이션으로 바꿔줍니다. 이는 모든 유틸리티에 대한 액세스를 제어하고 학생들이 승인되지 않은 리소스를 사용하는 것을 방지합니다.

스튜디오 13.0.1 보내기
강력한 PHP 통합 개발 환경

mPDF
mPDF는 UTF-8로 인코딩된 HTML에서 PDF 파일을 생성할 수 있는 PHP 라이브러리입니다. 원저자인 Ian Back은 자신의 웹 사이트에서 "즉시" PDF 파일을 출력하고 다양한 언어를 처리하기 위해 mPDF를 작성했습니다. HTML2FPDF와 같은 원본 스크립트보다 유니코드 글꼴을 사용할 때 속도가 느리고 더 큰 파일을 생성하지만 CSS 스타일 등을 지원하고 많은 개선 사항이 있습니다. RTL(아랍어, 히브리어), CJK(중국어, 일본어, 한국어)를 포함한 거의 모든 언어를 지원합니다. 중첩된 블록 수준 요소(예: P, DIV)를 지원합니다.

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

Atom Editor Mac 버전 다운로드
가장 인기 있는 오픈 소스 편집기
