浅谈Golang的方法传递值应该注意的地方

作者:piperck 时间:2024-02-12 07:40:20 

其实最近看了不少Golang接口以及方法的阐述都有一个地方没说得特别明白。就是在Golang编译隐式转换传递给方法使用的时候,和调用函数时的区别。

我们都知道,在我们为一个类型变量申明了一个方法的时候,我们可以使用类似于self.method来调用这个方法,而且无论你申明的方法的 * 是指针 * 还是值 * ,Golang都可以帮你隐式转换为正确的值供方法使用。

让我们来看一个例子:


package main
import "fmt"
type duration int
func (d *duration) pretty() string {
 return fmt.Sprintf("Duration: %d", d)
}
func main() {
 var kk duration
 kk = 3
 kk.pretty()
}

在这个例子中,创建了一个类型为duration的变量kk,并且duration这个类型上有指针 * 的方法pretty()这个时候无论你使用kk.pretty()还有使用(&kk).pretty()都会正确执行,并且就算 * 不是指针类型而是值类型,同上一样。Golang编译器会将你传入的值隐式转换为正确的传入对象。

这个不难理解,但是有一个跟他很像的特性,却会让这个问题变得很绕。那就是调用接口的时候出现的情况

同样我们来看一个例子:


package main
import (
 "fmt"
)
type notifier interface {
 notify()
}
type user struct {
 name string
 email string
}
func(u *user) notify() {
 fmt.Printf("Sending user email to %s<%s>\n",
 u.name,
 u.email)
}
func sendNotification(n notifier) {
 n.notify()
}
func main() {
 u := user{"Bill", "bill@xiachufang.com"}
 sendNotification(&u)
}

这个例子就不是用类型直接调用自己的方法了,而是把自己当作参数传递给接口。让接口去执行对应方法。

这里注意,接口对于类型的要求就十分严格了,接口在神明的时候会指定,拥有哪些方法(这里的方法指 方法名, 方法参数,以及方法返回类型)。实现了这些方法就实现了这个接口。这里我们调用sendNotification这个方法需要传递进实现了notifier这个接口的变量做参数。查看notifier代码可以注意到,他实现了一个notify的方法。而我们的user实现了一个指针 * 的notify方法。但是这里注意,传递值必须遵守一个条件即:

如果接口实现方法,类型自己的实现使用的是值 * ,那么在传递值的时候无论使用指针还是值都可以。

如果接口实现方法,类型自己的实现使用的是指针 * ,那么在传递值的时候必须传递地址。

所以上面的例子, * 是指针 * ,我们必须传递地址,如果传递值则会报错。

那么是为什么这里又不能进行隐式转换了呢?

实际上是因为,编译器并不能总能自动获得一个值的地方,也就是说你传u,编译器不一定能知道u的地址是啥。。他可能没有办法帮你完成转换。

补充:Golang 数组(切片)的值传递与引用传递

Go语言中函数的参数都是按值进行传递的,即使参数是指针,也是指针的一个副本。习惯上把指针的函数参数称之为地址传参,即引用传递,而非指针的函数参数称为值传参

地址传参在大对象上效率比值传参好,在内部相当于用指针地址赋值,而不用复制整个对象

一、数组的值传递

Golang数组作为参数传入函数时,进行的是值传递,这里与Java数组的引用传递是不同的,示例如下


package main
import "fmt"
func main() {
arr := [8]int{}
for i := 0; i < 8; i++ {
arr[i] = i
}
fmt.Println(arr)
exchange(arr)
fmt.Println(arr)
}
func exchange(arr [8]int) {
for k, v := range arr {
arr[k] = v * 2
}
}

运行结果如下:

浅谈Golang的方法传递值应该注意的地方

二、数组的引用传递

默认情况下Golang的数组传递是值传递,但当我们想要对传入的数组进行修改时,可以使用指针来对数组进行操作,如下


package main
import "fmt"
func main() {
arr := [8]int{}
for i := 0; i < 8; i++ {
arr[i] = i
}
fmt.Println(arr)
exchangeByAddress(&arr)
fmt.Println(arr)
}
func exchangeByAddress(arr *[8]int) {
for k, v := range *arr {
arr[k] = v * 2
}
}

运行结果如下:

浅谈Golang的方法传递值应该注意的地方

三、切片的引用传递

Golang中的切片与Java中的ArrayList集合类似,进行的是引用传递


package main
import "fmt"
func main() {
slice := []int{1,2,3,4,5}
fmt.Println(slice)
exchangeSlice(slice)
fmt.Println(slice)
}
func exchangeSlice(slice []int) {
for k, v := range slice {
slice[k] = v * 2
}
}

运行结果:

浅谈Golang的方法传递值应该注意的地方

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。如有错误或未考虑完全的地方,望不吝赐教。

来源:https://www.cnblogs.com/piperck/p/6550019.html

标签:Golang,传递值
0
投稿

猜你喜欢

  • 一个css垂直水平居中布局,css效果

    2008-11-03 11:40:00
  • javascript cookie的基本操作(添加和删除)

    2024-05-11 09:43:24
  • Go与C语言的互操作实现

    2024-02-04 08:39:28
  • MySql安装与配置方法(MySQL添加用户、删除用户与授权)

    2024-01-25 07:25:01
  • Python字符串拼接、截取及替换方法总结分析

    2023-12-03 14:38:57
  • 对python cv2批量灰度图片并保存的实例讲解

    2022-06-11 18:21:36
  • python sys.argv[]用法实例详解

    2023-10-15 17:21:55
  • 详解微信小程序开发之下拉刷新 上拉加载

    2024-05-10 13:59:52
  • PHP实现将MySQL重复ID二维数组重组为三维数组的方法

    2023-11-18 03:28:57
  • MySQL 5.5的max_allowed_packet属性的修改方法

    2024-01-19 05:21:02
  • JavaScript实现的一个计算数字步数的算法分享

    2024-05-03 15:32:42
  • 解决pycharm 格式报错tabs和space不一致问题

    2022-10-12 11:36:12
  • TensorFlow中关于tf.app.flags命令行参数解析模块

    2021-10-17 03:40:40
  • javascript替换已有元素replaceChild()使用介绍

    2024-04-10 13:55:08
  • base href 使用方法详解

    2008-05-18 13:27:00
  • Tensorflow Summary用法学习笔记

    2023-08-12 15:08:53
  • pyinstaller打包python3.6和PyQt5中各种错误的解决方案汇总

    2021-01-13 18:58:48
  • Python使用monkey.patch_all()解决协程阻塞问题

    2021-05-11 17:27:19
  • nlp计数法应用于PTB数据集示例详解

    2023-10-26 17:24:07
  • 用户凭什么跟你注册?

    2011-06-10 13:16:00
  • asp之家 网络编程 m.aspxhome.com