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

Go语言之路————指针、结构体、方法

Go语言之路————指针、结构体、方法

  • 前言
  • 指针
  • 结构体
    • 声明
    • 初始化
    • 使用
    • 组合引用
    • 结构体和指针
    • 结构体的标签
  • 方法
    • 例子
    • 结合结构体
    • 总结

前言

  • 我是一名多年Java开发人员,因为工作需要现在要学习go语言,Go语言之路是一个系列,记录着我从0开始接触Go,到后面能正常完成工作上的业务开发的过程,如果你也是个小白或者转Go语言的,希望我这篇文章对你有所帮助。
  • 有关go其他基础的内容的文章大家可以查看我的主页,接下来主要就是把这个系列更完,更完之后我会在每篇文章中挂上连接,方便大家跳转和复习。

go中的指针,通常在结构体中用的特别多,而方法又是结构体一部分,所以我把这三个知识点放在一起来说,这样大家可以连贯起来方便理解和吸收。

指针

指针你只需要记住两个操作符,一个是取地址符&,另一个是解引用符 *。对一个变量进行取地址,会返回对应类型的指针,下面我简单举个例子:
我们先看取地址符号:&

package mainimport "fmt"func main() {a := 1fmt.Printf("%T\n", a)b := &afmt.Println(b)fmt.Printf("%T\n", b)
}console打印:
int
*int
0xc00008c0a8

我们对变量a用&符号取地址得到变量b,打印出来b的值就是a的地址,打印出b的类型,就是一个指针,这时候再回顾一下上面这句话:对一个变量进行取地址,会返回对应类型的指针。指针b存储的是变量a的地址

我们再看看解引用符:*
解引用符第一个用处,就跟它的命名一样,解除引用,就是解除指针的引用而获得具体的值,下面我们看个例子:

import "fmt"func main() {a := 123b := &aresult := *bfmt.Println(result)fmt.Printf("%T", result)
}console打印:
123
int

通过这个代码可以看到,我们通过&取得了指向a地址的指针b,但是通过解引符号*,用*b就可以解除指针引用直接获得这个地址对应的值,也就是a的值,打印result的数据类型也是int类型。

解引用符第一个用处:声明一个变量的类型为指针类型。
这里我们指定一个变量a它的类型为int类型的指针

var a *int

打印一下a看看呢

<nil>

因为我们没有给a赋值或者初始化,所以打印出来的为nil,要么使用取地址符将其他变量的地址赋值给该指针,要不就使用内置函数:new
说到:new。go中的new和Java中的new有区别的是,go中的new是专门为指针服务的,它的用处就是新建或者说初始化一个指针
看看代码

func main() {var a *intfmt.Println(a)a = new(int)fmt.Println(a)
}

我们用new去初始化了a,看看输出呢:

<nil>
0xc00000a120

为啥输出来是一个地址呢,因为该函数会为该指针分配内存,并且指针指向对应类型的零值
用上面的知识点,接引符*来验证下是不是零值呢:

func main() {var a *intfmt.Println(a)a = new(int)fmt.Println(a)fmt.Println(*a)
}
console打印:
<nil>
0xc00000a120
0

这么一看还真是对的,我们点进new函数,看看源码怎么写的:

func new(Type) *Type

通过代码分析,我们定义的a为int,这里new中传入的是int,那么返回的就是int,正好和a类型一致,是不是就是初始化了啊,是不是很简单啊。
而上面的示例代码,我们一般使用短赋值,简单一点:

func main() {a:=new(int)fmt.Println(a)
}

ps:在go中指针是不能运算的,而且这里我们还要区分一下new和make,前者是为指针服务器的,后者是为具体数据类型的值服务的,不要搞混了。

结构体

go中的结构体,你可以理解为Java中的实体类,但是他们又有细微的差别,但是不是很多,下面我就一一道来。
既然是结构体,那么定义它的关键词就是:struct。我们先通过一个例子简单看下。
定义一个UserInfo的结构体,里面分别有name、age、phone三个字段:

声明

type UserInfo struct {name  stringage   intphone int
}

这是一个简单的声明,跟函数一样,如果遇到相同的数据类型,也可以写一起,所以上面的age和phone可以这样写:

type UserInfo struct {name       stringage, phone int
}

初始化

注意,上面只是声明,在Java中,是以new关键词创建一个类,比如这里:new UserInfo(),但是在go中没有那么复杂,直接调用传参,看下面例子:

type UserInfo struct {name       stringage, phone int
}func main() {//这里需要注意点,为了方便阅读,或者灵活传参,这里尽量用这种格式字段名称:字段值,//也可以省略字段名称,但是就要传所有参数并且可读性很差,不推荐。var user = UserInfo{name:  "John",age:   42,phone: 1000,}fmt.Println(user)
}console打印:
{John 42 1000}

注意:这里的结构体命名和里面字段的命名,都遵循首字母大小写的规则,可能有同学忘了,这里提一下,go中首字母大写的方法就是public,小写的就是private,切记。

使用

我们访问和结构体和修改结构体中的值也很简单,直接用.就行:
获取值:

func main() {var user = UserInfo{name:  "John",age:   42,phone: 1000,}fmt.Println(user.name)fmt.Println(user.age)
}打印:
John
42Process finished with the exit code 0

赋值或修改值:

func main() {var user = UserInfo{name:  "John",age:   42,phone: 1000,}user.name = "一颗知足的心"user.age = 18fmt.Println(user.name)fmt.Println(user.age)
}打印:
一颗知足的心
18Process finished with the exit code 0

如果实例化过程比较复杂,可以编写一个函数来实例化结构体,就像下面这样,你也可以把它理解为一个构造函数,但是go中函数不能重载,所以你想像Java那样通过参数不同用多个一样的函数名是不行的。

func main() {user := NewUser("一颗知足的心", 18, 9527)fmt.Println(user)
}func NewUser(name string, age int, phone int) *UserInfo {return &UserInfo{name: name, age: age, phone: phone}
}

组合引用

和Java一样,直接在内部字段声明就行,请看下面例子:

type Person struct {name stringage  int
}type Student struct {p      Personschool string
}

看看使用:

student := Student{p:      Person{name: "jack", age: 18},school: "lili school",
}
fmt.Println(student.p.name)

结构体和指针

结构体的指针和值类型的指针使用上有个小的区别,就是结构体指针在使用的时候不用解引,请看下面例子:

type UserInfo struct {name       stringage, phone int
}func main() {user := &UserInfo{name:  "一颗知足的心",age:   18,phone: 9527,}fmt.Println(user.name)
}

可以看到我们直接用user.name就可以调用,和普通的结构体调用一样,因为这是go的语法糖,编译器会自动编译成(*user).name

结构体的标签

这里简单提一点,了解一下就行,标签就是在结构体定义字段的时候,在后面打上标签

type UserInfo struct {Name string `json:"name"`Age  int    `yaml:"age"`
}

结构体标签最广泛的应用就是在各种序列化格式中的别名定义,标签的使用需要结合反射才能完整发挥出其功能。

方法

方法与函数的区别在于,方法拥有接收者,而函数没有,且只有自定义类型能够拥有方法。先来看一个例子。

例子

type IntSlice []intfunc (i IntSlice) Get(index int) int {return i[index]
}
func (i IntSlice) Set(index, val int) {i[index] = val
}func (i IntSlice) Len() int {return len(i)
}

先声明了一个类型IntSlice,其底层类型为[]int,再声明了三个方法Get,Set和Len,方法的长相与函数并无太大的区别,只是多了一小段(i IntSlice) 。i就是接收者,IntSlice就是接收者的类型,接收者就类似于其他语言中的this或self,只不过在 Go 中需要显示的指明。

func main() {var intSlice IntSliceintSlice = []int{1, 2, 3, 4, 5}fmt.Println(intSlice.Get(0))intSlice.Set(0, 2)fmt.Println(intSlice)fmt.Println(intSlice.Len())
}

结合结构体

根据上面的例子,我们把方法和结构体结合一下。这里补充一点,接收者也分两种类型,值接收者和指针接收者
我们先看值接受者:

type UserInfo struct {name       stringage, phone int
}func main() {user := &UserInfo{name:  "一颗知足的心",age:   18,phone: 9527,}user.updateAge(20)fmt.Println(user.age)
}func (receiver UserInfo) updateAge(age int) {receiver.age = age
}console打印:
18Process finished with the exit code 0

我们可以看到,虽然我们在代码中,将age改为了20,但是最后user结构体中还是18,也就是说值接收者的方法,并不能改变接收者本身的属性
那要改变接收者本身的属性,就到了指针接收者,我们还是直接看代码:

func main() {user := &UserInfo{name:  "一颗知足的心",age:   18,phone: 9527,}user.updateAge(20)fmt.Println(user.age)
}func (receiver *UserInfo) updateAge(age int) {receiver.age = age
}console打印:
20Process finished with the exit code 0

看到变化没,我们只是把receiver UserInfo改为receiver *UserInfo,变成指针接受者,就可以改变接收者本身的属性。
这是为什么呢:因为值接收者可以简单的看成一个形参,而修改一个形参的值,并不会对方法外的值造成任何影响而用指针接收者,Go 会将其解释为(&receiver).age = age。所以方法的接收者为指针时,不管调用者是不是指针,都可以修改内部的值

总结

函数的参数传递过程中,是值拷贝的,如果传递的是一个整型,那就拷贝这个整型,如果是一个切片,那就拷贝这个切片,但如果是一个指针,就只需要拷贝这个指针,显然传递一个指针比起传递一个切片所消耗的资源更小,接收者也不例外,值接收者和指针接收者也是同样的道理。在大多数情况下,都推荐使用指针接收者,不过两者并不应该混合使用,要么都用,要么就都不用

相关文章:

  • Python 基础核心知识
  • (done) 吴恩达版提示词工程 6. 转换 (翻译,通用翻译,语气风格变换,文本格式转换,拼写检查和语法检查)
  • javaWeb开发---前后端开发全景图解(基础梳理 + 技术体系)
  • 2025-4-25 情绪周期视角复盘(mini)
  • view、reshape、resize 的区别
  • 简单的 shell 程序
  • 前端-介绍一个好用的波浪背景生成器
  • LeetCode热题100--438.找到字符串中所有字母异位词--中等
  • 参数规模:衡量大语言模型体量的标尺
  • 互联网的下一代脉搏:深入理解 QUIC 协议
  • iterm2 使用 zmodem(lrzsz)传输文件
  • 大模型——Spring.new快速构建AI驱动的定制化商业应用
  • Linux系统编程 day11 锁 (两天没有更新了,中期完就休息了)
  • 开关电源实战(六)ADDC反激电源
  • 【MySQL数据库】函数操作
  • PH热榜 | 2025-04-27
  • 论文速报《ChatBEV:理解BEV地图的视觉语言模型新突破》
  • H5实现一个二维码生成器页面
  • 【MySQL】Java代码操作MySQL数据库 —— JDBC编程
  • 接口测试详解
  • 马上评|演唱会云集,上海如何把“流量”变“留量”
  • 首映|《人生开门红》:段子背后都是案子
  • 银川市市长信箱被指已读乱回,官方回应
  • 人民日报读者点题:规范涉企执法,怎样防止问题反弹、提振企业信心?
  • 人民日报任仲平:为什么中国意味着确定性、未来性、机遇性
  • 铜钴巨头洛阳钼业一季度净利润同比大增九成,最新宣布首度进军黄金矿产