登录
首页 >  Golang >  Go问答

混淆:Go 中函数参数中的指针、切片和接口

来源:stackoverflow

时间:2024-03-24 21:54:44 420浏览 收藏

Go 中传递函数参数时,指针、切片和接口的行为可能令人困惑。当传递切片时,传递的是副本,而对该副本的修改不会影响原始切片。然而,如果将接口作为参数传递,则传递的是指向原始值的指针,对该值的修改将影响原始值。这两种情况下,变量 i 的值不同,因为在第一种情况下,它持有指向切片的指针,而在第二种情况下,它持有指向 interface{} 值的指针。

问题内容

我一直在阅读有关 go 如何通过指针与值将参数传递给函数的内容。我一直在阅读有关接口类型的内容。而且我一直在篡改反射包。但显然,由于这里的示例代码,我仍然不明白它是如何工作的:

package main

import (
  "reflect"
  "fmt"
)
type Business struct {
  Name string
}

func DoSomething(b []Business) {

  var i interface{}
  i = &b
  v := reflect.ValueOf(i).Elem()

  for c:=0 ;c<10; c++ {

    z := reflect.New(v.Type().Elem())
    s := reflect.ValueOf(z.Interface()).Elem()
    s.Field(0).SetString("Pizza Store "+ fmt.Sprintf("%v",c))
    v.Set(reflect.Append(v, z.Elem()))
  }
  fmt.Println(b)

}

func main() {

  business := []Business{}
  DoSomething(business)

}

当我运行此代码时,它将打印十个 business 结构的列表,其中 business.name 为 pizza 0 到 9。据我所知,在我的示例中,我的 dosomething 函数收到了业务切片的副本,并且因此,我的主函数中的 business 变量仍然不受 dosomething 所做的任何影响。

接下来我所做的是将 func dosomething(b []business) 更改为 func dosomething(b interface{})。现在,当我尝试运行脚本时,出现 panic: reflect: elem of invalid type on 行上的运行时错误 z := reflect.new(v.type().elem())

我注意到,对于 dosomething(b []business),变量 i == &[]。但对于 dosomething(b interface{}),变量 i == 0xc42000e1d0。为什么这两种情况下变量 i 不同?


解决方案


您的调试器很可能使用(或至少遵循)fmt 包的默认格式规则:

对于复合对象,元素是使用这些规则递归打印的,布局如下:

struct:             {field0 field1 ...}
array, slice:       [elem0 elem1 ...]
maps:               map[key1:value1 key2:value2 ...]
pointer to above:   &{}, &[], &map[]

在第一种情况下,i 拥有 *[]business 类型的值。因此,如果正在打印(或检查)的值是指向切片的指针,则它将打印为 &[values]

在第二种情况下,i 保存一个指向 interface{} 值的指针,该值的类型为 *interface{}。打印此类型的值时,使用默认的 %p 格式,该格式仅将内存地址打印为前缀为 0x 的十六进制值。

好了,本文到此结束,带大家了解了《混淆:Go 中函数参数中的指针、切片和接口》,希望本文对你有所帮助!关注golang学习网公众号,给大家分享更多Golang知识!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>