2 回答

TA貢獻1868條經驗 獲得超4個贊
Go 中沒有“深拷貝”的概念,一切都是按值傳遞的。
不,你的樣本甚至不是一個淺拷貝你傳遞了一個指針(字符串的地址):
如果你改變myTest.name了main函數,那么你在改變之后再次打印它,你會看到它會改變,請看這個證明示例代碼:
package main
import (
"fmt"
"sync"
"time"
)
type person struct {
name string
id int
phone int
}
var wg sync.WaitGroup
func main() {
myTest := person{"Alex", 22, 123}
fmt.Printf("%T : %[1]v %v\n", myTest.name, &myTest.name) // string : Alex 0xc042006740
wg.Add(1)
go func() {
fmt.Printf("%T : %[1]v %v\n", myTest.name, &myTest.name) // string : Alex 0xc042006740
time.Sleep(500 * time.Millisecond)
fmt.Printf("%T : %[1]v %v\n", myTest.name, &myTest.name) // string : J 0xc042006740
wg.Done()
}()
time.Sleep(100 * time.Millisecond)
myTest.name = "J"
wg.Wait()
}
首先person struct這樣定義:
type person struct {
name string
id int
phone int
}
第二次使用sync.WaitGroup等待 goroutine 完成。
關于你的主要問題,你可以像這樣自己測試:
package main
import (
"fmt"
"sync"
"time"
)
type person struct {
name string
id int
phone int
}
var wg sync.WaitGroup
func main() {
myTest := person{"Alex", 22, 123}
wg.Add(1)
go func() {
fmt.Printf("%T : %[1]v\n", myTest.name) // string : Alex
time.Sleep(500 * time.Millisecond)
fmt.Printf("%T : %[1]v\n", myTest.name) // string : J
wg.Done()
}()
time.Sleep(100 * time.Millisecond)
myTest.name = "J"
wg.Wait()
}
因此,正如您在此示例中看到的name,主函數中的字符串內容更改反映到 goroutine,因此它不是副本。
如果您需要這樣的復制調用:
package main
import (
"fmt"
"sync"
"time"
)
type person struct {
name string
id int
phone int
}
var wg sync.WaitGroup
func main() {
myTest := person{"Alex", 22, 123}
wg.Add(1)
go func(name string) {
fmt.Printf("%T : %[1]v\n", name) // string : Alex
time.Sleep(500 * time.Millisecond)
fmt.Printf("%T : %[1]v\n", name) // string : Alex
wg.Done()
}(myTest.name)
time.Sleep(100 * time.Millisecond)
myTest.name = "J"
wg.Wait()
}

TA貢獻1824條經驗 獲得超5個贊
Go 中沒有“深拷貝”的概念,一切都是按值傳遞的。
在您的實例中,這不是副本,您直接引用變量,如果您想要副本執行以下操作:
myTest := myTest
time.Sleep(10 * time.Second)
fmt.Println(myTest.name)
- 2 回答
- 0 關注
- 162 瀏覽
添加回答
舉報