Rumah > pembangunan bahagian belakang > Golang > Adakah pengkompil Go mengoptimumkan kod jauh yang menambah pembolehubah dalam goroutine?

Adakah pengkompil Go mengoptimumkan kod jauh yang menambah pembolehubah dalam goroutine?

DDD
Lepaskan: 2024-10-29 05:55:30
asal
955 orang telah melayarinya

 Does the Go compiler optimize away code that increments a variable in a goroutine?

Adakah pengkompil Go mengoptimumkan kod?

Dalam kod ini:

package main

import "time"

func main() {
    i := 1
    go func() {
        for {
            i++
        }
    }()
    <-time.After(1 * time.Second)
    println(i)
}
Salin selepas log masuk

Output sentiasa 1 . Walau bagaimanapun, adalah menghairankan bahawa 1s sudah cukup untuk gelung untuk berulang kali. Sebabnya adalah kerana pengkompil Go sedang mengoptimumkan kod.

Model memori Go menentukan keadaan di mana bacaan pembolehubah dalam satu goroutine boleh dijamin untuk memerhatikan nilai yang dihasilkan oleh menulis kepada pembolehubah yang sama dalam goroutine yang berbeza. Tugasan kepada i, melalui kenaikan i (i = i 1), tidak diikuti oleh sebarang peristiwa penyegerakan, jadi ia tidak dijamin untuk dipatuhi oleh mana-mana goroutine lain. Malah, pengkompil yang agresif mungkin memadamkan keseluruhan pernyataan i.

Sebagai contoh, dalam kod ini:

package main

import "time"

func main() {
    i := 1
    go func() {
        for {
            i++
        }
    }()
    <-time.After(1 * time.Millisecond)
    println(i)
}
Salin selepas log masuk

Outputnya ialah 1. Goroutine dikurangkan kepada:

"".main.func1 STEXT nosplit size=2 args=0x8 locals=0x0
    0x0000 00000 (elide.go:7)   TEXT    "".main.func1(SB), NOSPLIT, <pre class="brush:php;toolbar:false">for {
    i++
}
Salin selepas log masuk
-8 0x0000 00000 (elide.go:7) FUNCDATA
package main

import "time"

func main() {
    i := 1
    go func() {
        for {
            i++
            println("+1")
        }
    }()
    <-time.After(1 * time.Millisecond)
    println(i)
}
Salin selepas log masuk
, gclocals·2a5305abe05176240e61b8620e19a815(SB) 0x0000 00000 (elide.go:7) FUNCDATA , gclocals·33cdeccccebe80329f1fdbee7f5874cb(SB) 0x0000 00000 (elide.go:9) JMP 0

Kepada pengkompil, gelung for boleh dilaksanakan dengan menambah daftar selama-lamanya, pada asasnya gelung tanpa operasi:

+1
+1
<< SNIP >>
+1
+1
432
Salin selepas log masuk

Selepas memasukkan pernyataan cetakan,

"".main.func1 STEXT size=81 args=0x8 locals=0x18
    0x0000 00000 (elide.go:7)   TEXT    "".main.func1(SB), -8
    0x0000 00000 (elide.go:7)   MOVQ    (TLS), CX
    0x0009 00009 (elide.go:7)   CMPQ    SP, 16(CX)
    0x000d 00013 (elide.go:7)   JLS 74
    0x000f 00015 (elide.go:7)   SUBQ    , SP
    0x0013 00019 (elide.go:7)   MOVQ    BP, 16(SP)
    0x0018 00024 (elide.go:7)   LEAQ    16(SP), BP
    0x001d 00029 (elide.go:7)   FUNCDATA    <pre class="brush:php;toolbar:false">==================
WARNING: DATA RACE

Read at 0x00c420094000 by 
main goroutine:
  main.main()
      /home/peter/gopath/src/lucky.go:14 +0xac

Previous write at 0x00c420094000 by 
goroutine 5:
  main.main.func1()
      /home/peter/gopath/src/lucky.go:9 +0x4e

Goroutine 5 (running) created at:
  main.main()
      /home/peter/gopath/src/lucky.go:7 +0x7a
==================
Salin selepas log masuk
, gclocals·a36216b97439c93dafebe03e7f0808b5(SB) 0x001d 00029 (elide.go:7) FUNCDATA , gclocals·33cdeccccebe80329f1fdbee7f5874cb(SB) 0x001d 00029 (elide.go:8) MOVQ "".&i+32(SP), AX 0x0022 00034 (elide.go:9) INCQ (AX) 0x0025 00037 (elide.go:10) PCDATA
package main

import (
    "sync"
    "time"
)

func main() {
    mx := new(sync.Mutex)
    i := 1
    go func() {
        for {
            mx.Lock()
            i++
            mx.Unlock()
        }
    }()
    <-time.After(1 * time.Second)
    mx.Lock()
    println(i)
    mx.Unlock()
}
Salin selepas log masuk
,
41807838
Salin selepas log masuk
0x0025 00037 (elide.go:10) CALL runtime.printlock(SB) 0x002a 00042 (elide.go:10) LEAQ go.string."+1\n"(SB), AX 0x0031 00049 (elide.go:10) MOVQ AX, (SP) 0x0035 00053 (elide.go:10) MOVQ , 8(SP) 0x003e 00062 (elide.go:10) PCDATA , 0x003e 00062 (elide.go:10) CALL runtime.printstring(SB) 0x0043 00067 (elide.go:10) PCDATA , 0x0043 00067 (elide.go:10) CALL runtime.printunlock(SB) 0x0048 00072 (elide.go:9) JMP 29 0x004a 00074 (elide.go:9) NOP 0x004a 00074 (elide.go:7) PCDATA , $-1 0x004a 00074 (elide.go:7) CALL runtime.morestack_noctxt(SB) 0x004f 00079 (elide.go:7) JMP 0

Outputnya ialah:

Groutine mengembang kepada:

Kerumitan goroutine yang meningkat bermakna bahawa pengkompil tidak lagi mempertimbangkan untuk mendedikasikan daftar kepada nilai i. Nilai dalam ingatan i dinaikkan, yang menjadikan kemas kini kelihatan, dengan perlumbaan data, kepada goroutine utama.

Untuk mendapatkan hasil yang diharapkan, tambahkan beberapa penyegerakan:

Output:

Atas ialah kandungan terperinci Adakah pengkompil Go mengoptimumkan kod jauh yang menambah pembolehubah dalam goroutine?. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

Kenyataan Laman Web ini
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn
Tutorial Popular
Lagi>
Muat turun terkini
Lagi>
kesan web
Kod sumber laman web
Bahan laman web
Templat hujung hadapan