Golang中&和*运算符的真实作用机制

发布时间 - 2026-01-08 00:00:00    点击率:
&是取地址操作符,仅返回变量内存地址;在声明时为类型修饰符(如int),使用时为解引用运算符;二者必须配对使用才能实现通过指针修改原变量。

& 是取地址操作符,不是“声明指针”的语法

很多人误以为 & 是用来“定义指针类型”的,其实它只干一件事:返回变量在内存中的地址。它不创建新类型,也不改变变量本身,只是把那个地址值拿出来用。

  • &x 的结果是一个值(比如 0xc000014088),这个值的类型是 *int —— 注意,* 在这里属于类型签名的一部分,不是运算符
  • 你不能对常量或字面量取地址,比如 &42&"hello" 会编译报错:cannot take the address of …
  • 对函数返回值取地址也受限:只有当返回的是可寻址的变量(如结构体字段、切片元素、映射值的地址需配合 & + 变量名)才合法;直接写 &foo() 多数情况非法

* 有两种完全不同的语义:声明时是类型修饰符,使用时是解引用运算符

* 在 Go 中是“上下文敏感”的:出现在变量声明里,它是类型的一部分;出现在表达式中,它是在做解引用动作。这是新手最容易混淆的点。

  • 声明时:var p *int 中的 * 表示“p 是一个指向 int 的指针”,此时它和 int 构成完整类型 *int
  • 使用时:*p = 42 中的 * 是运算符,意思是“把 42 写进 p 所指向的那块内存”;fmt.Println(*p) 则是读取那块内存的当前值
  • 如果对空指针(nil)解引用,运行时 panic:panic: runtime error: invalid memory address or nil pointer dereference

函数传参时 & 和 * 必须配对使用,否则无法修改原变量

Go 默认是值传递,想让函数内部改动影响到调用方的变量,必须显式传地址,并在函数内解引用。漏掉任一环节都会失败。

func increment(p *int) {
    *p++ // ✅ 正确:通过指针修改原内存
}
func main() {
    x := 5
    increment(&x) // ✅ 正确:传 x 的地址
    fmt.Println(x) // 输出 6
// ❌ 错误示范:
// increment(x)      // 编译错误:cannot use x (type int) as type *int
// increment(&x + 1) // 无意义,&x + 1 不是合法地址

}

  • 传参用 &x,函数接收用 *int,函数体内操作用 *p —— 这三者缺一不可
  • 对大结构体传指针不只是为了“能改”,更是避免复制开销;但小类型(如 int, bool)传值反而更高效
  • 方法接收者用指针(func (p *Person) SetName(...))时,Go 会自动帮你补上 &*,但底层机制没变:仍是地址传递 + 解引用

&p 和 p 的区别:指针变量自己也有地址

初学者常以为 p 就是“地址”,其实 p 是一个变量,它存储着另一个地址;而 &p 是这个变量自身的地址 —— 它们层级不同,用途完全不同。

func main() {
    x := 42
    p := &x   // p 存的是 x 的地址
    fmt.Printf("p 的值(即 x 的地址): %p\n", p)     // e.g., 0xc000014088
    fmt.Printf("&p 的值(p 自己的地址): %p\n", &p) // e.g., 0xc000006028
    fmt.Printf("*p 的值(x 的内容): %d\n", *p)     // 42
}
  • p 是一个变量,占内存,有地址(&p),也有值(x 的地址)
  • *p 是间接访问,&p 是直接取址 —— 前者面向数据,后者面向变量本身
  • 这种嵌套关系在实现二级指针(**int)、或需要修改指针本身(比如重定向它指向别处)时才真正用得上

实际写代码时,最常被忽略的是:指针不是魔法,它只是个存地址的整数变量;&* 是配套的“出入接口”,一进一出,少一个就断链。理解这点,比背口诀管用得多。


# go  # golang  # ai  # 区别  # 编译错误  # 常量  # 运算符  # Error  # 结构体  # bool  # int  # 指针  # 接口  # 指针类型  # var  # 值传递  # pointer  # 空指针  # 切片  # nil  # 是一个  # 的是  # 也有  # 出现在  # 那块  # 它只  # 自己的  # 这是  # 是个 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: javascript中对象的定义、使用以及对象和原型链操作小结  如何快速配置高效服务器建站软件?  Laravel怎么实现模型属性的自动加密  IOS倒计时设置UIButton标题title的抖动问题  高端建站如何打造兼具美学与转化的品牌官网?  Laravel Facade的原理是什么_深入理解Laravel门面及其工作机制  打开php文件提示内存不足_怎么调整php内存限制【解决方案】  如何在 Pandas 中基于一列条件计算另一列的分组均值  MySQL查询结果复制到新表的方法(更新、插入)  高端网站建设与定制开发一站式解决方案 中企动力  EditPlus中的正则表达式 实战(1)  使用spring连接及操作mongodb3.0实例  黑客如何通过漏洞一步步攻陷网站服务器?  Windows10如何更改计算机工作组_Win10系统属性修改Workgroup  在线教育网站制作平台,山西立德教育官网?  Angular 表单中正确绑定输入值以确保提交与验证正常工作  如何在建站之星绑定自定义域名?  如何打造高效商业网站?建站目的决定转化率  浅谈redis在项目中的应用  安克发布新款氮化镓充电宝:体积缩小 30%,支持 200W 输出  如何在景安服务器上快速搭建个人网站?  laravel服务容器和依赖注入怎么理解_laravel服务容器与依赖注入解析  CSS3怎么给轮播图加过渡动画_transition加transform实现【技巧】  Laravel如何使用Gate和Policy进行权限控制_Laravel权限判定与策略规则配置  武汉网站设计制作公司,武汉有哪些比较大的同城网站或论坛,就是里面都是武汉人的?  Laravel怎么解决跨域问题_Laravel配置CORS跨域访问  网站页面设计需要考虑到这些问题  如何在万网利用已有域名快速建站?  jquery插件bootstrapValidator表单验证详解  香港服务器网站测试全流程:性能评估、SEO加载与移动适配优化  如何在VPS电脑上快速搭建网站?  JavaScript如何实现音频处理_Web Audio API如何工作?  如何在宝塔面板中创建新站点?  JavaScript常见的五种数组去重的方式  简历在线制作网站免费版,如何创建个人简历?  详解jQuery停止动画——stop()方法的使用  Laravel的.env文件有什么用_Laravel环境变量配置与管理详解  浏览器如何快速切换搜索引擎_在地址栏使用不同搜索引擎【搜索】  如何在云服务器上快速搭建个人网站?  南京网站制作费用,南京远驱官方网站?  Laravel如何实现密码重置功能_Laravel密码找回与重置流程  Laravel如何实现全文搜索_Laravel Scout集成Algolia或Meilisearch教程  HTML5段落标签p和br怎么选_文本排版常用标签对比【解答】  详解免费开源的DotNet二维码操作组件ThoughtWorks.QRCode(.NET组件介绍之四)  如何在腾讯云免费申请建站?  悟空识字如何进行跟读录音_悟空识字开启麦克风权限与录音  详解Nginx + Tomcat 反向代理 负载均衡 集群 部署指南  如何在浏览器中启用Flash_2025年继续使用Flash Player的方法【过时】  Laravel Docker环境搭建教程_Laravel Sail使用指南  如何选择PHP开源工具快速搭建网站?