转自:https://segmentfault.com/a/1190000012329213

值类型的变量和指针类型的变量

先声明一个结构体:

type T struct {Name string
}func (t T) M1() { t.Name = "name1" } func (t *T) M2() { t.Name = "name2" }

M1() 的接收者是值类型 T, M2() 的接收者是值类型 *T , 两个方法内都是改变Name值。

下面声明一个 T 类型的变量,并调用 M1() 和 M2() 。

    t1 := T{"t1"}fmt.Println("M1调用前:", t1.Name)t1.M1()fmt.Println("M1调用后:", t1.Name) fmt.Println("M2调用前:", t1.Name) t1.M2() fmt.Println("M2调用后:", t1.Name)

输出结果为:

M1调用前: t1
M1调用后: t1
M2调用前: t1
M2调用后: name2

下面猜测一下go会怎么处理。

先来约定一下:接收者可以看作是函数的第一个参数,即这样的: func M1(t T), func M2(t *T)。 go不是面向对象的语言,所以用那种看起来像面向对象的语法来理解可能有偏差。

当调用 t1.M1() 时相当于 M1(t1) ,实参和行参都是类型 T,可以接受。此时在M1()中的t只是t1的值拷贝,所以M1()的修改影响不到t1。

当调用 t1.M2() => M2(t1),这是将 T 类型传给了 *T 类型,go可能会取 t1 的地址传进去: M2(&t1)。所以 M2() 的修改可以影响 t1 。

  1. 类型的变量这两个方法都是拥有的。

下面声明一个 *T 类型的变量,并调用 M1() 和 M2() 。

    t2 := &T{"t2"}fmt.Println("M1调用前:", t2.Name)t2.M1()fmt.Println("M1调用后:", t2.Name) fmt.Println("M2调用前:", t2.Name) t2.M2() fmt.Println("M2调用后:", t2.Name)

输出结果为:

M1调用前: t2
M1调用后: t2
M2调用前: t2
M2调用后: name2

t2.M1() => M1(t2), t2 是指针类型, 取 t2 的值并拷贝一份传给 M1。

t2.M2() => M2(t2),都是指针类型,不需要转换。

*T 类型的变量也是拥有这两个方法的。

传给接口会怎样?

先声明一个接口

type Intf interface {M1()M2()
}

使用:

    var t1 T = T{"t1"}t1.M1() t1.M2() var t2 Intf = t1 t2.M1() t2.M2()

报错:

./main.go:9: cannot use t1 (type T) as type Intf in assignment:

T does not implement Intf (M2 method has pointer receiver)

var t2 Intf = t1 这一行报错。

t1 是有 M2() 方法的,但是为什么传给 t2 时传不过去呢?

简单来说,按照接口的理论:传过去【赋值】的对象必须实现了接口要求的方法,而t1没有实现M2(),t1的指针实现了M2()。另外和c语言一样,函数名本身就是指针

当把 var t2 Intf = t1 修改为 var t2 Intf = &t1 时编译通过,此时 t2 获得的是 t1 的地址, t2.M2() 的修改可以影响到 t1 了。

如果声明一个方法 func f(t Intf) , 参数的传递和上面的直接赋值是一样的情况。

嵌套类型

声明一个类型 S,将 T 嵌入进去

type S struct { T }

使用下面的例子测试一下:

    t1 := T{"t1"}     s := S{t1}      fmt.Println("M1调用前:", s.Name)     s.M1()     fmt.Println("M1调用后:", s.Name) fmt.Println("M2调用前:", s.Name) s.M2() fmt.Println("M2调用后:", s.Name) fmt.Println(t1.Name)

输出:

M1调用前: t1 
M1调用后: t1 
M2调用前: t1 
M2调用后: name2 
t1

将 T 嵌入 S, 那么 T 拥有的方法和属性 S 也是拥有的,但是接收者却不是 S 而是 T。

所以 s.M1() 相当于 M1(t1) 而不是 M1(s)。

最后 t1 的值没有改变,因为我们嵌入的是 T 类型,所以 S{t1} 的时候是将 t1 拷贝了一份。

假如我们将 s 赋值给 Intf 接口会怎么样呢?

    var intf Intf = s     intf.M1()     intf.M2()

报错:

cannot use s (type S) as type Intf in assignment: S does not implement Intf (M2 method has pointer receiver)

还是 M2() 的问题,因为 s 此时还是值类型。

var intf Intf = &s 这样的话编译通过了,如果在 intf.M2() 中改变了 Name 的值, s.Name 被改变了,但是 t1.Name 依然没变,因为现在 t1 和 s 已经没有联系了。

下面嵌入 *T 试试:

type S struct { *T }

使用时这样:

    t1 := T{"t1"}     s := S{&t1}      fmt.Println("M1调用前:", s.Name)     s.M1()     fmt.Println("M1调用后:", s.Name) fmt.Println("M2调用前:", s.Name) s.M2() fmt.Println("M2调用后:", s.Name) fmt.Println(t1.Name)

M1调用前: t1
M1调用后: t1
M2调用前: t1
M2调用后: name2
name2
惟一的区别是最后 t1 的值变了,因为我们复制的是指针。

接着赋值给接口试试:

    var intf Intf = s     intf.M1()     intf.M2()     fmt.Println(s.Name)

编译没有报错。这里我们传递给 intf 的是值类型而不是指针,为什么可以通过呢?

拷贝 s 的时候里面的 T 是指针类型,所以调用 M2() 的时候传递进去的是一个指针。

var intf Intf = &s 的效果和上面一样。


转载于:https://www.cnblogs.com/xiaouisme/p/10639090.html

golang中值类型/指针类型的变量区别总结相关推荐

  1. java中的就近原则、方法中值传递和引用传递的区别、什么是构造方法、this关键字用法、什么是封装

    你知道java中的就近原则嘛? package com.Test.java; /*** * 在java里面有一个"就近原则"详情可以参照下面代码**/public class Ac ...

  2. 面试中常被问到(四)C/C++中数组,指针和引用的区别

    C/C++中数组,指针和引用的区别 数组和指针: 数组 指针 保存数据 保存数据地址 可直接访问元素的值 间接访问(首先获取指针内容作为地址,通过寻址找到元素的值) 一般大小固定,数据类型相同 一般用 ...

  3. C++中数组和指针的关系(区别)详解

    C++中数组和指针的关系(区别)详解 本文转自:http://c.biancheng.net/view/1472.html 博主在阅读后将文中几个知识点提出来放在前面: 没有方括号和下标的数组名称实际 ...

  4. golang 中string和int类型相互转换

    总结了golang中字符串和各种int类型之间的相互转换方式: string转成int: test_int, err := strconv.Atoi(test_string) if err != ni ...

  5. Golang中 int int8 int16 int32 int64的区别和取值范围

    先说结论吧,方便快速查询验证. 总结 区别 int 类型大小为 8 字节 int8 类型大小为 1 字节 int16 类型大小为 2 字节 int32 类型大小为 4 字节 int64 类型大小为 8 ...

  6. 三分钟让你掌握JavaScript中值传递和引用传递的区别

    值传递:传递的是实际参数的一个副本.基本数据类型Undefined,Null,Boolean,Number.String都是值传递. 引用传递:传递的是实际参数的地址.引用数据类型Object,Arr ...

  7. 函数返回值为指针类型的总结

    参考博客:https://blog.csdn.net/zxccaoya/article/details/53468500 char*GetString(void) {char p[]= "h ...

  8. golang中数组和slice作为参数的区别

    最近项目中有遇到类似问题,做个记录. 举个例子,[5]int为数组,[]int为slice(数组切片),数组是值类型,而slice为引用类型,值类型作为参数传入函数,只是拷贝了个副本,修改并不会作用到 ...

  9. golang中的值类型和引用类型

    值类型与引用类型 不管是Java还是golang中,都有值类型和引用类型的概念.在使用两者时,发现这两种语言之间还是有差异的. 值类型 值类型:这些类型的变量直接指向存在内存中的值,值类型的变量的值存 ...

最新文章

  1. 来看看如何在 C# 中使用反射
  2. c语言基本类型学习小结
  3. U3D协程Coroutine之WWW与Update()的并行测试
  4. 使用$.ajax向服务器发送请求
  5. docker log 文件 清理
  6. 计算机 函数的应用,职称计算机:公式与函数的应用(2)
  7. 如何迅速分析出系统CPU的瓶颈在哪里?
  8. c oracle更新参数化,Dapper的参数化更新和插入?
  9. 指针笔试题及其解析、字符串左旋、字符串追加strcat
  10. 轻量级网络模型之MobileNet系列
  11. mysql 5.7连接java_MACOS mysql 5.7 Navicat 可以连接 , Java 报错
  12. python刷屏代码_python 刷屏
  13. OPPO A57怎么刷机 OPPO A57的刷机教程 OPPO A57完美解除账号锁
  14. 【拉格朗日差值法】 公式
  15. html能计算吗,使用(JavaScript和HTML)计算总数
  16. 智能语音助手的时代,我们还有这样的一股清流:114!
  17. Spring MVC的数据绑定(简单数据绑定+POJO类型绑定)
  18. 大学计算机基础教学目标,大学计算机基础教学大纲2015.doc
  19. 《打造Facebook》书摘(1):CEO小扎
  20. 计算机开机图片怎么换,如何把电脑开机画面换成自己的图片?

热门文章

  1. echarts图表随着窗口大小改变自刷新问题解决
  2. mysql change column_Modify column Vs change column
  3. npm工具运行Vue项目
  4. go语言报错:main redeclared in this block
  5. 网络基础知识:FTP命令及应答码
  6. mysql引擎机制_MySQL引擎特性:InnoDB同步机制
  7. Java学习路线,Java SE,EE,ME的区别,SSM框架基本概念
  8. 模幂运算问题,使用朴素算法和重复-平方算法(快速幂+C#计算程序运行时间)
  9. 【LOJ10050】The XOR Largest Pair(字典树)
  10. 【POJ2259】Team Queue(队列,模拟)