Go语言什么时候该使用指针 与 指针使用分析

news/2025/1/13 10:17:30/文章来源:https://www.cnblogs.com/cheyunhua/p/18243244

Go语言什么时候该使用指针 与 指针使用分析

Go语言圈
Go语言开发者的学习好助手,分享Go语言知识,技术技巧,学习与交流Go语言开发经验,互动才有助于技术的提升,每天5分钟,助你GO语言技术快乐成长
161篇原创内容

最近在学习Golang,所以将学习过程记为笔记,以后翻阅的时候也方便,顺便也给大家做一点分享,希望能坚持下去。

 

学习与交流:Go语言技术交流微信群

 

现在就开始你的Go语言学习之旅吧!人生苦短,let’s Go.

 


图片

图片

什么是指针
我们都知道,程序运行时的数据是存放在内存中的,每一个存储在内存中的数据都有一个编号,这个编号就是内存地址。我们可以根据这个内存地址来找到内存中存储的数据,而内存地址可以被赋值给一个指针。我们也可以简单的理解为指针就是内存地址。

指针的声明和定义
在Go语言中,获取一个指针,直接使用取地址符&就可以。
示例:

func main() {
  name := "Go语言圈"
  nameP := &name //取地址
  fmt.Println("name变量值为:", name)
  fmt.Println("name变量的内存地址为:", nameP)
}
//运行结果:
//name变量值为:Go语言圈
//name变量的内存地址为: 0xc00004e240

nameP 指针的类型是 string Go语言中,类型名表示一个对应的指针类型

图片

从上面表格可以看到:

  • 普通变量 name 的值是Go语言圈,存放在内存地址为 0xc00004e240 的内存中

  • 指针变量 namep 的值是普通变量的内存地址 0xc00004e240

  • 指针变量 nameP 的值存放在 内存地址为 0xc00004e360 的内存中

  • 普通变量存的是数据,指针变量存的是数据的地址

var 关键字声明
我们也可以使用 var 关键字声明

var nameP *string
nameP = &name

new 函数声明

nameP := new(string)
nameP = &name

可以传递类型给这个内置的 new 函数,它会返回对应的指针类型。

 

指针的操作
这里强调一下:

指针变量是一个变量,这个变量的值是指针(内存地址)!
指针变量是一个变量,这个变量的值是指针(内存地址)!
指针变量是一个变量,这个变量的值是指针(内存地址)!

获取指针指向的值:
只需要在指针变量钱加 * 号即可获得指针变量值所对应的数据:

nameV := *nameP
fmt.Println("nameP指针指向的值为:",nameV) //nameP指针指向的值为: Go语言圈

修改指针指向的值:

*nameP = "公众号:Go语言圈" //修改指针指向的值
fmt.Println("nameP指针指向的值为:",*nameP)
fmt.Println("name变量的值为:",name)
//运行结果:
//nameP指针指向的值为: 公众号:Go语言圈
//name变量的值为: 公众号:Go语言圈
  • 我们发现nameP 指针指向的值被改变了,变量 name 的值也被改变了

  • 因为变量 name 存储数据的内存就是指针 nameP 指向的内存,这块内存被 nameP 修改后,变量 name 的值也被修改了。

通过 var 关键字直接定义的指针变量是不能进行赋值操作的,因为它的值为 nil,也就是还没有指向的内存地址

//错误示例
var intP *int
*intP = 10  //错误,应该先给分配一块内存,内存地址作为变量 intP 的值,这个内存就可以存放 10 了。

//应该使用
var intP *int  //声明int类型的指针变量 intP
intP = new(int) // 给指针分配一块内存
*intP = 66 
fmt.Println(":::",intP)  //::: 0xc0000ac088
fmt.Println(*intP) //66
//简短写法
var intP := new(int)
*intP=66

指针参数
当给一个函数使用指针作为参数的时候,就可以在函数中,通过形参改变实参的值:

func main() {
    name := "疯子"
    modify(&name)
    fmt.Println("name的值为:",name)
}
func modify(name *string)  {
    *name = "wucs"
}
//运行结果:
//name的值为: wucs

指针接收者

  • 如果接收者类型是 map、slice、channel 这类引用类型,不使用指针;

  • 如果需要修改接收者,那么需要使用指针;

  • 如果接收者是比较大的类型,可以考虑使用指针,因为内存拷贝廉价,所以效率高。

普通指针
和C语言一样, 允许用一个变量来存放其它变量的地址, 这种专门用于存储其它变量地址的变量, 我们称之为指针变量.

和C语言一样, Go语言中的指针无论是什么类型占用内存都一样(32位4个字节, 64位8个字节)

package main

import (
    "fmt"
    "unsafe"
)

func main() {

    var p1 *int;
    var p2 *float64;
    var p3 *bool;
    fmt.Println(unsafe.Sizeof(p1)) // 8
    fmt.Println(unsafe.Sizeof(p2)) // 8
    fmt.Println(unsafe.Sizeof(p3)) // 8
}

和C语言一样, 只要一个指针变量保存了另一个变量对应的内存地址, 那么就可以通过*来访问指针变量指向的存储空间

package main

import (
    "fmt"
)

func main() {

    // 1.定义一个普通变量
    var num int = 666
    // 2.定义一个指针变量
    var p *int = &num
    fmt.Printf("%p\n", &num) // 0xc042064080
    fmt.Printf("%p\n", p) // 0xc042064080
    fmt.Printf("%T\n", p) // *int
    // 3.通过指针变量操作指向的存储空间
    *p = 888
    // 4.指针变量操作的就是指向变量的存储空间
    fmt.Println(num) // 888
    fmt.Println(*p) // 888
}

指向数组指针
在Go语言中通过数组名无法直接获取数组的内存地址

package main
import "fmt"

func main() {
    var arr [3]int = [3]int{1, 3, 5}
    fmt.Printf("%p\n", arr) // 乱七八糟东西
    fmt.Printf("%p\n", &arr) // 0xc0420620a0
    fmt.Printf("%p\n", &arr[0]) // 0xc0420620a0
}

在Go语言中, 因为只有数据类型一模一样才能赋值, 所以只能通过&数组名赋值给指针变量, 才代表指针变量指向了这个数组

package main

import "fmt"

func main() {
    // 1.错误, 在Go语言中必须类型一模一样才能赋值
    // arr类型是[3]int, p1的类型是*[3]int
    var p1 *[3]int
    fmt.Printf("%T\n", arr)
    fmt.Printf("%T\n", p1)
    p1 = arr // 报错
    p1[1] = 6
    fmt.Println(arr[1])

    // 2.正确, &arr的类型是*[3]int, p2的类型也是*[3]int
    var p2 *[3]int
    fmt.Printf("%T\n", &arr)
    fmt.Printf("%T\n", p2)
    p2 = &arr
    p2[1] = 6
    fmt.Println(arr[1])

    // 3.错误, &arr[0]的类型是*int, p3的类型也是*[3]int
    var p3 *[3]int
    fmt.Printf("%T\n", &arr[0])
    fmt.Printf("%T\n", p3)
    p3 = &arr[0] // 报错
    p3[1] = 6
    fmt.Println(arr[1])
}

注意点:
Go语言中的指针, 不支持C语言中的+1 -1和++ – 操作

package main

import "fmt"

func main() {


    var arr [3]int = [3]int{1, 3, 5}
    var p *[3]int
    p = &arr
    fmt.Printf("%p\n", &arr) // 0xc0420620a0
    fmt.Printf("%p\n", p) // 0xc0420620a0
    fmt.Println(&arr) // &[1 3 5]
    fmt.Println(p) // &[1 3 5]
    // 指针指向数组之后操作数组的几种方式
    // 1.直接通过数组名操作
    arr[1] = 6
    fmt.Println(arr[1])
    // 2.通过指针间接操作
    (*p)[1] = 7
    fmt.Println((*p)[1])
    fmt.Println(arr[1])
    // 3.通过指针间接操作
    p[1] = 8
    fmt.Println(p[1])
    fmt.Println(arr[1])

    // 注意点: Go语言中的指针, 不支持+1 -1和++ --操作
    *(p + 1) = 9 // 报错
    fmt.Println(*p++) // 报错
    fmt.Println(arr[1])
}

指向切片的指针
值得注意点的是切片的本质就是一个指针指向数组, 所以指向切片的指针是一个二级指针

package main

import "fmt"

func main() {
    // 1.定义一个切片
    var sce[]int = []int{1, 3, 5}
    // 2.打印切片的地址
    // 切片变量中保存的地址, 也就是指向的那个数组的地址 sce = 0xc0420620a0
    fmt.Printf("sce = %p\n",sce )
    fmt.Println(sce) // [1 3 5]
    // 切片变量自己的地址, &sce = 0xc04205e3e0
    fmt.Printf("&sce = %p\n",&sce )
    fmt.Println(&sce) // &[1 3 5]
    // 3.定义一个指向切片的指针
    var p *[]int
    // 因为必须类型一致才能赋值, 所以将切片变量自己的地址给了指针
    p = &sce
    // 4.打印指针保存的地址
    // 直接打印p打印出来的是保存的切片变量的地址 p = 0xc04205e3e0
    fmt.Printf("p = %p\n", p)
    fmt.Println(p) // &[1 3 5]
    // 打印*p打印出来的是切片变量保存的地址, 也就是数组的地址 *p = 0xc0420620a0
    fmt.Printf("*p = %p\n", *p)
    fmt.Println(*p) // [1 3 5]

    // 5.修改切片的值
    // 通过*p找到切片变量指向的存储空间(数组), 然后修改数组中保存的数据
    (*p)[1] = 666
    fmt.Println(sce[1])
}

指向字典指针
与普通指针并无差异

package main
import "fmt"
func main() {

    var dict map[string]string = map[string]string{"name":"lnj", "age":"33"}
    var p *map[string]string = &dict
    (*p)["name"] = "zs"
    fmt.Println(dict)
}

指向结构体指针
Go语言中指向结构体的指针和C语言一样
结构体和指针
创建结构体指针变量有两种方式

package main
import "fmt"
type Student struct {
    name string
    age int
}
func main() {
  // 创建时利用取地址符号获取结构体变量地址
  var p1 = &Student{"lnj", 33}
  fmt.Println(p1) // &{lnj 33}

  // 通过new内置函数传入数据类型创建
  // 内部会创建一个空的结构体变量, 然后返回这个结构体变量的地址
  var p2 = new(Student)
  fmt.Println(p2) // &{ 0}
}

利用结构体指针操作结构体属性

package main
import "fmt"
type Student struct {
    name string
    age int
}
func main() {
  var p = &Student{}
  // 方式一: 传统方式操作
  // 修改结构体中某个属性对应的值
  // 注意: 由于.运算符优先级比*高, 所以一定要加上()
  (*p).name = "lnj"
  // 获取结构体中某个属性对应的值
  fmt.Println((*p).name) // lnj

  // 方式二: 通过Go语法糖操作
  // Go语言作者为了程序员使用起来更加方便, 在操作指向结构体的指针时可以像操作接头体变量一样通过.来操作
  // 编译时底层会自动转发为(*p).age方式
  p.age = 33
  fmt.Println(p.age) // 33
}

什么情况下使用指针

  • 不要对 map、slice、channel 这类引用类型使用指针;

  • 如果需要修改方法接收者内部的数据或者状态时,需要使用指针;

  • 如果需要修改参数的值或者内部数据时,也需要使用指针类型的参数;

  • 如果是比较大的结构体,每次参数传递或者调用方法都要内存拷贝,内存占用多,这时候可以考虑使用指针;

  • 像 int、bool 这样的小数据类型没必要使用指针;

  • 如果需要并发安全,则尽可能地不要使用指针,使用指针一定要保证并发安全;

  • 指针最好不要嵌套,也就是不要使用一个指向指针的指针,虽然 Go 语言允许这么做,但是这会使你的代码变得异常复杂。

 

 

 

文章链接:https://www.jb51.net/article/233949.htm

更多相关Go语言的技术文章或视频教程,请关注本公众号获取并查看,感谢你的支持与信任!

学Go语言哪些事儿 · 目录
上一篇有些人不知道,Go语言也可以实现 并发爬虫的
阅读 221
 
写下你的留言
 
 
 
 
 
 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/723601.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

手把手教你搭建Docker私有仓库Harbor

1、什么是Docker私有仓库 Docker私有仓库是用于存储和管理Docker镜像的私有存储库。Docker默认会有一个公共的仓库Docker Hub,而与Docker Hub不同,私有仓库是受限访问的,只有授权用户才能够上传、下载和管理其中的镜像。这种私有仓库可以部署在本地云环境中,用于组织内部开…

TimerWheel(计时轮)在Rust中的实现及源码解析

TimerWheel算法通过其独特的数据结构和运行原理,实现了高效、可扩展且灵活的定时任务管理。该结构用于对高性能的定时器框架,尤其密集程度越高的定时器效率越高。计时器轮(TimerWheel),模拟时钟格式组成的高效计时器 TimerWheel算法原理环形数据结构:TimerWheel,即时间轮…

记一次 .NET某游戏币自助机后端 内存暴涨分析

一:背景 1. 讲故事 前些天有位朋友找到我,说他们的程序内存会偶发性暴涨,自己分析了下是非托管内存问题,让我帮忙看下怎么回事?哈哈,看到这个dump我还是非常有兴趣的,居然还有这种游戏币自助机类型的程序,下次去大玩家看看他们出币的机器后端是不是C#写的?由于dump是l…

腾讯公益赛个人冲刺博客19(2024.6.7)

今天测试成功,可以实现ai聊天

腾讯公益赛个人冲刺博客20(2024.6.10)

今天总结第二阶段成果,对于新加的录像功能和帮扶功能的接收板块进行系统综合测试。

Android自动化-如何获取视图元素属性?

在做Android自动化时候,我们需要知道视图有哪些元素,元素都有哪些属性,获取到属性我们才能获取到元素从而做自动化控制,所以做Android自动化获取元素属性是必要的第一步 获取视图元素属性最便捷的方式就是使用Android SDK中的 uiautomatorviewer,当你配置好Android的开发环…

SAP: SALV GRID 追加按钮

利用类追加刷新按钮创建一个利用控制器的SALV程序。SAP: ABAP SALV GRID 追加按钮 , 利用类追加刷新按钮创建一个利用控制器的SALV程序。 异常的描述: 运行时错误: UNCAUGHT_EXCEPTION 优质生活从拆开始

[转帖]Linux 最新SO_REUSEPORT特性

https://oms.inspur.com/cwbase/web/gsprtf/main.aspx 1、前言昨天总结了一下Linux下网络编程“惊群”现象,给出Nginx处理惊群的方法,使用互斥锁。为例发挥多核的优势,目前常见的网络编程模型就是多进程或多线程,根据accpet的位置,分为如下场景:(1)单进程或线程创建soc…

Android自动化无障碍服务开源库-Assists v3.0.0

Assists v3.0.0 Android无障碍服务(AccessibilityService)开发框架,快速开发复杂自动化任务、远程协助、监听等Android无障碍服务能做什么 利用Android无障碍服务可以开发一些Android系统内的自动化任务,比如经典的微信自动抢红包、支付宝蚂蚁森林自动浇水、芭芭农场自动施…

dotnet 如何访问到 UNO 框架里面的 internal 不公开成员

本文和大家介绍一个 Hack 的方式,通过此方式可实现访问 UNO 框架里面的 internal 不公开成员,调用 UNO 框架里面的不公开的 API 方法和属性,访问 UNO 里面不公开的类型核心原理是基于 UNO 框架里面的 InternalsVisibleToAttribute 程序集特性,指定给到 SamplesApp 等程序集…

codesandbox 使用记录

1 登录2 导入github项目3 配置虚拟机和系统环境 3.1 配置虚拟机配置3.2 根据项目开发语言版本配置系统环境模板最后点击apply启动项目,等待依赖安装完成,即可在csbox运行自己的项目啦

大语言模型越狱, 你未曾想到的全新方法

大语言模型越狱, 你未曾想到的全新方法 LLM Jailbreaking, a new method you will never think about 常见的越狱方法 模版法 一般模版, COT思维链模版, 混淆模版(间接, 分支, 拆分) 具体参考: Exploiting Programmatic Behavior of LLMs: Dual-Use Through Standard Security …