首頁 > 後端開發 > Golang > 為什麼混合 `println` 和 `fmt.Println` 會影響 Go 中的堆疊成長?

為什麼混合 `println` 和 `fmt.Println` 會影響 Go 中的堆疊成長?

DDD
發布: 2024-11-17 13:58:01
原創
457 人瀏覽過

Why Does Mixing `println` and `fmt.Println` Impact Stack Growth in Go?

混合 Println 和 Fmt.Println:了解堆疊成長

在 Go 中,堆疊動態成長以容納變數和函數參數的儲存。但是,當使用不同的列印函數時,此行為可能會受到影響。

考慮以下範例:

package main

import "fmt"

const size = 1024

func main() {
    fmt.Println("Start")
    s := "HELLO"
    stackCopy(&s, 0, [size]int{})
}

func stackCopy(s *string, c int, a [size]int) {
    println("println: ", s, *s)
    //fmt.Println("fmt:     ", s, *s)
    c++
    if c == 10 {
        return
    }
    stackCopy(s, c, a)
}
登入後複製

僅使用println 時,字串s 的位址隨著堆疊的增長而變化:

Start
println:  0xc000107f58 HELLO
println:  0xc000117f58 HELLO
println:  0xc000117f58 HELLO
登入後複製

但是,當fmt .Println 與println 混合使用,或單獨使用時,s的位址保持不變:

Start
println:  0xc00010a040 HELLO
fmt:      0xc00010a040 HELLO
println:  0xc00010a040 HELLO
fmt:      0xc00010a040 HELLO
登入後複製

理解差異

差異源自於 println 和 fmt.Println 處理參數的方式。

println 是內建函數,不保留傳遞給它的任何參數。因此,參數不會逃逸到堆,而是在堆疊上分配。

fmt.Println 另一方面,來自標準函式庫,被視為自訂函數。編譯器不會假設 fmt.Println 不保留其參數,因此它們可能會逃逸到堆中。因此,fmt.Println 的參數分配在堆上而不是堆疊上。

由於 stackCopy 函數採用較大的參數 ([size]int),因此初始堆疊可能會變得不足,並且將需要一個更大的堆疊分配。如果在堆疊上分配的變數作為參數傳遞給可以逃逸到堆疊的函數,則隨著堆疊的增長,它們將被移動到堆疊。這就是為什麼我們在僅使用 println 時觀察到 s 位址的變化。

結論

混合 println 和 fmt.Println 會影響堆疊成長,因為他們處理論點的方式不同。 println 將參數保留在堆疊上,而 fmt.Println 可能會將它們分配到堆中,這可能會導致堆疊佈局和變數位址發生變化。

以上是為什麼混合 `println` 和 `fmt.Println` 會影響 Go 中的堆疊成長?的詳細內容。更多資訊請關注PHP中文網其他相關文章!

來源:php.cn
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
熱門教學
更多>
最新下載
更多>
網站特效
網站源碼
網站素材
前端模板