Go语言之结构体与方法

结构体是一系列属性的集合(类似于 Python 中的类)

1、结构体的定义与使用

// 定义
type Person struct {
   Name string
   Age int
   Sex string
}

func main() {
   // 使用
   var per Person
   per.Name="XiaoYang"
   fmt.Println(per)
}

2、定义并赋初值

type Person struct {
   Name string
   Age  int
   Sex  string
}

func main() {
   var per1 Person = Person{Name: "XiaoYang"}           // 按关键字传参,可以少传
   var per2 Person = Person{"Bob", 20, "男"}     // 按位置传参,全传

   fmt.Println(per1)    // 输出:{XiaoYang 0 }
   fmt.Println(per2)    // 输出:{Bob 20 男}
}

3、匿名结构体(只使用一次,没有名字)

// 定义个匿名结构体并实例化之后赋值给了 hobby 变量
hobby := struct {
   HobbyId   int
   HobbyName string
}{HobbyId: 1, HobbyName: "篮球"}

fmt.Println(hobby)      // 输出:{1 篮球}
fmt.Println(hobby.HobbyName)    // 输出:篮球

4、结构体的零值

定义好的结构体没有被初始化时,该结构体的字段将默认赋值为零值

也就是我属性的零值,所以他是值类型,参数传递,copy 传递,在函数中修改不会影响原来的

type Person struct {
   Name string
   Age  int
   Sex  string
}

func main() {
   var per Person=Person{"Bob", 20, "男"}

   fmt.Println(per) // 输出:{Bob 20 男}
   test(per)        // 输出:{Bob 20 男}
   fmt.Println(per) // 输出:{Bob 20 男}
}

func test(per Person)  {
   per.Age=20
   fmt.Println(per)
}

5、结构体的指针

// & 放在变量前,表示取该变量的地址
// * 放在类型前,表示指向该类型的指针(变量定义,指定类型时才会用到) *[3]int  和 [3]*int
// * 放在变量前,表示解引用(取出指针指向的具体的值)

type Person struct {
    Name string
    Age  int
    Sex  string
}

func main() {
    var per1 *Person
    fmt.Println(per1)   // 输出:      表示指针类型

    // 定义并初始化
    var per2 *Person = &Person{}
    fmt.Println(per2)   // 输出:&{ 0 }

    // 把per2的名字改成XiaoYang
    (*per2).Name = "XiaoYang"

    // 也支持直接使用
    per2.Name = "Bob"
    fmt.Println(per2)   // 输出:&{Bob 0 }

}

6、匿名字段(字段没有名字,只有类型)

可用于【变量提升 / 提升字段】类似于面向对象的继承

// 定义一个结构体,匿名字段类型就是字段名字,所有类型不能重复
type Person struct {
   string
   int
   Sex  string
}

func main() {
   per := Person{"XiaoYang", 20, "男"}   // 字段匿名,类型就是字段名
   fmt.Println(per)         // 输出:{XiaoYang 20 男}
   fmt.Println(per.string)  // 输出:XiaoYang
}

7、嵌套结构体(结构体中套结构体)

type Person struct {
   Name  string
   Age   int
   Sex   string
   Hobby Hobby       // Person中嵌套Hobby结构体字段
}

type Hobby struct {
   HobbyId   int
   HobbyName string
}

func main() {
   per := Person{Name:"XiaoYang", Age: 20, Sex: "男", Hobby: Hobby{1,"篮球"}}

   fmt.Println(per)     // 输出:{XiaoYang 20 男 {1 篮球}}
   fmt.Println(per.Name)    // 输出:XiaoYang
   fmt.Println(per.Hobby.HobbyName) // 输出:篮球
}

8、字段提升

结构体中有匿名的结构体类型字段,则该匿名结构体里的字段就称为提升字段,可以从外部直接访问

type Person struct {
   Name  string
   Age   int
   Sex   string
   Hobby     // Person中嵌套Hobby匿名结构体字段
}

type Hobby struct {
   HobbyId   int
   HobbyName string
}

func main() {
   per := Person{Name:"XiaoYang", Age: 20, Sex: "男", Hobby: Hobby{1,"篮球"}}

   // 打印爱好的名字(Hobby是一个匿名字段,会字段提升)
   fmt.Println(per.HobbyName)           // 输出:篮球

   // per.hobby 类似于面向对象中的super()
   fmt.Println(per.Hobby.HobbyName)     // 输出:篮球
}

// ------------------------------------------------------------------------------------

// 当子类和父类中的字段名一样时,就像面向对象的继承,子类继承父类(结构体嵌套,匿名字段),子类可以直接调用父类中的属性或方法

type Person struct {
   Name  string
   Age   int
   Sex   string
   Hobby     // Person中嵌套Hobby匿名字段
}

type Hobby struct {
   HobbyId   int
   Name      string
}

func main() {
   per := Person{Name:"XiaoYang", Age: 20, Sex: "男", Hobby: Hobby{1,"篮球"}}

   fmt.Println(per.Name)        // 输出:XiaoYang  ——>优先使用自己的
   fmt.Println(per.Hobby.Name)      // 输出:篮球       ——>指定打印Hobby的名字
}

9、结构体相等性

结构体是值类型。

如果它的每一个字段都是可比较的,则该结构体也是可以比较的。

如果两个结构体变量的对应字段相等,则两个变量也是相等的。

如果结构体包含不可比较的字段,则结构体变量也不可比较。

type Person struct {
   Name string
   Age  int
   Sex  string
   // 包含不可比较的字段
   Test []int   ——>这是引用类型不可比较
}

func main() {
   // 值类型可以直接==比较,引用类型只能跟nil用==比较
   per1 := Person{Name: "XiaoYang"}
   per2 := Person{Name: "XiaoYang"}
   per3 := Person{Name: "XiaoYang", Age: 20}

   fmt.Println(per1 == per2)  // 输出:ture    ——>包含不可比较类型都会直接报错
   fmt.Println(per1 == per3)  // 输出:false
}

方法就是一个特殊函数,在函数的基础上加了一些东西

func这个关键字和方法名中间加入一个特殊的接收器类型,接收器可以是结构体类型,也可以是非结构体类型

1、方法的定义和使用

type Person struct {
   Name string
   Age  int
   Sex  string
}

// 定义一个方法:给Person结构体绑定一个方法,oneself(名字随便取)类似于Python类中的self
func (oneself Person) printName()  {
   // 在方法内使用oneself
   fmt.Println(oneself.Name)
}

func main() {
   // 使用,对象调用方法
   per := Person{}
   per.Name = "XiaoYang"
   // 绑定给对象的方法
   per.printName()       // 输出:XiaoYang
}

2、有了函数为啥还需要方法?

方法功能都能实现,但是呢?它就能指定给某个对象了 。

type Person struct {
   Name string
   Age  int
   Sex  string
}

// 方法
func (oneself Person) printName() {
   fmt.Println(oneself.Name)
}

// 函数
func printName(oneself Person)  {
   fmt.Println(oneself.Name)
}

func main() {
   per := Person{Name: "XiaoYang"}

   per.printName()  // 方法的特殊之处,可以自动传递值
   printName(per)   // 函数需要手动传递值
}

3、指针接收器与值接收器

type Person struct {
   Name string
   Age  int
   Sex  string
}

// 值接收器修改名字
func (oneself Person) changeName(name string) {
   oneself.Name = name
}

// 指针接收器修改年龄
func (oneself *Person) changeAge(age int) {
   oneself.Age = age
}

func main() {
   per := Person{Name: "XiaoYang", Age: 20}
   fmt.Println(per)     // 输出:{XiaoYang 20 }

   per.changeName("Bob")// 由于这个是值接收器,它是copy一份传递过去所以修改的是copy的不会改掉原来的
   per.changeAge(18)    // 指针接收器,它传递的是指针

   fmt.Println(per)     // 输出:{XiaoYang 18 }
}

/*
什么时候用指针接收器,什么时候使用值接收器:
    -想改原来的,就用指针
    -不想改原来的,就用值
*/

5、匿名字段的方法(方法提升)

type Person struct {
   Name  string
   Age   int
   Sex   string
   Hobby // 匿名字段
}

type Hobby struct {
   Id   int
   Name string
}

// 打印Person的名字
func (oneself Person) printName() {
   fmt.Println(oneself.Name)
}

// 打印Hobby的名字
func (oneself Hobby) printHobbyName() {
   fmt.Println(oneself.Name)
}

// 打印Hobby的名字
func (oneself Hobby) printName() {
   fmt.Println(oneself.Name)
}

func main() {
   per := Person{Name: "XiaoYang", Hobby: Hobby{1, "篮球"}}
   per.printName()      // 输出:XiaoYang
   per.printHobbyName() // 输出:篮球

   // 如果方法重名了,优先使用结构体自己的
   per.printName()       // 输出:XiaoYang
   per.Hobby.printName() // 输出:篮球
}

6、在方法中使用值接收器 与 在函数中使用值参数

type Person struct {
    Name string
    Age  int
    Sex  string
}

// 在方法中使用值接收器
func (oneself Person) printName() {
    fmt.Println(oneself.Name)
}

// 在函数中使用值参数
func printName(oneself Person) {
    fmt.Println(oneself.Name)
}

func main() {
    per1 := &Person{Name: "XiaoYang"} // per1是个指针
    per2 := Person{Name: "Bob"}

    printName(*per1)    // 输出:XiaoYang
    per1.printName()    // 输出:XiaoYang      ———> 值收器:可以用值来调,也可以用指针来调
    per2.printName()    // 输出:Bob
}

7、在方法中使用指针接收器 与 在函数中使用指针参数

type Person struct {
   Name string
   Age  int
   Sex  string
}

// 在方法中使用指针接收器
func (oneself *Person) printName() {
   fmt.Println(oneself.Name)
}

func (oneself *Person)changeName(name string)  {
   oneself.Name=name

}

// 在函数中使用指针参数
func printName(oneself *Person) {
   fmt.Println(oneself.Name)
}

func main() {
   per1 := Person{Name: "XiaoYang"}
   per2 := &Person{Name: "Bob"}   // per1是个指针

   per1.printName()   // 值可以调用
   printName(&per1)

   per2.printName()   // 指针可以调用
   printName(per2)

   per1.changeName("Alen")    // 可以修改
   fmt.Println(per1)

   per2.changeName("YS")     // 可以修改
   fmt.Println(per2)
}

/*
总结:
    -不管是值类型接收器还是指针类型接收器,都可以用值来调用,或者指针来调用。
    -不管是值还是指针来调用,只要是值类型接收器,改的就是新的,只要是指针类型接收器,改的是原来的。
*/

8、非结构体上绑定方法

不允许在基础数据类型上绑定方法(如:int、string … )

但是自己定义的类型可以绑定方法

type Myint int

func (i *Myint) add() {
   (*i)++
}

func main() {
   var a Myint = 10
   fmt.Println(a)   // 输出:10
   a.add()
   a.add()
   a.add()
   fmt.Println(a)   // 输出:13
}

Original: https://www.cnblogs.com/XiaoYang-sir/p/15376845.html
Author: Mr-Yang`
Title: Go语言之结构体与方法

原创文章受到原创版权保护。转载请注明出处:https://www.johngo689.com/516568/

转载文章受原作者版权保护。转载请注明原作者出处!

(0)

大家都在看

  • Go切片全解析

    目录结构:数组切片 底层结构 创建 普通声明 make方式 截取 边界问题 追加 拓展表达式 扩容机制 切片传递的坑 切片的拷贝 浅拷贝 深拷贝 var n [4]int fmt….

    Go语言 2023年5月25日
    075
  • golang拾遗:自定义类型和方法集

    golang拾遗主要是用来记录一些遗忘了的、平时从没注意过的golang相关知识。 很久没更新了,我们先以一个谜题开头练练手: package main import ( &quo…

    Go语言 2023年5月25日
    052
  • Kubernetes容器编排探索与实践v1.22.1-上半部分

    概述 本人博客网站 IT小神; www.itxiaoshen.com Kubernetes官网地址 https://kubernetes.ioKubernetes GitHub源码…

    Go语言 2023年5月25日
    091
  • websocket:二.Golang实现Websocket消息通知

    我们在设计产品的时候通常都会遇到消息通知的时候,比如用户下单支付成功,比如用户有站内信来可以实时通知。而http是单向的,客户端请求,服务端返回,这次请求就已经结束。而websoc…

    Go语言 2023年5月25日
    060
  • Golang开源流媒体服务器(RTMP/RTSP/HLS/FLV等协议)

    一. lal 简介 lal是开源直播流媒体网络传输项目,主要由三部分组成: lalserver:流媒体转发服务器。类似于 nginx-rtmp-module等服务,但支持更多的协议…

    Go语言 2023年5月25日
    062
  • 第十五章:指针类型

    本篇翻译自《Practical Go Lessons》 Chapter 15: Pointer type 1 你将在本章将学到什么? 什么是指针? 什么时指针类型? 如何去创建并使…

    Go语言 2023年5月25日
    058
  • Golang使用swaggo自动生成Restful API文档

    相信很多程序猿和我一样不喜欢写API文档。写代码多舒服,写文档不仅要花费大量的时间,有时候还不能做到面面具全。但API文档是必不可少的,相信其重要性就不用我说了,一份含糊的文档甚至…

    Go语言 2023年5月25日
    056
  • Minio SDK访问Bucket的策略配置

    配置用户来访问 Bucket Minio 是高性能的对象存储服务,基于golang开发的,可以本地部署。用它来管理自己系统中的上传下载的文件很方便。​ 通过 SDK 访问 Mini…

    Go语言 2023年5月25日
    081
  • 大道如青天,协程来通信,Go lang1.18入门精炼教程,由白丁入鸿儒,Go lang通道channel的使用EP14

    众所周知,Go lang的作用域相对严格,数据之间的通信往往要依靠参数的传递,但如果想在多个协程任务中间做数据通信,就需要通道(channel)的参与,我们可以把数据封装成一个对象…

    Go语言 2023年5月25日
    084
  • mit6.824 lab1 (2022)

    go;gutter:true; lab1 要求按照论文实现一个mapReduce 框架</p> <pre><code> ;gutter:true…

    Go语言 2023年5月25日
    054
  • go 错误处理设计思考

    前段时间准备对线上一个golang系统服务进行内部开源,对代码里面的错误处理进行了一波优化。 优化的几个原因: 错误处理信息随意,未分类未定义。看到错误日志不能第一时间定位 错误的…

    Go语言 2023年5月25日
    049
  • 许式伟:Go+ 演进之路

    7 月 10 日,一年一度的 ECUG Con 2022 在线上圆满举行。许式伟作为七牛云 CEO、ECUG 社区发起人、Go+ 语言发明人,为大家来带了《Go+ 演进之路》的主题…

    Go语言 2023年5月25日
    083
  • Golang通脉之指针

    指针的概念 指针是存储另一个变量的内存地址的变量。 变量是一种使用方便的占位符,用于引用计算机内存地址。 一个指针变量可以指向任何一个值的内存地址。 在上面的图中,变量b的值为15…

    Go语言 2023年5月25日
    063
  • GO后端开发+VUE实列

    因为我是从java转到go,代码结构跟我之前用java的很像 在这里只浅显的实战运用,没有过多理论讲解 工作环境:IDE:Goland , Go 1.17.7 框架 Gin+Gor…

    Go语言 2023年5月25日
    060
  • go微服务框架Kratos笔记(四)使用nacos作为远端配置中心

    初识nacos nacos是阿里开源的一款用于动态服务发现、配置管理和服务管理的平台。 官方介绍,Nacos 致力于帮助您发现、配置和管理微服务。Nacos 提供了一组简单易用的特…

    Go语言 2023年5月25日
    076
  • golang低级编程:一.unsafe包

    go语言在设计上确保了一些安全的属性,限制了程序可能出错的途径。例如严格的类型转换规则。但也使得很多实现的细节无法通过go程序来访问,例如对于聚合类型(如结构体)的内存布局,或者一…

    Go语言 2023年5月25日
    043
亲爱的 Coder【最近整理,可免费获取】👉 最新必读书单  | 👏 面试题下载  | 🌎 免费的AI知识星球