golang中值类型/指针类型的变量区别总结
前言
值类型:所有像int、float、bool和string这些类型都属于值类型,使用这些类型的变量直接指向存在内存中的值,值类型的变量的值存储在栈中。当使用等号=将一个变量的值赋给另一个变量时,如j=i,实际上是在内存中将i的值进行了拷贝。可以通过&i获取变量i的内存地址
指针类型:简单地说go语言的指针类型和C/C++的指针类型用法是一样的,除了出去安全性的考虑,go语言增加了一些限制,包括如下几条:
- 不同类型的指针不能互相转化,例如*int,int32,以及int64
- 任何普通指针类型*T和uintptr之间不能互相转化
- 指针变量不能进行运算,比如C/C++里面的++,--运算
下面将给大家详细介绍golang中值类型/指针类型的变量的一些区别,下面话不多说了,来一起看看详细的介绍吧。
值类型的变量和指针类型的变量
先声明一个结构体:
typeTstruct{ Namestring } func(tT)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会怎么处理。
先来约定一下:接收者可以看作是函数的第一个参数,即这样的:funcM1(tT),funcM2(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。
类型的变量这两个方法都是拥有的。
下面声明一个*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类型的变量也是拥有这两个方法的。
传给接口会怎样?
先声明一个接口
typeIntfinterface{ M1() M2() }
使用:
vart1T=T{"t1"} t1.M1() t1.M2() vart2Intf=t1 t2.M1() t2.M2()
报错:
./main.go:9:cannotuset1(typeT)astypeIntfinassignment:
TdoesnotimplementIntf(M2methodhaspointerreceiver)
vart2Intf=t1这一行报错。
t1是有M2()方法的,但是为什么传给t2时传不过去呢?
简单来说,按照接口的理论:传过去【赋值】的对象必须实现了接口要求的方法,而t1没有实现M2(),t1的指针实现了M2()。另外和c语言一样,函数名本身就是指针
当把vart2Intf=t1修改为vart2Intf=&t1时编译通过,此时t2获得的是t1的地址,t2.M2()的修改可以影响到t1了。
如果声明一个方法funcf(tIntf),参数的传递和上面的直接赋值是一样的情况。
嵌套类型
声明一个类型S,将T嵌入进去
typeSstruct{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接口会怎么样呢?
varintfIntf=s intf.M1() intf.M2()
报错:
cannotuses(typeS)astypeIntfinassignment:SdoesnotimplementIntf(M2methodhaspointerreceiver)
还是M2()的问题,因为s此时还是值类型。
varintfIntf=&s这样的话编译通过了,如果在intf.M2()中改变了Name的值,s.Name被改变了,但是t1.Name依然没变,因为现在t1和s已经没有联系了。
下面嵌入*T试试:
typeSstruct{*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的值变了,因为我们复制的是指针。
接着赋值给接口试试:
varintfIntf=si ntf.M1() intf.M2() fmt.Println(s.Name)
编译没有报错。这里我们传递给intf的是值类型而不是指针,为什么可以通过呢?
拷贝s的时候里面的T是指针类型,所以调用M2()的时候传递进去的是一个指针。
varintfIntf=&s的效果和上面一样。
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对毛票票的支持。
声明:本文内容来源于网络,版权归原作者所有,内容由互联网用户自发贡献自行上传,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任。如果您发现有涉嫌版权的内容,欢迎发送邮件至:czq8825#qq.com(发邮件时,请将#更换为@)进行举报,并提供相关证据,一经查实,本站将立刻删除涉嫌侵权内容。