Go语言之指针篇(四)

转发自自己的公众号(灰子学技术)

一、写在前面的话:

在介绍Go的指针之前,我们需要先介绍下,指针是什么?为什么我们需要指针?(备注:这里指的是原始指针。)

1.指针是什么?

首先指针是一个变量,全名叫做指针变量,只不过这个变量里面存储的值是一个地址而已。所以指针,哪怕是空指针,都是有地址的,因为变量都是有地址的。例子如下:

Output:

从下面的输出,我们可以看出空指针a的地址是存在,在a指向b之前,指针a的值为nil,指向b之后,数值变成了变量b的地址,而对a  做操作*a的话,数值为变量b对应的数值10。

2.为什么我们需要指针?

我们之所以需要指针,是因为我们要做间接寻址,就是在程序运行的时候,我们希望执行到一个地址段的时候,去跳到另外一个地址段去执行。详细介绍参见:

知乎:为什么需要指针?

二、指针的用法:

讲完了这些之后,让我们回归正题,指针在Go中是什么样子的,它与C++中有何不同之处。

1.使用规则

1).C++使用操作符->,Go使用的操作符变成了.。

2).*的操作是相同的,表示的是取指针指向地址所存储的数据。

3).C++中,指针需要创建和删除,new和delete,用来管理和释放内存空间。Go可以通过make来创建,但是不需要自己释放。

2.内存中的存储位置不同

C++中,指针分配的内存在堆中,而Go是在栈中。

C++中,指针指向一个局部变量的话,如果这个局部变量的地址被销毁了,那么这个指针指向的地址里面的数据,有可能是脏数据。Go中,指针可以指向局部变量,因为变量的外溢,就算这个变量超出了自己的作用域,也不会被释放,因为还有指针再用,这部分操作是Go在语言层面保证的。

例子:

C++:

该例子示范了,栈内的变量在被指针指向 的时候,一旦改内存位置被再次使用,指针所指向的内存的数据,就变成了一个脏数据。

Output:

b[1024*128]就是为了覆盖原来局部变量b中的数值,从输出的结果来看*a的数值,不是10,是32522,变成了一个不期望的数值。

Go:

与C++例子的思路相同,我们通过程序能够看出来,局部变量b所指向的内存,一旦被指针使用,在指针还有效的前提下,内存是不会被覆盖的。(备注:该结论也可以通过反汇编代码来确定。)

Output:

通过输出来看,就算重新定义了b[1024*1024]大小,局部变量b所占内存的数值,依然是10,并没有变化。

备注:Go中指针的这种使用方式,有些类似C++中的shared_ptr,在reference的数值变为0之前,所指向的内存不会被释放,不过shared_ptr所指向的内存,是在堆中,并非栈中,后续会单独整理C++的智能指针篇章,欢迎关注。

其他:

C++的几种new和delete操作:C++中几种new操作

C++的几种智能指针:后续会整理,敬请期待。


欢迎关注,订阅,评论,共同学习,共同进步!

灰子学技术:

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Go语言中,指针变量和指针地址的概念与其他编程语言类似。指针变量是存储内存地址的变量,而指针地址是指针变量所指向的内存地址。 以下是关于Go语言指针变量和指针地址的介绍和演示: 1. 声明指针变量和指针地址: ```go var var_name *var-type ``` 其中,`var_name`是指针变量名,`*var-type`是指针类型,`*`号用于指定变量是作为一个指针。 2. 指针变量的使用: ```go package main import "fmt" func main() { var a int = 10 var ptr *int // 声明指针变量 ptr = &a // 将变量a的地址赋值给指针变量ptr fmt.Printf("a的值:%d\n", a) // 输出:10 fmt.Printf("a的地址:%d\n", &a) // 输出:变量a的地址 fmt.Printf("ptr的值:%d\n", *ptr) // 输出:指针变量ptr所指向的值 fmt.Printf("ptr的地址:%d\n", &ptr) // 输出:指针变量ptr的地址 } ``` 运行结果为: ``` a的值:10 a的地址:变量a的地址 ptr的值:10 ptr的地址:指针变量ptr的地址 ``` 3. 指针数组和指向指针指针: ```go package main import "fmt" func main() { var ptr [3]*int // 声明指针数组 a := [...]int{10, 100, 200} // 实际数组 for i := 0; i < len(a); i++ { ptr[i] = &a[i] // 地址赋值给指针数组 fmt.Printf("第%d个元素的指针地址:%d\n", i, &a[i]) } for j := 0; j < len(a); j++ { fmt.Printf("a[%d] = %d\n", j, *ptr[j]) // 使用指针变量值指向值进行遍历 } } ``` 运行结果为: ``` 第0个元素的指针地址:变量a[0]的地址 第1个元素的指针地址:变量a[1]的地址 第2个元素的指针地址:变量a[2]的地址 a[0] = 10 a[1] = 100 a[2] = 200 ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值