一文带你速通go语言指针

简介: Go语言指针入门指南:简述指针用于提升效率,通过地址操作变量。文章作者sharkChili是Java/CSDN专家,维护Java Guide项目。文中介绍指针声明、取值,展示如何通过指针修改变量值及在函数中的应用。通过实例解析如何使用指针优化函数,以实现对原变量的直接修改。作者还邀请读者加入交流群深入探讨,并鼓励关注其公众号“写代码的SharkChili”。

写在文章开头

关于go语言的系列文章更新了有一段时间了,从阅读量来看大部分接触go语言的读者都是Java开发,因为Java这门语言没有指针的概念,所以笔者专门整理了一篇文章带读者快速了解一下指针的概念。

Hi,我是 sharkChili ,是个不断在硬核技术上作死的 java coder ,是 CSDN的博客专家 ,也是开源项目 Java Guide 的维护者之一,熟悉 Java 也会一点 Go ,偶尔也会在 C源码 边缘徘徊。写过很多有意思的技术博客,也还在研究并输出技术的路上,希望我的文章对你有帮助,非常欢迎你关注我的公众号: 写代码的SharkChili

因为近期收到很多读者的私信,所以也专门创建了一个交流群,感兴趣的读者可以通过上方的公众号获取笔者的联系方式完成好友添加,点击备注 “加群” 即可和笔者和笔者的朋友们进行深入交流。

go语言指针详解

简介

指针即指向变量的地址,在计算机宝贵的内存中存在成千上万的变量,对于某些可复用的变量,我们可以通过指针进行操作,避免拷贝的开销,从而提升系统的执行效率。

声明和取值

我们通过短变量声明了一个num变量,通过Println打印可知,加上取地址运算符的输出结果就是num的地址,而这种标识地址的变量在编程语言中我们都称之为指针:

package main

import "fmt"

func main() {
   
   
    num := 6
    //打印变量的值
    fmt.Println(num)
    //打印变量的地址
    fmt.Println(&num)
}

输出结果:

6
0xc0000a6058

我们继续说说指针类型,我们基于上述例子手动创建一个指针,语言很简单,即* type,例如下面这段代码,即声明一个整型的指针变量,通过取地址运算符将num的地址赋值给p,最后我们通过reflectTypeOf查看输出结果。

import (
    "fmt"
    "reflect"
)

func main() {
   
   
    num := 6
    //打印变量的值
    fmt.Println(num)
    //打印变量的地址(指向内存地址的值称为指针)
    fmt.Println(&num)
    //说明一个指向int的指针
    var p *int
    //分配一个int的指针
    p = &num
    fmt.Println("&num的类型:", reflect.TypeOf(&num))
    fmt.Println("p的类型:", reflect.TypeOf(p))

}

可以看到取地址运算符和p指针得到的类型都是整型指针类型:

6
0xc0000a6058           
&num的类型: *int       
p的类型: *int

既然我们拿到变量的地址空间,那么我们就可以打印指针所指向的值,如下所示,通过指针访问元素值的语法如下,即* variable意味获取p处的值。


    fmt.Println("p1处的值", *p1)

输出结果:

p1处的值 6

同理我们若要修改指针的值,也可以通过*号定位到p处的值,然后进行修改通过赋值运算值直接完成修改:

//修改p1处的值为18,仅改变值,地址不变
    *p1 = 18
    fmt.Println("p1处的值", *p1)
    fmt.Println("p1处的地址", p1)
    fmt.Println("num的地址", &num)

输出结果:

p1处的值 18            
p1处的地址 0xc0000a6058
num的地址 0xc0000a6058

可以看到无论是num还是p的地址都是不变的,我们改变的仅仅是地址空间的值:

函数指针

函数指针就是返回指针的函数,通过对这个函数的调用我们可以得到一个变量的指针,这意味着每个调用该函数得到的变量就是当前函数栈帧上的变量的指针。

var num = 6

func main() {
   
   
    pointer := createPointer()
    fmt.Println("pointer值", pointer)
    fmt.Println("pointer处的值", *pointer)
}

// createPointer 返回int类型的指针
func createPointer() *int {
   
   
    fmt.Println("num的地址:", &num)
    return &num
}

输出结果如下:

num的地址: 0x91e340
pointer值 0x91e340
pointer处的值 6

createPointer输出的num地址和main方法调用createPointer得到的变量的地址是一致的,这意味我们对于某些场景我们可以通过函数调用的方式或者公用变量的指针从而全局进行操作:

基于指针优化普通函数实践

我们现在有这样一段代码,我们希望给定一个布尔参数,通过函数negate设置为取反后的值,例如我们变量a为false,希望调用negate后a的值会变为true,原始代码如下:

func main() {
   
   

    truth := true
    negate(truth)
    fmt.Println(truth)

    lies := false
    negate(lies)
    fmt.Println(lies)
}

func negate(b bool) bool {
   
   
    return !b
}

输出结果可以看到,结果是不变的:

true
false

上述方法传入的是变量的值,即我们传入的参数都会直接拷贝到形参变量上,这使得形参无论如何修改都不会影响实参。

所以我改造时要按照如下步骤执行:

  1. 让函数得到变量指针。
  2. 通过指针访问实参地址。
  3. 取反并赋值。

最终代码如下:

import "fmt"

func main() {
   
   

    truth := true
    negate(&truth)
    fmt.Println(truth)

    lies := false
    negate(&lies)
    fmt.Println(lies)
}

func negate(b *bool) {
   
   
    *b = !(*b)
}

通过将指针传入函数,通过地址定位到变量的地址从而完成变量修改,最终我们完成了变量修改操作:

false
true

小结

以上便是笔者对于go语言指针的扫盲实践,本文笔者从指针的基本声明、实用、最佳实践多个角度的案例演示了指针的常见操作,希望对你有帮助。

Hi,我是 sharkChili ,是个不断在硬核技术上作死的 java coder ,是 CSDN的博客专家 ,也是开源项目 Java Guide 的维护者之一,熟悉 Java 也会一点 Go ,偶尔也会在 C源码 边缘徘徊。写过很多有意思的技术博客,也还在研究并输出技术的路上,希望我的文章对你有帮助,非常欢迎你关注我的公众号: 写代码的SharkChili

因为近期收到很多读者的私信,所以也专门创建了一个交流群,感兴趣的读者可以通过上方的公众号获取笔者的联系方式完成好友添加,点击备注 “加群” 即可和笔者和笔者的朋友们进行深入交流。

参考

Head First Go语言程序设计:https://book.douban.com/subject/35237045/

目录
相关文章
|
5天前
|
存储 JSON 监控
Viper,一个Go语言配置管理神器!
Viper 是一个功能强大的 Go 语言配置管理库,支持从多种来源读取配置,包括文件、环境变量、远程配置中心等。本文详细介绍了 Viper 的核心特性和使用方法,包括从本地 YAML 文件和 Consul 远程配置中心读取配置的示例。Viper 的多来源配置、动态配置和轻松集成特性使其成为管理复杂应用配置的理想选择。
23 2
|
4天前
|
Go 索引
go语言中的循环语句
【11月更文挑战第4天】
13 2
|
4天前
|
Go C++
go语言中的条件语句
【11月更文挑战第4天】
15 2
|
8天前
|
程序员 Go
go语言中的控制结构
【11月更文挑战第3天】
84 58
|
7天前
|
监控 Go API
Go语言在微服务架构中的应用实践
在微服务架构的浪潮中,Go语言以其简洁、高效和并发处理能力脱颖而出,成为构建微服务的理想选择。本文将探讨Go语言在微服务架构中的应用实践,包括Go语言的特性如何适应微服务架构的需求,以及在实际开发中如何利用Go语言的特性来提高服务的性能和可维护性。我们将通过一个具体的案例分析,展示Go语言在微服务开发中的优势,并讨论在实际应用中可能遇到的挑战和解决方案。
|
4天前
|
Go
go语言中的 跳转语句
【11月更文挑战第4天】
12 4
|
4天前
|
JSON 安全 Go
Go语言中使用JWT鉴权、Token刷新完整示例,拿去直接用!
本文介绍了如何在 Go 语言中使用 Gin 框架实现 JWT 用户认证和安全保护。JWT(JSON Web Token)是一种轻量、高效的认证与授权解决方案,特别适合微服务架构。文章详细讲解了 JWT 的基本概念、结构以及如何在 Gin 中生成、解析和刷新 JWT。通过示例代码,展示了如何在实际项目中应用 JWT,确保用户身份验证和数据安全。完整代码可在 GitHub 仓库中查看。
15 1
|
8天前
|
Go 数据处理 API
Go语言在微服务架构中的应用与优势
本文摘要采用问答形式,以期提供更直接的信息获取方式。 Q1: 为什么选择Go语言进行微服务开发? A1: Go语言的并发模型、简洁的语法和高效的编译速度使其成为微服务架构的理想选择。 Q2: Go语言在微服务架构中有哪些优势? A2: 主要优势包括高性能、高并发处理能力、简洁的代码和强大的标准库。 Q3: 文章将如何展示Go语言在微服务中的应用? A3: 通过对比其他语言和展示Go语言在实际项目中的应用案例,来说明其在微服务架构中的优势。
|
8天前
|
Go 数据处理 调度
探索Go语言的并发模型:Goroutines与Channels的协同工作
在现代编程语言中,Go语言以其独特的并发模型脱颖而出。本文将深入探讨Go语言中的Goroutines和Channels,这两种机制如何协同工作以实现高效的并发处理。我们将通过实际代码示例,展示如何在Go程序中创建和管理Goroutines,以及如何使用Channels进行Goroutines之间的通信。此外,本文还将讨论在使用这些并发工具时可能遇到的常见问题及其解决方案,旨在为Go语言开发者提供一个全面的并发编程指南。
|
5天前
|
Go 调度 开发者
探索Go语言中的并发模式:goroutine与channel
在本文中,我们将深入探讨Go语言中的核心并发特性——goroutine和channel。不同于传统的并发模型,Go语言的并发机制以其简洁性和高效性著称。本文将通过实际代码示例,展示如何利用goroutine实现轻量级的并发执行,以及如何通过channel安全地在goroutine之间传递数据。摘要部分将概述这些概念,并提示读者本文将提供哪些具体的技术洞见。