源码网商城,靠谱的源码在线交易网站 我的订单 购物车 帮助

源码网商城

golang中值类型/指针类型的变量区别总结

  • 时间:2022-05-05 12:48 编辑: 来源: 阅读:
  • 扫一扫,手机访问
摘要:golang中值类型/指针类型的变量区别总结
[b]前言[/b] 值类型:所有像int、float、bool和string这些类型都属于值类型,使用这些类型的变量直接指向存在内存中的值,值类型的变量的值存储在栈中。当使用等号=将一个变量的值赋给另一个变量时,如 j = i ,实际上是在内存中将 i 的值进行了拷贝。可以通过 &i 获取变量 i 的内存地址 指针类型:简单地说go语言的指针类型和C/C++的指针类型用法是一样的,除了出去安全性的考虑,go语言增加了一些限制,包括如下几条: [list] [*]不同类型的指针不能互相转化,例如*int, int32, 以及int64[/*] [*]任何普通指针类型*T和uintptr之间不能互相转化[/*] [*]指针变量不能进行运算, 比如C/C++里面的++, --运算[/*] [/list] 下面将给大家详细介绍golang中值类型/指针类型的变量的一些区别,下面话不多说了,来一起看看详细的介绍吧。 [b]值类型的变量和指针类型的变量[/b] 先声明一个结构体:
type T struct {
 Name string
}
func (t T) M1() {
 t.Name = "name1"
}
func (t *T) M2() {
 t.Name = "name2"
}
[code]M1() [/code]的接收者是值类型 T, [code]M2() [/code]的接收者是值类型 *T , 两个方法内都是改变Name值。 下面声明一个 T 类型的变量,并调用 [code]M1()[/code] 和 [code]M2()[/code] 。
 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会怎么处理。 先来约定一下:接收者可以看作是函数的第一个参数,即这样的: [code]func M1(t T)[/code] , [code]func M2(t *T)[/code] 。 go不是面向对象的语言,所以用那种看起来像面向对象的语法来理解可能有偏差。 当调用[code] t1.M1() [/code]时相当于 [code]M1(t1)[/code] ,实参和行参都是类型 T,可以接受。此时在[code]M1()[/code]中的t只是t1的值拷贝,所以[code]M1()[/code]的修改影响不到t1。 当调用 [code]t1.M2() => M2(t1)[/code] ,这是将 T 类型传给了 *T 类型,go可能会取 t1 的地址传进去: [code]M2(&t1)[/code] 。所以 [code]M2() [/code]的修改可以影响 t1 。 类型的变量这两个方法都是拥有的。 下面声明一个 *T 类型的变量,并调用 [code]M1()[/code] 和 [code]M2() [/code]。
 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 [code]t2.M1() => M1(t2) [/code], t2 是指针类型, 取 t2 的值并拷贝一份传给 M1。 [code]t2.M2() => M2(t2)[/code] ,都是指针类型,不需要转换。 *T 类型的变量也是拥有这两个方法的。 [b]传给接口会怎样?[/b] 先声明一个接口
type Intf interface {
 M1()
 M2()
}
使用:
 var t1 T = T{"t1"}
 t1.M1()
 t1.M2()
 var t2 Intf = t1
 t2.M1()
 t2.M2()
报错: [code]./main.go:9: cannot use t1 (type T) as type Intf in assignment:[/code]
T does not implement Intf (M2 method has pointer receiver)
[code]var t2 Intf = t1[/code] 这一行报错。 t1 是有 [code]M2() [/code]方法的,但是为什么传给 t2 时传不过去呢? 简单来说,按照接口的理论:传过去【赋值】的对象必须实现了接口要求的方法,而t1没有实现[code]M2()[/code] ,t1的指针实现了[code]M2()[/code] 。另外和c语言一样,函数名本身就是指针 当把 [code]var t2 Intf = t1[/code] 修改为 [code]var t2 Intf = &t1[/code] 时编译通过,此时 t2 获得的是 t1 的地址, [code]t2.M2()[/code] 的修改可以影响到 t1 了。 如果声明一个方法 fun[code]c f(t Intf)[/code] , 参数的传递和上面的直接赋值是一样的情况。 [b]嵌套类型[/b] 声明一个类型 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。 所以 [code]s.M1() [/code]相当于 [code]M1(t1) [/code]而不是 [code]M1(s)[/code] 。 最后 t1 的值没有改变,因为我们嵌入的是 T 类型,所以 S{t1} 的时候是将 t1 拷贝了一份。 假如我们将 s 赋值给 Intf 接口会怎么样呢?
 var intf Intf = s 
 intf.M1() 
 intf.M2()
报错: [code]cannot use s (type S) as type Intf in assignment: S does not implement Intf (M2 method has pointer receiver)[/code] 还是 [code]M2()[/code] 的问题,因为 s 此时还是值类型。 [code]var intf Intf = &s[/code] 这样的话编译通过了,如果在 [code]intf.M2() [/code]中改变了 Name 的值, [code]s.Name[/code] 被改变了,但是 [code]t1.Name[/code] 依然没变,因为现在 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 i
 ntf.M1() 
 intf.M2() 
 fmt.Println(s.Name)
编译没有报错。这里我们传递给 intf 的是值类型而不是指针,为什么可以通过呢? 拷贝 s 的时候里面的 T 是指针类型,所以调用 [code]M2() [/code]的时候传递进去的是一个指针。 [code]var intf Intf = &s[/code] 的效果和上面一样。 [b]总结[/b] 以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对编程素材网的支持。
  • 全部评论(0)
联系客服
客服电话:
400-000-3129
微信版

扫一扫进微信版
返回顶部