【Go 进阶】Go 语言到底是值传递,还是引用传递?(三)

简介: 【Go 进阶】Go 语言到底是值传递,还是引用传递?(三)

3、函数接收者中的 type

3.1 值接收者和指针接收者

比如我们有两个结构体:

type Man struct {}
type Woman struct {}

我们分别使用指针接收者和值接收者给它们添加一个Say()方法:

// Say()方法的全名为(*Man).Say(),即只有指针类型*Man才有Say()方法
func (*Man) Say() {
    fmt.Println("man say")
}
// Say()方法的全名为(Woman).Say(),即只有值类型Woman才有Say()方法
func (Woman) Say() {
    fmt.Println("woman say")
}

这里虽然它们都是 Say() 方法,但实际上方法名是不一样的,如果你使用指针接收者,方法的全名为(*Man).Say(), 如果是值类型,则全名为 Woman.Say()。严格的来说,对于前者,只能使用 (*Man) 类型来调用 Say() 方法,后者则是只能使用 Woman 类型来调用,因为值类型 Man 并没有 Say()方法,同理指针类型 *Woman 也没有 Say() 方法。

但实际编码时,使用 Man.Say() 也能通过编译,为什么呢?因为编译器做了一层隐式转换,比如:

man := Man{} // man是个值类型
man.Say() // man say   ok, 编译器将man隐式转换成了 &Man

这是因为 go 的编译器为了我们做了一次隐式转换,即将 man.Say() 转换成了 (&man).Say(),也就是对 man 做了取地址操作。同理,如果实参是值类型而形参(方法接收者)是指针类型:

ptrWoman := &Woman{}
ptrWoman.Say() // woman say   ok, 编译器将ptrWoman隐式转换成了*Woman

编译器也会为了通过编译而尽量把指针类型 ptrWomanWoman 类型上"套",这个"套"法就是对 ptrWoman 做隐式转换,转换成 (*ptrWoman).Say(),这样就跟方法名匹配上了。

那么既然编译器这么勤劳,为什么我们还需要关心这个问题呢?原因是对于接口类型,编译器"偷懒"了,并不会主动为我们做类型转换,比如我们定义一个 CanTalk 接口,里面就有这个 Say() 方法:

// 定义一个说话接口
type CanTalk interface {
  Say()
}

这样一来,WomanMan 类型应该都实现了这个接口,对吧?其实不然,因为 ManSay() 方法是指针接收者,所以严格来说是指针类型 *Man 实现了这个接口,而值类型 Man 并没有。同理,因为 WomanSay() 方法是值类型,所以严格来说是 Woman 实现了这个接口,而 *Woman 则没有。所以,如果你把值类型 Man 的变量赋值给接口 CanTalk 是会报错的:

man := Man{} 
var canTalk CanTalk
canTalk = man // error, Man类型没有Say()方法
// error:cannot use man (type Man) as type CanTalk in assignment: 
// Man does not implement CanTalk (Say method has pointer receiver)

而反过来,如果将 指针类型的 *man 或者 值类型/指针类型的 woman 的变量赋值给CanTalk则没有问题:

man := &Man{} 
var canTalk CanTalk
canTalk = man
canTalk.Say() // man say

在Go语言中,接口变量实际上是一个由两部分组成的数据结构:一个指向实际值的指针(或数据),和一个指向类型信息的指针。而该类型信息包括方法集合信息、包路径等内容。

在将man := Man{}赋值给canTalk变量时,由于Man类型只实现了(*Man) Say()方法,而没有实现(Man) Say()方法,因此不能被赋值给CanTalk类型的变量,因为它无法满足CanTalk接口的要求。因此,这段代码会在编译期间报错:cannot use man (type Man) as type CanTalk in assignment: Man does not implement CanTalk (missing Say method)

尽管*Man类型已经实现了Say方法,但是根据Go语言规范,只有带指针接收器的方法才可以被使用指针类型的变量调用。在这个例子中,虽然 manMan 的一个值类型实例,它并不具备Man类型的Say方法可以被指针类型 CanTalk 变量调用的能力。因此,将man直接赋值给实现了Say方法的 CanTalk 类型变量会导致编译错误。解决这个问题的方法是,将man的地址赋值给 CanTalk 类型变量,即使用取地址符"&"进行显式转换:

var canTalk CanTalk
canTalk = &man
canTalk.Say()

这样就可以正确地通过 CanTalk 类型变量调用Man类型的Say方法了。

再来看看 Woman

woman := Woman{} 
// or 
woman := &Woman{} 
var canTalk CanTalk
canTalk = woman
canTalk.Say() // woman say

woman := Woman{}woman := &Woman{} 都是创建一个 Woman 类型的变量,但它们的类型和使用方式有所不同。

woman := Woman{} 是直接创建了一个 Woman 类型的值,并将其赋给 woman 变量。这时 woman 变量的类型就是 Woman,而不是指向 Woman 类型的指针。因此,对 woman 变量进行方法调用时,需要使用值类型,即:woman.Say()。同时,如果将 woman 赋值给实现了该类型的接口变量(如 var canTalk CanTalk = woman),Go 会在编译阶段自动为 woman 值生成一个指向该值的指针并赋给接口变量。

这里需要注意的是,由于值类型无法通过地址访问到,因此只有实现了 func (t T) MethodName() {...} 方法声明的值类型才可以被赋值给对应接口类型的变量。例如:func (Woman) Say() {...}

woman := &Woman{} 则是创建了一个指向 Woman 类型的指针,并将该指针赋给 woman 变量。这时 woman 变量的类型就是 *Woman,而不是 Woman。因此,对 woman 变量进行方法调用时,需要使用指针类型,即:woman.Say() 或者 (*woman).Say()。同时,将 woman 赋值给实现了该类型的接口变量时(如 var canTalk CanTalk = woman),Go 会直接将 woman 指针赋给接口变量,无需做任何转换。

因此,使用 woman := &Woman{} 创建指向 Woman 类型的指针更为常见,可以方便地对对象进行修改和传递。如果需要直接操作值本身的数据,可以使用值类型。

需要注意的是:编译器确实会进行隐式转换,但这并不意味着你应该依赖于它。在一些情况下,隐式转换可能会导致编译错误或者程序行为不正确的问题。因此,尽量避免过度依赖编译器进行隐式转换,而是应该显式地转换数据类型以确保代码的正确性和可读性。

这里就得出一个经验,一旦你的函数的接收者的类型是指针类型并且实现一个接口的时候,在将一个实例赋值给接口类型的变量的时候,一定要是指针类型,否则编译器会报错。而如果你的函数接受者是值类型并且实现一个接口的时候,实例不管是值类型还是指针类型都会被编译器隐式生成一个指针指向接口变量,就不会报错了。

相关文章
|
8天前
|
Go
go语言map、实现set
go语言map、实现set
13 0
|
8天前
|
Go
go语言数组与切片
go语言数组与切片
16 0
|
1天前
|
存储 Go API
一个go语言编码的例子
【7月更文挑战第2天】本文介绍Go语言使用Unicode字符集和UTF-8编码。Go中,`unicode/utf8`包处理编码转换,如`EncodeRune`和`DecodeRune`。`golang.org/x/text`库支持更多编码转换,如GBK到UTF-8。编码规则覆盖7位至21位的不同长度码点。
67 1
一个go语言编码的例子
|
4天前
|
JSON 算法 测试技术
在go语言中调试程序
【6月更文挑战第29天】Go语言内置`testing`包支持单元测试、基准测试和模糊测试。`go test`命令可执行测试,如`-run`选择特定测试,`-bench`运行基准测试,`-fuzz`进行模糊测试。
16 2
在go语言中调试程序
|
2天前
|
安全 Go
Go语言的iota关键字有什么用途?
**Go语言中的`iota`是常量生成器,用于在`const`声明中创建递增的常量。`iota`在每个新的`const`块重置为0,然后逐行递增,简化了枚举类型或常量序列的定义。例如,定义星期枚举:** ```markdown ```go type Weekday int const ( Sunday Weekday = iota // 0 Monday // 1 Tuesday // 2 ... ) ``` 同样,`iota`可用于定义不同组的常量,如状态码和标志位,保持各自组内的递增,提高代码可读性。
|
7天前
|
Devops Go 云计算
Go语言发展现状:历史、应用、优势与挑战
Go语言发展现状:历史、应用、优势与挑战
|
8天前
|
Go
go语言的hello,world
go语言的hello,world
10 1
|
2天前
|
IDE Linux Go
记录一个go语言与IDE之间的问题
【7月更文挑战第1天】本文介绍在IDE中调试Go应用可能遇到的问题。当问题与IDE的自动完成有关,可以试着使用其他编辑器如Linux的vim是否无此问题。这可以验证表明IDE可能不完全兼容最新语言版本,建议使用无自动检测工具临时解决。
18 0
|
6天前
|
编译器 Go C++
必知的技术知识:go语言快速入门教程
必知的技术知识:go语言快速入门教程
|
7天前
|
编译器 Go 开发者