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

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

先声明一个结构体:

  1. type T struct {
  2. Name string
  3. }
  4. func (t T) M1() {
  5. t.Name = "name1"
  6. }
  7. func (t *T) M2() {
  8. t.Name = "name2"
  9. }

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

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

  1. t1 := T{"t1"}
  2. fmt.Println("M1调用前:", t1.Name)
  3. t1.M1()
  4. fmt.Println("M1调用后:", t1.Name)
  5. fmt.Println("M2调用前:", t1.Name)
  6. t1.M2()
  7. 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() 。

  1. t2 := &T{"t2"}
  2. fmt.Println("M1调用前:", t2.Name)
  3. t2.M1()
  4. fmt.Println("M1调用后:", t2.Name)
  5. fmt.Println("M2调用前:", t2.Name)
  6. t2.M2()
  7. 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 类型的变量也是拥有这两个方法的。

传给接口会怎样?

先声明一个接口

  1. type Intf interface {
  2. M1()
  3. M2()
  4. }

使用:

  1. var t1 T = T{"t1"}
  2. t1.M1()
  3. t1.M2()
  4. var t2 Intf = t1
  5. t2.M1()
  6. t2.M2()

报错:

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

  1. 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 嵌入进去

  1. type S struct { T }

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

  1. t1 := T{"t1"}
  2. s := S{t1}
  3. fmt.Println("M1调用前:", s.Name)
  4. s.M1()
  5. fmt.Println("M1调用后:", s.Name)
  6. fmt.Println("M2调用前:", s.Name)
  7. s.M2()
  8. fmt.Println("M2调用后:", s.Name)
  9. 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 接口会怎么样呢?

  1. var intf Intf = s
  2. intf.M1()
  3. 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 试试:

  1. type S struct { *}

使用时这样:

  1. t1 := T{"t1"}
  2. s := S{&t1}
  3. fmt.Println("M1调用前:", s.Name)
  4. s.M1()
  5. fmt.Println("M1调用后:", s.Name)
  6. fmt.Println("M2调用前:", s.Name)
  7. s.M2()
  8. fmt.Println("M2调用后:", s.Name)
  9. fmt.Println(t1.Name)

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

接着赋值给接口试试:

  1. var intf Intf = s i
  2. ntf.M1()
  3. intf.M2()
  4. fmt.Println(s.Name)

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

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

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


golang中值类型/指针类型的变量区别总结的更多相关文章

  1. golang(3):strings和strconv使用 & 时间和日期类型 & 指针类型 & 流程控制 & 函数

    strings和strconv使用 . strings.HasPrefix(s string, prefix string) bool: // 判断字符串s是否以prefix开头 . . string ...

  2. 《挑战30天C++入门极限》新手入门:C/C++中数组和指针类型的关系

        新手入门:C/C++中数组和指针类型的关系 对于数组和多维数组的内容这里就不再讨论了,前面的教程有过说明,这里主要讲述的数组和指针类型的关系,通过对他们之间关系的了解可以更加深入的掌握数组和指 ...

  3. golang中值类型的嵌入式字段和指针类型的嵌入式字段

    总结: 1. 值类型的嵌入式字段,该类型拥有值类型的方法集,没有值指针类型的方法集 2. 指针类型的嵌入式字段,该类型拥有值指针类型的方法集,没有值类型的方法集,并且,该类型的指针类型也有值指针类型的 ...

  4. js中值的基本类型与引用类型,以及对象引用,对象的浅拷贝与深拷贝

    js有两种类型的值:栈:原始数据类型(undefinen,null,boolead,number,string)堆:引用数据类型(对象,函数和数组)两种类型的区别是:储存位置不同,原始数据类型直接存储 ...

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

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

  6. 《精通C#》自定义类型转化-扩展方法-匿名类型-指针类型(11.3-11.6)

    1.类型转化在C#中有很多,常用的是int类型转string等,这些都有微软给我们定义好的,我们需要的时候直接调用就是了,这是值类型中的转化,有时候我们还会需要类类型(包括结构struct)的转化,还 ...

  7. golang中字符串、bytes类型切片、16进制字符串之间的转换

    func main() { // 字符串转bytes类型 name := "马亚南" fmt.Println(name) // 马亚南 bName := []byte(name) ...

  8. redis中键值对中值的各种类型

    1 value的最基本的数据类型是String 2 如果value是一张图片 先对图片进行base64编码成一个字符串,然后再保存到redis中,用的时候进行base64解码即可. 这是base64的 ...

  9. golang中结构体指针的应用

    package main import ( "fmt" ) type School struct { brand string city string } type Class s ...

随机推荐

  1. 记录Ubuntu & Windows下安装PyV8

    https://blog.csdn.net/hanshileiai/article/details/51628173

  2. Mysql You can change this value on the server by setting the max_allowed_packet' variable. 异常

    MySQL根据配置文件会限制server接受的数据包大小. 有时候大的插入和更新会被max_allowed_packet 参数限制掉,导致失败. 查看目前配置, Windows 系统 配置文件为 my ...

  3. CODEVS 1074 食物链 2001年NOI全国竞赛(洛谷 P2024)

    题目描述 Description 动物王国中有三类动物 A,B,C,这三类动物的食物链构成了有趣的环形.A吃B,B吃C,C吃A. 现有N个动物,以1-N编号.每个动物都是A,B,C中的一种,但是我们并 ...

  4. java的类和对象

    创建狗狗类: /** * 狗狗类 * @author Administrator * */ public class Dog { String name="无名氏"; //姓名 i ...

  5. [CF1038D]Slime

    [CF1038D]Slime 题目大意: 有\(n(n\le5\times10^5)\)只史莱姆,每只史莱姆有一个分数\(w_i(|w_i|le10^9)\),每次一只史莱姆可以吞掉左边的或者右边的史 ...

  6. 通过xml处理sql语句时对小于号与大于号的处理转换

    以上方法,很容易使用,直接ss < #{ss} 法二   <![CDATA[>=]]>表示大于等于      变量<![CDATA[ < ]]>#{变量}表示 ...

  7. linux上下载安装mysql,并使用

    一:官网找mysql的安装包 可参考https://blog.csdn.net/a18852867035/article/details/81610611, 我是在https://dev.mysql. ...

  8. Mybatis 传递多个参数

    Mybatis提供了4种传递多个参数的方法: 1 Map sql语句 接口 调用方法 这个方法虽然简单易用,但是存在一个弊端:Map存储的元素是键值对,可读性不好. 2 注解 使用MyBatis的参数 ...

  9. 使用iscroll,无法正常滑动的原因

    iscroll的dom元素的结构是固定的,swiper是容器,scroll是需要滚动的容器,list是滚动的内容 <div class="swiper"> <di ...

  10. 最长递增子序列( LIS)

    LIS LIS的优化说白了其实是贪心算法,比如说让你求一个最长上升子序列把,一起走一遍. 比如说(4, 2, 3, 1, 2,3,5)这个序列,求他的最长上升子序列,那么来看,如果求最长的上升序列,那 ...