최대 고 루틴 수
고통없이 얼마나 많은 고 루틴을 사용할 수 있습니까? 예를 들어 위키피디아는 Erlang에서 2 천만개의 프로세스를 성능 저하없이 생성 할 수 있다고 말합니다.
업데이트 : 방금 고 루틴 성능을 조사한 결과 다음과 같은 결과가 나왔습니다.
- goroutine 수명이 sqrt ()를 1000 번 계산하는 것보다 더 많은 것 같습니다 (~ 45µs), 유일한 제한은 메모리입니다.
- 고 루틴 비용 4 — 4.5KB
고 루틴이 차단되면 다음을 제외하고는 관련 비용이 없습니다.
- 메모리 사용량
- 느린 쓰레기 수거
비용 (메모리 및 실제로 고 루틴 실행을 시작하는 데 걸리는 평균 시간)은 다음과 같습니다.
Go 1.6.2 (April 2016)
32-bit x86 CPU (A10-7850K 4GHz)
| Number of goroutines: 100000
| Per goroutine:
| Memory: 4536.84 bytes
| Time: 1.634248 µs
64-bit x86 CPU (A10-7850K 4GHz)
| Number of goroutines: 100000
| Per goroutine:
| Memory: 4707.92 bytes
| Time: 1.842097 µs
Go release.r60.3 (December 2011)
32-bit x86 CPU (1.6 GHz)
| Number of goroutines: 100000
| Per goroutine:
| Memory: 4243.45 bytes
| Time: 5.815950 µs
4GB의 메모리가 설치된 시스템에서는 최대 고 루틴 수를 1 백만 미만으로 제한합니다.
소스 코드 (위에 인쇄 된 숫자를 이미 이해하고있는 경우 읽을 필요 없음) :
package main
import (
"flag"
"fmt"
"os"
"runtime"
"time"
)
var n = flag.Int("n", 1e5, "Number of goroutines to create")
var ch = make(chan byte)
var counter = 0
func f() {
counter++
<-ch // Block this goroutine
}
func main() {
flag.Parse()
if *n <= 0 {
fmt.Fprintf(os.Stderr, "invalid number of goroutines")
os.Exit(1)
}
// Limit the number of spare OS threads to just 1
runtime.GOMAXPROCS(1)
// Make a copy of MemStats
var m0 runtime.MemStats
runtime.ReadMemStats(&m0)
t0 := time.Now().UnixNano()
for i := 0; i < *n; i++ {
go f()
}
runtime.Gosched()
t1 := time.Now().UnixNano()
runtime.GC()
// Make a copy of MemStats
var m1 runtime.MemStats
runtime.ReadMemStats(&m1)
if counter != *n {
fmt.Fprintf(os.Stderr, "failed to begin execution of all goroutines")
os.Exit(1)
}
fmt.Printf("Number of goroutines: %d\n", *n)
fmt.Printf("Per goroutine:\n")
fmt.Printf(" Memory: %.2f bytes\n", float64(m1.Sys-m0.Sys)/float64(*n))
fmt.Printf(" Time: %f µs\n", float64(t1-t0)/float64(*n)/1e3)
}
Hundreds of thousands, per Go FAQ: Why goroutines instead of threads?:
It is practical to create hundreds of thousands of goroutines in the same address space.
The test test/chan/goroutines.go creates 10,000 and could easily do more, but is designed to run quickly; you can change the number on your system to experiment. You can easily run millions, given enough memory, such as on a server.
To understand the max number of goroutines, note that the per-goroutine cost is primarily the stack. Per FAQ again:
…goroutines, can be very cheap: they have little overhead beyond the memory for the stack, which is just a few kilobytes.
A back-of-the-envelop calculation is to assume that each goroutine has one 4 KiB page allocated for the stack (4 KiB is a pretty uniform size), plus some small overhead for a control block (like a Thread Control Block) for the runtime; this agrees with what you observed (in 2011, pre-Go 1.0). Thus 100 Ki routines would take about 400 MiB of memory, and 1 Mi routines would take about 4 GiB of memory, which is still manageable on desktop, a bit much for a phone, and very manageable on a server. In practice the starting stack has ranged in size from half a page (2 KiB) to two pages (8 KiB), so this is approximately correct.
The starting stack size has changed over time; it started at 4 KiB (one page), then in 1.2 was increased to 8 KiB (2 pages), then in 1.4 was decreased to 2 KiB (half a page). These changes were due to segmented stacks causing performance problems when rapidly switching back and forth between segments ("hot stack split"), so increased to mitigate (1.2), then decreased when segmented stacks were replaced with contiguous stacks (1.4):
Go 1.2 Release Notes: Stack size:
In Go 1.2, the minimum size of the stack when a goroutine is created has been lifted from 4KB to 8KB
Go 1.4 Release Notes: Changes to the runtime:
the default starting size for a goroutine's stack in 1.4 has been reduced from 8192 bytes to 2048 bytes.
Per-goroutine memory is largely stack, and it starts low and grows so you can cheaply have many goroutines. You could use a smaller starting stack, but then it would have to grow sooner (gain space at cost of time), and the benefits decrease due to the control block not shrinking. It is possible to eliminate the stack, at least when swapped out (e.g., do all allocation on heap, or save stack to heap on context switch), though this hurts performance and adds complexity. This is possible (as in Erlang), and means you’d just need the control block and saved context, allowing another factor of 5×–10× in number of goroutines, limited now by control block size and on-heap size of goroutine-local variables. However, this isn’t terribly useful, unless you need millions of tiny sleeping goroutines.
Since the main use of having many goroutines is for IO-bound tasks (concretely to process blocking syscalls, notably network or file system IO), you’re much more likely to run into OS limits on other resources, namely network sockets or file handles: golang-nuts › The max number of goroutines and file descriptors?. The usual way to address this is with a pool of the scarce resource, or more simply by just limiting the number via a semaphore; see Conserving File Descriptors in Go and Limiting Concurrency in Go.
That depends entirely on the system you are running on. But goroutines are very lightweight. An average process should have no problems with 100.000 concurrent routines. Whether this goes for your target platform is, of course, something we can't answer without knowing what that platform is.
To paraphrase, there are lies, damn lies, and benchmarks. As the author of the Erlang benchmark confessed,
It goes without saying that there wasn't enough memory left in the machine to actually do anything useful. stress-testing erlang
What is your hardware, what is your operating system, where is your benchmark source code? What is the benchmark trying to measure and prove/disprove?
Here's a great article by Dave Cheney on this topic: http://dave.cheney.net/2013/06/02/why-is-a-goroutines-stack-infinite
If the number of goroutine ever become an issue, you easily can limit it for your program:
See mr51m0n/gorc and this example.
Set thresholds on number of running goroutines
Can increase and decrease a counter when starting or stopping a goroutine.
It can wait for a minimum or maximum number of goroutines running, thus allowing to set thresholds for the number ofgorc
governed goroutines running at the same time.
참고URL : https://stackoverflow.com/questions/8509152/max-number-of-goroutines
'development' 카테고리의 다른 글
동일한 PC에 여러 Firefox 버전 (0) | 2020.12.05 |
---|---|
Copy-Item은 디렉토리와 내용을 UNC 경로로 복사합니다. (0) | 2020.12.05 |
준비된 문에서 테이블 이름을 매개 변수화 할 수 있습니까? (0) | 2020.12.05 |
제네릭 유형 매개 변수에서`.class` 속성을 얻으려면 어떻게해야합니까? (0) | 2020.12.05 |
CLion으로 C 프로젝트를 만드는 방법 (0) | 2020.12.05 |