Go 值传递与引用传递的方法

 更新时间:2019年03月18日 09:44:30   作者:chencanxin  
这篇文章主要介绍了Go 值传递与引用传递的方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧

问题引入

  • 什么时候选择 T 作为参数类型,什么时候选择 *T 作为参数类型?
  • [ ] T 是传递的指针还是值?选择 [ ] T 还是 [ ] *T ?
  • 哪些类型复制和传递的时候会创建副本?
  • 什么情况下会发生副本创建?

T 和 *T 当做函数参数时都是传递它的副本

先看传 T 的情况:

type user struct {
  id int
  name string
}

func passByValue(_u user){
  _u.id++
  _u.name="jack"

  // when printing structs, the plus flag (%+v) adds field names
  fmt.Printf("_u 值:%+v;地址:%p; \n",_u,&_u)
}

func exp2(){
  u:=user{1,"peter"}
  fmt.Printf("原始 u 值:%+v; 地址: %p;\n",u,&u)
  passByValue(u)
  fmt.Printf("执行完函数后 u 值:%+v; 地址: %p;\n",u,&u)
}

执行 exp2 方法,输出结果为:


结果说明:

  • _u 是 u 的一份拷贝,地址不同
  • 函数内对参数的改变不影响原始的对象

再看传 *T 的情况:

type user struct {
  id int
  name string
}

func passByPointer(_u *user){
  _u.id++
  _u.name="jack"
  fmt.Printf("_u 值:%+v ;u指向的地址:%p; u本身存放地址:%p; \n",*_u,_u,&_u)
}

func exp3(){
  u:=&user{1,"peter"}
  fmt.Printf("原始u 值:%+v; 指向的地址: %p;u本身存放地址: %p; \n",*u,u,&u)
  passByPointer(u)
  fmt.Printf("原始u 值:%+v; 指向的地址: %p;u本身存放地址: %p; \n",*u,u,&u)
}

执行 exp3 方法的输出结果为:

注意到,虽然参数 _u 仍然是 u 的一份拷贝对象,但是原始对象的值还是改变了。可以这么理解,因为 u 指针和 _u 指针都指向同一个对象,即 0xc0000484a0 地址上存放的对象,_u.name="jack"可以看做*(_u).name="jack,即取值后再改变值。

改变指针参数的地址

type user struct {
  id int
  name string
}

func changeAddress(_u *user){
  _u=&user{2,"jack"}
  fmt.Printf("参数_u 值:%+v ;u指向的地址:%p; u本身存放地址:%p; \n",*_u,_u,&_u)
  return
}

func exp4(){
  u:=&user{1,"peter"}
  fmt.Printf("原始u 值:%+v; 指向的地址: %p;u本身存放地址: %p; \n",*u,u,&u)
  changeAddress(u)
  fmt.Printf("执行函数后 u 值:%+v; 指向的地址: %p;u本身存放地址: %p; \n",*u,u,&u)
}

输出结果如下:

注意,执行函数后 u 值没有改变!改变了参数指向的地址,原来的对象肯定就不受影响了。

传递数组参数 vs 传递切片参数

func passSlice(_s []int){
  _s[0]=99
  fmt.Printf("_s 值:%v,地址:%p\n",_s,&_s)
}

func exp6(){
  s:=[]int{11,22,33,44}
  fmt.Printf("s 值:%v,地址:%p\n",s,&s)
  passSlice(s)
  fmt.Printf("执行函数后 s 值:%v,地址:%p\n",s,&s)
}

对切片参数的修改会影响原来的切片。

再看传递数组

func passArray(_a [3]int){
  _a[0]=99
  fmt.Printf("_a 值:%v,地址:%p\n",_a,&_a)
}

func exp7(){
  a:=[3]int{22,33,44}
  fmt.Printf("a 值:%v,地址:%p\n",a,&a)
  passArray(a)
  fmt.Printf("执行函数后 a 值:%v,地址:%p\n",a,&a)
}

对数组参数的修改并不会影响原来的切片。

总结会发生副本创建的情况

  • 赋值操作,如 u1:=u2。包括 slice,map,array 在初始化和按索引设置的时候都会创建副本
  • for-range循环也是将元素的副本赋值给循环变量,但注意一点,循环变量是被复用的,所以地址不会变
  • 将变量作为参数传递。但注意一点, slice,map,chanel 三者都和 *T 一样,属于引用传递,虽然是发生了副本创建,但是函数内对参数的值进行修改会影响原来的值。而数组不同于 slice,函数内对数组参数的值进行修改不会影响原来数组
  • 将返回值赋值给其它变量或者传递给其它的函数和方法
  • 字符串比较特殊,它的值不能修改,任何想对字符串的值做修改都会生成新的字符串
  • 函数也是一个指针类型,对函数对象的赋值只是又创建了一个对次函数对象的指针。

总结指针类型

  • slice
  • map
  • chanel
  • 函数

如何选择 T 和 *T

对函数的参数或者返回值定义成 T 还是 *T 要考虑以下几点:

  • 一般的判断标准是看副本创建的成本和需求。
  • 如果不想变量被函数所修改,那么选择类型 T
  • 如果变量是一个很大的struct或者数组,副本的创建相对会影响性能,这个时候要考虑使用*T,只创建新的指针
  • 对于函数作用域内的参数,如果定义成 T , Go 编译器尽量将对象分配到栈上,而 *T 很可能会分配到对象上,这对垃圾回收会有影响

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • Go通道channel通过通信共享内存

    Go通道channel通过通信共享内存

    这篇文章主要为大家介绍了Go通道channel通过通信共享内存示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-07-07
  • golang中的select关键字用法总结

    golang中的select关键字用法总结

    这篇文章主要介绍了golang中的select关键字用法总结,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-06-06
  • Golang使用原生http实现中间件的代码详解

    Golang使用原生http实现中间件的代码详解

    中间件(middleware):常被用来做认证校验、审计等,家常用的Iris、Gin等web框架,都包含了中间件逻辑,但有时我们引入该框架显得较为繁重,本文将介绍通过golang原生http来实现中间件操作,需要的朋友可以参考下
    2024-05-05
  • Golang sync.Map原理深入分析讲解

    Golang sync.Map原理深入分析讲解

    go中map数据结构不是线程安全的,即多个goroutine同时操作一个map,则会报错,因此go1.9之后诞生了sync.Map,sync.Map思路来自java的ConcurrentHashMap
    2022-12-12
  • 从入门到精通:Go语言XML数据解析指南

    从入门到精通:Go语言XML数据解析指南

    Go语言的XML包提供了强大的数据解析功能,让你轻松处理各种XML格式的数据,这个指南将带你深入了解如何使用Go语言的XML包,快速上手XML数据解析,准备好开启XML解析之旅了吗?Let's Go!
    2024-03-03
  • Go并发与锁的两种方式该如何提效详解

    Go并发与锁的两种方式该如何提效详解

    如果没有锁,在我们的项目中,可能会存在多个goroutine同时操作一个资源(临界区),这种情况会发生竞态问题(数据竞态),下面这篇文章主要给大家介绍了关于Go并发与锁的两种方式该如何提效的相关资料,需要的朋友可以参考下
    2022-12-12
  • Golang如何实现任意进制转换的方法示例

    Golang如何实现任意进制转换的方法示例

    进制转换是人们利用符号来计数的方法,进制转换由一组数码符号和两个基本因素“基数”与“位权”构成,这篇文章主要给大家介绍了关于Golang如何实现10进制转换62进制的方法,文中给出了详细的示例代码供大家参考学习学习,下面随着小编来一起学习学习吧。
    2017-09-09
  • Go函数全景从基础到高阶深度剖析

    Go函数全景从基础到高阶深度剖析

    本文深入探索Go语言中的函数特性,从基础函数定义到特殊函数类型,再到高阶函数的使用和函数调用优化,通过详细的代码示例和专业解析,读者不仅可以掌握函数的核心概念,还能了解如何在实践中有效利用这些特性来提高代码质量和性能
    2023-10-10
  • Go语言实现管理多个数据库连接

    Go语言实现管理多个数据库连接

    在软件开发过程中,使用 MySQL、PostgreSQL 或其他数据库是很常见的,由于配置和要求不同,管理这些连接可能具有挑战性,下面就来和大家聊聊如何在Go中管理多个数据库连接吧
    2023-10-10
  • go kratos源码及配置解析

    go kratos源码及配置解析

    这篇文章主要为大家介绍了go kratos源码及配置解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-12-12

最新评论