当前位置: 首页 > news >正文

go 里面的指针

指针

在 Go 中,指针(pointer)是一个变量的内存地址,就像 C 语言那样:

a := 10
p := &a  // p 是一个指向 a 的指针
fmt.Println(*p) // 输出 10,通过指针解引用

• &a 表示获取变量 a 的地址

  • p 表示通过地址访问对应的值
特性C 语言Go 语言
指针声明int *pvar p *int
指针的指针✅ 支持如 int **pp❌ 不推荐/不常用(语法上不直接暴露)
指针运算✅ 可对指针加减偏移❌ 不允许指针运算
内存分配malloc, freenew, make, GC 自动管理
野指针问题⚠️ 常见✅ 几乎不存在(有 GC)

🚫 Go 不支持的指针特性(与 C 相比)

  • ❌ 不支持指针算术(如 p + 1)
  • ❌ 不支持指针的指针的广泛操作(虽然语法允许 **T,但极少使用)
  • ❌ 不能随便转换指针类型(除非用 unsafe 包)

应用场景

  1. 函数传参避免拷贝(提升性能)
func modify(x *int) {*x = 100
}a := 5
modify(&a)
fmt.Println(a) // 100
  1. 结构体指针
type Person struct {Name string
}func changeName(p *Person) {p.Name = "Alice">>>>> 等价于(*p).Name = "Alice"
}

在 Go 中,如果你有一个指向结构体的指针 p *Person,Go 会自动帮你解引用:

type Person struct {Name string
}func changeName(p *Person) {// 实际上这是 (*p).Name = "Alice"p.Name = "Alice"
}func main() {person := Person{Name: "Bob"}changeName(&person)fmt.Println(person.Name) // 输出:Alice
}

自动解引用访问结构体字段,是 Go 的语法特性,目的是让代码更简洁、易懂

3.切片、map、channel 等内置类型是“引用类型”,无需显式指针

Go 的指针机制和 C 类似,但更安全,限制更多,没有“指针的指针”这种复杂操作,也不支持指针算术,更加鼓励值语义 + 显式传指针来控制性能和可变性。

Go 是值传递(pass-by-value)的语言,但可以传递“指针值”

*func modify(x int)

这个函数的参数 x 是一个“指向 int 的指针”,也就是说它接收的是一个 *int 类型的值(本质上是内存地址)。

modify(&a)

a := 5
modify(&a) // &a 是变量 a 的内存地址(类型是 *int)>>>>>modify(&a) 是把变量 a 的地址作为值传进函数 modify(x *int),
>>>>>函数通过指针 x 修改了原变量的值。
  • &a:取变量 a 的内存地址,类型是 *int
  • modify(&a):把这个地址传给函数 modify 的参数 x

传进去的是 a 的地址(而不是 a 的值),但这个地址本身是按值传入的,就像传入一个 int、string、float64 一样。

表达式意义类型说明
a变量本身int值是 5
&aa 的地址*int值是内存地址,例如 0x1400012fc08
x函数参数,接收到 &a*int是地址
*x取出地址指向的内容int是 a 的值
*x = 100改变地址指向的值把 a 改为 100
// a 是个箱子,里面装了 5
a := 5// &a 是这个箱子的编号(地址),我们把编号给了 modify 函数
modify(&a)// modify 函数通过编号找到了箱子,把里面的 5 改成了 100
方式描述会修改原变量?
值传递复制了一份值(浅拷贝)❌ 否
指针传递传递的是地址本身✅ 是
深拷贝创建了新的独立值(递归复制所有内容)❌(除非修改返回值)
误区真相
Go 是引用传递❌ Go 是值传递,包括传指针也是“传值传地址”
new 和 make 一样❌ new 分配内存,make 初始化引用类型
指针越多越高效❌ 指针会导致逃逸,频繁使用反而性能差
指针不能用于方法接收者❌ 指针接收者用于修改对象、避免复制结构体
http://www.xdnf.cn/news/946171.html

相关文章:

  • [蓝桥杯 2024 国 Java B] 美丽区间
  • pymilvus
  • VRFF: Video Registration and FusionFramework 论文详解
  • 启动已有小程序项目
  • 详解K8s 1.33原地扩缩容功能:原理、实践、局限与发展
  • 【K8S】Kubernetes从入门到实战:全面指南
  • 云原生K8s+Docker+KubeSphere+DevOps
  • K8S认证|CKS题库+答案| 9. 网络策略 NetworkPolicy
  • 上位机开发过程中的设计模式体会(1):工厂方法模式、单例模式和生成器模式
  • AspectJ 在 Android 中的完整使用指南
  • 博睿数据×华为, 共筑智慧金融新未来
  • UE5 学习系列(一)创建一个游戏工程
  • 机器学习监督学习实战六:五种算法对新闻组英文文档进行文本分类(20类),词频统计和TF-IDF 转换特征提取方法理论和对比解析
  • 【安全篇】金刚不坏之身:整合 Spring Security + JWT 实现无状态认证与授权
  • 让 Kubernetes (K8s) 集群 使用 GPU
  • 阿里云Ubuntu 22.04 64位搭建Flask流程(亲测)
  • k8s从入门到放弃之Service负载均衡
  • AWS vs 阿里云:功能、服务与性能对比指南
  • 轻量级关键点 blaze pose 2025
  • SpringCloud优势
  • 软件工程教学评价
  • CentOS 7 部署 Samba 使用虚拟用户笔记
  • centos 7 部署awstats 网站访问检测
  • GO语言---init函数
  • Vue 3 实战:【加强版】公司通知推送(WebSocket + token 校验 + 心跳机制)
  • 软件工程 期末复习
  • 【Linux开发】Hi3516dv300-DC-182型开发板显示开机logo的相关操作
  • 统计学(第8版)——假设检验学习笔记(考试用)
  • 阿里云服务器 篇十七:网站悬浮球
  • Jenkins构建时出现报错`ERROR: Failed to install JDK. Exit code=2`的终极解决方案