go 指针接收器的一个疑问

2023-12-08 17:44:01 +08:00
 yujianwjj
import "testing"

type S struct {
	Name string
}

func (s *S) Write() {
	s.Name = "123"
}

func TestX(t *testing.T) {
	s := S{}

	// 这里能编译通过
	s.Write()

	sVals := map[int]S{1: {"A"}}

	// 这能编译通过:
	s2 := sVals[1]
	s2.Write()

	// 这里不能编译通过
	sVals[1].Write()

}

就是最后一个为什么不能编译通过。

1694 次点击
所在节点    Go 编程语言
14 条回复
vvhhaaattt
2023-12-08 17:50:33 +08:00
原因应该是不能取 map 的 value 的地址?
slice 或者数组应该可以正常调用
doraemonki
2023-12-08 17:52:29 +08:00
因为 map 返回的不是指针,s.Write()是语法糖,自动帮你取了地址
nagisaushio
2023-12-08 17:52:51 +08:00
map 取值永远会进行拷贝,就算是第一种方法,执行了 Write 后也不会改变 map 内的值
zuiwu
2023-12-08 17:53:28 +08:00
sVals 会自动给你转成 pointer 类型
chenchengbin
2023-12-08 17:55:19 +08:00
map 的 value 不能直接使用,要取出来
zuiwu
2023-12-08 17:57:47 +08:00
kidlj
2023-12-08 18:50:50 +08:00
The Go Programming Language 这本书有一节专门讲了指针接收器方法调用的几种情况,见图 1.

https://v2ex.com/i/h4s8n61X.png

对你这个例子而言,`sVals[1]` 不是一个变量,而是一个 temporary literal value ,无法获取它的地址,所以不能以它作为接收器调用指针接收器方法。不过经过测试(书里也提到一句),slice/array literal value 的 element value 可以调用对应类型的指针接收器方法,可能是一个考虑到某种常见场景下设置的特例。见图 2.

https://v2ex.com/i/RMH7n3o6.png
kidlj
2023-12-08 18:57:13 +08:00
补充个图:

kidlj
2023-12-08 18:59:59 +08:00
voidmnwzp
2023-12-08 19:08:11 +08:00
go 的 struct 某些时候可以理解为一种值类型
lance6716
2023-12-08 21:21:28 +08:00
你这个 map 的 value 类型是 S 不是*S ,而且是临时值无法自动转换
lysShub
2023-12-08 21:37:56 +08:00
s3 := sVals[1]
s3.Write()

这样就可以
lysShub
2023-12-08 21:39:59 +08:00
这种基本是不正确的写法,go 故意报错的,就像便利 map 故意随机一样
flyv2x
2023-12-09 13:32:23 +08:00
sVals := map[int]*S{1: {"A"}} 这里改下就能通过了

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://www.v2ex.com/t/998776

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX