不能随意互换。Go接口赋值取决于具体类型的方法集:值类型T仅含值接收者方法,T则包含值和指针接收者方法;若接口方法由指针接收者定义,则只有T实现该接口,T会编译报错。
不能随意互换。Go 接口变量存储的是 type 和 value 两部分,当具体类型实现接口时,只有该类型(或其指针)**实际实现了接口的所有方法**,才能被赋值给接口。如果一个类型 T 的指针方法集包含接口方法,而值方法集不包含,那么只有 *T 能赋值给该接口,T 会编译报错:cannot use。
t (type T) as type MyInterface in assignment: T does not implement MyInterface
T 只能调用值接收者方法;指针类型 *T 既能调用值接收者,也能调用指针接收者方法*T 实现了它,T 没有实现 —— 即使你传的是 &t,赋值目标也必须是 *T 类型表达式即使两者都能赋值(比如所有方法都是值接收者),传 T 还是 *T 仍会影响行为:
T:每次赋值都会拷贝整个结构体。如果 T 很大(例如含 slice、map 或大量字段),开销明显*T:只拷贝 8 字节指针,但后续接口内方法调用若为值接收者,会隐式解引用再拷贝 —— 不改变原值;若为指针接收者,则可修改原值func (t *MyStruct) Mutate() 在接口上调用时,仍会修改原始实例;而 func (t MyStruct) CopyMutate() 永远不会影响调用方的值type Counter struct{ n int }
func (c Counter) Inc() int { c.n++; return c.n } // 值接收者 → 不影响原值
func (c *Counter) IncPtr() int { c.n++; return c.n } // 指针接收者 → 影响原值
var c Counter
var i interface{ Inc() int } = c // OK,但调用 i.Inc() 不改变 c.n
var j interface{ IncPtr() int } = &c // OK,调用 j.IncPtr() 会改变 c.n
interface{} 是个例外吗不是例外,而是最宽松的特例。因为 interface{} 没有方法,所以任何类型(包括 T 和 *T)都天然满足它。但这不意味着可以忽略差异:
var x interface{} = myStruct 存的是 myStruct 的完整副本var x interface{} = &myStruct 存的是指向 myStruct 的指针,后续类型断言得到的是 *MyStruct
interface{} 做 reflect.ValueOf(x).Interface() 后再断言,可能因反射路径丢失原始类型信息导致 panicinterface{} 得到的是 map[string]interface{} 等值类型,不是原始结构体指针,无法反向修改源数据看两个事实:是否需要修改 receiver 自身状态,以及类型大小是否适合拷贝。
立即学习“go语言免费学习笔记(深入)”;
type Point struct{ X, Y int }),值接收者更高效且语义清晰(不可变意图明确)最易被忽略的一点:接口变量本身是轻量的,但背后承载的值或指针决定了内存布局和可变性——别只盯着接口声明,要顺藤摸到具体类型的接收者签名。