Go 语言结构体

news/2025/2/13 2:01:10/

1、Go 语言结构体

Go 语言中数组可以存储同一类型的数据,但在结构体中我们可以为不同项定义不同的数据类型。

结构体是由一系列具有相同类型或不同类型的数据构成的数据集合。

1.1 定义结构体

结构体定义需要使用 typestruct 语句,struct 语句定义一个新的数据类型,结构体中有一个或多个成员。

type 语句设定了结构体的名称,结构体的格式如下:

type struct_variable_type struct {member definitionmember definition...member definition
}

一旦定义了结构体类型,它就能用于变量的声明,语法格式如下:

variable_name := structure_variable_type {value1, value2...valuen}
或
variable_name := structure_variable_type { key1: value1, key2: value2..., keyn: valuen}
package mainimport "fmt"type Books struct {title   stringauthor  stringsubject stringbook_id int
}func main() {// 创建一个新的结构体// {Go语言 Tom Go语言教程 6495407}fmt.Println(Books{"Go语言", "Tom", "Go语言教程", 6495407})// 也可以使用 key => value 格式// {Go语言 Tom Go语言教程 6495407}fmt.Println(Books{title: "Go语言", author: "Tom", subject: "Go语言教程", book_id: 6495407})// 忽略的字段为0或空// {Go语言 Tom  0}fmt.Println(Books{title: "Go语言", author: "Tom"})
}
package mainimport "fmt"type Person struct {name stringage  int
}func main() {// 使用new创建内置函数,宇段默认初始化为其类型的零值,返回值是指向结构的指针p := new(Person)p.name = "tom"p.age = 27fmt.Println(p.name)fmt.Println(p.age)person := Person{"marry", 100}fmt.Println(person.name)fmt.Println(person.age)
}
# 程序输出
tom
27
marry
100

1.2 访问结构体成员

如果要访问结构体成员,需要使用点号.操作符,格式为:

结构体.成员名

结构体类型变量使用 struct 关键字定义。

// 实例如下
package mainimport "fmt"type Books struct {title   stringauthor  stringsubject stringbook_id int
}func main() {/* 声明 Book1 为 Books 类型 */var Book1 Books/* 声明 Book2 为 Books 类型 */var Book2 Books/* book 1 描述 */Book1.title = "Go语言"Book1.author = "Tom"Book1.subject = "Go语言教程"Book1.book_id = 6495407/* book 2 描述 */Book2.title = "Python教程"Book2.author = "Marry"Book2.subject = "Python语言教程"Book2.book_id = 6495700/* 打印 Book1 信息 */fmt.Printf("Book 1 title : %s\n", Book1.title)fmt.Printf("Book 1 author : %s\n", Book1.author)fmt.Printf("Book 1 subject : %s\n", Book1.subject)fmt.Printf("Book 1 book_id : %d\n", Book1.book_id)/* 打印 Book2 信息 */fmt.Printf("Book 2 title : %s\n", Book2.title)fmt.Printf("Book 2 author : %s\n", Book2.author)fmt.Printf("Book 2 subject : %s\n", Book2.subject)fmt.Printf("Book 2 book_id : %d\n", Book2.book_id)
}
# 程序输出
Book 1 title : Go语言
Book 1 author : Tom
Book 1 subject : Go语言教程
Book 1 book_id : 6495407
Book 2 title : Python教程
Book 2 author : Marry
Book 2 subject : Python语言教程
Book 2 book_id : 6495700

1.3 结构体作为函数参数

你可以像其他数据类型一样将结构体类型作为参数传递给函数。

package mainimport "fmt"type Books struct {title   stringauthor  stringsubject stringbook_id int
}func main() {/* 声明 Book1 为 Books 类型 */var Book1 Books/* 声明 Book2 为 Books 类型 */var Book2 Books/* book 1 描述 */Book1.title = "Go语言"Book1.author = "Tom"Book1.subject = "Go语言教程"Book1.book_id = 6495407/* book 2 描述 */Book2.title = "Python教程"Book2.author = "Marry"Book2.subject = "Python语言教程"Book2.book_id = 6495700/* 打印 Book1 信息 */printBook(Book1)/* 打印 Book2 信息 */printBook(Book2)
}func printBook(book Books) {fmt.Printf("Book title : %s\n", book.title)fmt.Printf("Book author : %s\n", book.author)fmt.Printf("Book subject : %s\n", book.subject)fmt.Printf("Book book_id : %d\n", book.book_id)
}
# 程序输出
Book title : Go语言
Book author : Tom
Book subject : Go语言教程
Book book_id : 6495407
Book title : Python教程
Book author : Marry
Book subject : Python语言教程
Book book_id : 6495700

1.4 结构体指针

你可以定义指向结构体的指针类似于其他指针变量,格式如下:

var struct_pointer *Books

以上定义的指针变量可以存储结构体变量的地址。查看结构体变量地址,可以将 & 符号放置于结构体变量前:

struct_pointer = &Book1

使用结构体指针访问结构体成员,使用 . 操作符:

struct_pointer.title

实例:

package mainimport "fmt"type Books struct {title   stringauthor  stringsubject stringbook_id int
}func main() {var book = Books{"Go入门到放弃", "Tom", "go系列教程", 012231}var b *Booksb = &book// &{Go入门到放弃 Tom go系列教程 5273}fmt.Println(b)// {Go入门到放弃 Tom go系列教程 5273}fmt.Println(*b)// 0xc000006028fmt.Println(&b)// {Go入门到放弃 Tom go系列教程 5273}fmt.Println(book)
}
# b这个指针是Books类型的
var b *Books
# book是Books的一个实例化的结构,&book就是把这个结构体的内存地址赋给了b
b  = &book
# 那么在使用的时候,只要在b的前面加个*号,就可以把b这个内存地址对应的值给取出来了
*b
# 取了b这个指针的内存地址,也就是b这个指针是放在内存空间的什么地方的
&b
# 就是Books这个结构体,打印出来就是它自己,也就是指针b前面带了*号的效果
book      

只有一个特殊的地方,尽管 b 所表示的是 book 对象的内存地址,但是,在从 b 对应的内存地址取属性值的时

候,就不是 *b.title 了。而是直接使用b.title,这点很特殊,它的效果就相当于 book.title

package mainimport "fmt"type Books struct {title   stringauthor  stringsubject stringbook_id int
}func main() {var book = Books{"Go入门到放弃", "Tom", "go系列教程", 012231}var b *Booksb = &book// Go入门到放弃fmt.Println(b.title)// Go入门到放弃fmt.Println(book.title)// Tomfmt.Println(b.author)// Tomfmt.Println(book.author)
}

struct 类似于 java 中的类,可以在 struct 中定义成员变量。

要访问成员变量,可以有两种方式:

  • 通过 struct 变量.成员 变量来访问。

  • 通过 struct 指针.成员 变量来访问。

不需要通过 getter, setter 来设置访问权限。

package mainimport "fmt"// 定义矩形类
type Rect struct {// 类型只包含属性,并没有方法width, height float64
}// 为Rect类型绑定Area的方法,*Rect为指针引用可以修改传入参数的值
func (r *Rect) Area() float64 {// 方法归属于类型,不归属于具体的对象,声明该类型的对象即可调用该类型的方法return r.width * r.height
}func main() {var rect = Rect{3, 4}var p *Rect = &rectvar result = p.Area()// 12fmt.Println(result)
}

结构体是作为参数的值传递:

package mainimport "fmt"type Books struct {title   stringauthor  stringsubject stringbook_id int
}func changeBook(book Books) {book.title = "book1_change"
}func main() {var book1 Booksbook1.title = "go"book1.author = "tom"book1.book_id = 1changeBook(book1)// {go tom  1}fmt.Println(book1)
}

利用指针改变结构体对应的值:

package mainimport "fmt"type Books struct {title   stringauthor  stringsubject stringbook_id int
}func changeBook(book *Books) {book.title = "book1_change"
}func main() {var book1 Booksbook1.title = "go"book1.author = "tom"book1.book_id = 1changeBook(&book1)// {book1_change tom  1}fmt.Println(book1)
}

1.5 public 和 private属性

结构体中属性的首字母大小写问题:

  • 首字母大写相当于 public
  • 首字母小写相当于 private

这个 public 和 private 是相对于包(go 文件首行的 package 后面跟的包名)来说的。

当要将结构体对象转换为 JSON 时,对象中的属性首字母必须是大写,才能正常转换为 JSON。

package mainimport "fmt"
import "encoding/json"type Person struct {//Name字段首字母大写Name string//age字段首字母小写     age int
}func main() {person := Person{"小明", 18}//json.Marshal 将对象转换为json字符串if result, err := json.Marshal(&person); err == nil {// {"Name":"小明"}fmt.Println(string(result))}
}
package mainimport "fmt"
import "encoding/json"type Person struct {//Name字段首字母大写Name string//age字段首字母小写     Age int
}func main() {person := Person{"小明", 18}//json.Marshal 将对象转换为json字符串if result, err := json.Marshal(&person); err == nil {// {"Name":"小明","Age":18}fmt.Println(string(result))}
}

那这样 JSON 字符串以后就只能是大写了么? 当然不是,可以使用 tag 标记要返回的字段名。

package mainimport "fmt"
import "time"
import "encoding/json"type Person struct {//标记json名字为nameName string `json:"name"`Age  int    `json:"age"`// 标记忽略该字段Time int64 `json:"-"`
}func main() {person := Person{"小明", 18, time.Now().Unix()}if result, err := json.Marshal(&person); err == nil {// {"name":"小明","age":18}fmt.Println(string(result))}
}

定义的结构体如果只在当前包内使用,结构体的属性不用区分大小写。如果想要被其他的包引用,那么结构体的属

性的首字母需要大写。

package book// 结构体小写开头的属性只能包内调用
type Books struct {Title   stringAuthor  stringSubject stringbook_id int
}
package mainimport ("fmt""proj/book"
)func main() {/* 声明 book 为 Books 类型 */var book book.Books/* book  描述 */book.Title = "Go语言"book.Author = "Tom"book.Subject = "Go语言教程"// 如果进行了如下调用,则会报错// book.book_id = 6495407/* 打印 book 信息 */printBook(book)
}func printBook(book book.Books) {fmt.Printf("Book title : %s\n", book.Title)fmt.Printf("Book author : %s\n", book.Author)fmt.Printf("Book subject : %s\n", book.Subject)// 无法调用// fmt.Printf( "Book book_id : %d\n", book.book_id)
}
# 程序输出
Book title : Go语言
Book author : Tom
Book subject : Go语言教程

1.6 struct{}和struct{}{}

一般我们知道struct在Go语言中是用于定义结构类型

type User struct {Name stringAge  int
}

struct {}是一个无元素的结构体类型,通常在没有信息存储时使用。

优点是大小为0,不需要内存来存储 struct {} 类型的值。

struct {} {}是一个复合字面量,它构造了一个struct {}类型的值,该值也是空。

比如我们可以用map[string]struct{}来当作成一个set来用。

package mainimport "fmt"func main() {// 定义一个mapvar set map[string]struct{}set = make(map[string]struct{})// struct{}{}构造了一个struct{}类型的值set["red"] = struct{}{}value, ok := set["red"]// Is red in the map? truefmt.Println("Is red in the map?", ok)// {}fmt.Println(value)
}

输出内容:

# 程序输出
Is red in the map? true
{}

map可以通过comma ok机制来获取该key是否存在,value, ok := map["key"],如果没有对应的值,ok

false,这样可以通过定义成map[string]struct{}的形式,值不再占用内存。其值仅有两种状态,有或无。

其他知识点:

  • chan struct{}:可以用作通道的退出
// 1个goroutine
package mainimport ("fmt""time"
)func worker(name string, stopChan chan struct{}) {for {select {case <-stopChan:fmt.Println("receive a stop signal, ", name)returndefault:fmt.Println("I am worker ", name)time.Sleep(1 * time.Second)}}
}func main() {/*I am worker  aI am worker  areceive a stop signal,  a*/stopCh := make(chan struct{})go worker("tom", stopCh)time.Sleep(2 * time.Second)stopCh <- struct{}{}time.Sleep(1 * time.Second)
}
# 程序输出
I am worker  tom
I am worker  tom
receive a stop signal,  tom
package mainimport ("fmt""time"
)func worker(name string, stopchan chan struct{}) {for {select {case <-stopchan:fmt.Println("receive a stop signal, ", name)returndefault:fmt.Println("I am worker ", name)time.Sleep(1 * time.Second)}}
}func main() {stopCh := make(chan struct{})go worker("a", stopCh)go worker("b", stopCh)time.Sleep(2 * time.Second)stopCh <- struct{}{}time.Sleep(1 * time.Second)
}
# 程序输出
I am worker  b
I am worker  a
I am worker  b
I am worker  a
I am worker  b
receive a stop signal,  a
I am worker  b

也就是说a退出了,b没有退出,因为stopCh <- struct{}{}只发送一个信号,被a接收了,b不受影响。

如果想让2个goroutine同时退出,需要这样写:

package mainimport ("fmt""time"
)func worker(name string, stopchan chan struct{}) {for {select {case <-stopchan:fmt.Println("receive a stop signal, ", name)returndefault:fmt.Println("I am worker ", name)time.Sleep(1 * time.Second)}}
}func main() {stopCh := make(chan struct{})go worker("a", stopCh)go worker("b", stopCh)time.Sleep(2 * time.Second)close(stopCh)time.Sleep(1 * time.Second)
}
# 程序输出
I am worker  b
I am worker  a
I am worker  b
I am worker  a
receive a stop signal,  b
receive a stop signal,  a
  • 两个struct{}{}地址相等
package mainimport "fmt"func main() {var s1 = struct{}{}var s2 = struct{}{}// truefmt.Printf("%t", s1 == s2)
}

1.7 组合

1.7.1 内嵌字段的初始化和访问

struct的字段访问使用点操作符.struct的宇段可以嵌套很多层,只要内嵌的字段是唯一的即可,不需要使

用全路径进行访 。在以下示例中, 可以使用z.a代替,可以使用z.a代替z.Y.X.a

package maintype X struct {a int
}type Y struct {Xb int
}type Z struct {Yc int
}func main() {x := X{a: 1}y := Y{X: x,b: 2,}z := Z{Y: y,c: 3,}// z.a, z.Y.a, z.Y.X.a 三者是等价的, z.a z.Y.a是z.Y.X.a的简写// 1 1 1println(z.a, z.Y.a, z.Y.X.a)z = Z{}z.a = 2// 2 2 2println(z.a, z.Y.a, z.Y.X.a)
}

struct的多层嵌套中,不同嵌套层次可以有相同的字段,此时最好使用完全路径进行访问和初始化。在实际数

据结构的定义中应该尽量避开相同的字段,以免在使用中出现歧义。

package maintype X struct {a int
}type Y struct {Xa int
}type Z struct {Ya int
}func main() {x := X{a: 1}y := Y{X: x,a: 2,}z := Z{Y: y,a: 3,}// 此时的z.a, z.Y.a, z.Y.X.a 代表不同的字段// 3 2 1println(z.a, z.Y.a, z.Y.X.a)z = Z{}z.a = 4z.Y.a = 5z.Y.X.a = 6// 此时的z.a, z.Y.a, z.Y.X.a 代表不同的字段// 4 5 6println(z.a, z.Y.a, z.Y.X.a)
}

1.7.2 内嵌字段的方法调用

struct 类型方法调用也使用点操作符,不同嵌套层次的字段可以有相同的方法,外层变量调用内嵌字段的方法时也

可以像嵌套字段的访问一样使用简化模式。如果外层字段和内层字段有相同的方法,则使用简化模式访问外层的方

法会覆盖内层的方法。即在简写模式下,Go编译器优先从外向内逐层查找方法,同名方法中外层的方法能够覆盖

内层的方法。这个特性有点类似于面向对象编程中,子类覆盖父类的同名方法。

package mainimport "fmt"type X struct {a int
}type Y struct {Xb int
}type Z struct {Yc int
}func (x X) Print() {fmt.Printf("In X, a=%d\n", x.a)
}func (x X) XPrint() {fmt.Printf("In X, a=%d\n", x.a)
}func (y Y) Print() {fmt.Printf("In Y, b=%d\n", y.b)
}func (z Z) Print() {fmt.Printf("In Z, c=%d \n", z.c)//显式的完全路径调用内嵌字段的方法z.Y.Print()z.Y.X.Print()
}func main() {x := X{a: 1}y := Y{X: x,b: 2,}z := Z{Y: y,c: 3,}// 从外向内查找首先找到的是Z的Print()方法// In Z, c=3// In Y, b=2// In X, a=1z.Print()// 从外向内查找,最后找到的是X的XPrint()方法// In X, a=1z.XPrint()// In X, a=1z.Y.XPrint()
}

不推荐在多层的 struct类型中内嵌多个同名的字段;但是并不反对struct定义和内嵌字段同名方法的用法,因为这

提供了一种编程技术,使得 struct 能够重写内嵌字段的方法,提供面向对象编程中子类覆盖父类的同名方法的功

能。

1.8 组合的方法集

组合结构的方法集有如下规则:

(1)、若类型S包含匿名字段T,则S的方法集包含T的方法集。

(2)、若类型S包含匿名字段*T,则S的方法集包含T*T方法集。

(3)、不管类型S中嵌入的匿名字段是T还是*T*S方法集总是包含T*T方法集。

下面举个例子来验证这个规则的正确性,前面讲到方法集时提到Go编译器会对方法调用进行自动转换,为了阻止

自动转换,本示例使用方法表达式的调用方式,这样能更清楚地理解这个方法集的规约。

package maintype X struct {a int
}type Y struct {X
}type Z struct {*X
}func (x X) Get() int {return x.a
}func (x *X) Set(i int) {x.a = i
}func main() {x := X{a: 1}y := Y{X: x,}// 1println(y.Get())//此处编译器做了自动转换y.Set(2)// 2println(y.Get())//为了不让编译器做自动转换, 我们使用method expression格式调用(*Y).Set(&y, 3)// type Y的方法集合并没有Set这个方法, 所以下一句编译通不过// Y.Set(y, 3) // type Y has no method Set// 3println(y.Get())z := Z{X: &x,}//按照嵌套字段的方法集的规则//Z 内嵌字段*X,所以type Z和type *Z方法集都是 Get和Set//为了不让编译器做自动转换, 我们仍然使用method expression格式调用Z.Set(z, 4)// 4println(z.Get())(*Z).Set(&z, 5)// 5println(z.Get())}
# 程序输出
1
2
3
4
5

到目前为止还没有发现方法集有多大的用途,而且通过实践发现,Go编译器会进行自动转换,看起来不需要太关

注方法集,这种认识是错误的。编译器的自动转换仅适用于直接通过类型实例调用方法时才有效,类型实例传递给

接口时,编译器不会进行自动转换,而是会进行严格的方法集校验。

Go函数的调用实参都是值拷贝,方法调用参数传递也是一样的机制,具体类型变量传递给接口时也是值拷贝,如

果传递给接口变量的是值类型,但调用方法的接收者是指针类型,则程序运行时虽然能够将接收者转换为指针,但

这个指针是副本的指针,并不是我们期望的原变量的指针。所以语言设计者为了杜绝这种非期望的行为,在编译时

做了严格的方法集合的检查,不允许产生这种调用;如果传递给接口的变量是指针类型,则接口调用的是值类型的

方法,程序运行时能够自动转换为值类型,这种转换不会带来副作用,符合调用者的预期,所以这种转换是允许

的,而且这种情况符合方法集的规约。具体类型传递给接口时编译器会进行严格的方法集校验,掌握了方法集的概

念在后续章节学习接口时非常有用。

1.9 数组内嵌到 struct

package mainimport "fmt"func main() {a := [3]int{1, 2, 3}c := struct{ s [3]int }{s: a}// {[1 2 3]}fmt.Println(c.s)
}

http://www.ppmy.cn/news/589769.html

相关文章

【MySQL】- 07 影响MySQL性能的配置参数

影响MySQL性能的配置参数 1 连接2 查询缓存3 临时表4 会话内存5慢查询日志小结 1 连接 ​ 连接通常来自Web服务器&#xff0c;下面列出了一些与连接有关的参数&#xff0c;以及该如何设置它们。 ​ 1、max_connections ​ 这是Web服务器允许的最大连接数&#xff0c;记住每个连…

iOS 17 beta 2有哪些BUG?iOS 17 beta 2推荐升级吗?

虽然iOS 17 beta 2 带来了大量的功能更新&#xff0c;但毕竟是测试版&#xff0c;海量的适配BUG也一同随之而来。 想升级iOS 17 beta 2的用户不妨先查看下目前存在的问题汇总&#xff01; 一&#xff1a;存储空间更小了 升级beta1后存储空间缩小了大概3G左右&#xff0c;bet…

低功耗电路电池电压测量pcb设计

相信大家都遇到过低功耗电路电池电量检测的PCB设计。如何测量电池的电压呢&#xff1f;采用运放来进行测量肯定不考虑&#xff0c;因为运放也是耗电单元。 那么我们考虑一下电阻分压进行电压测量&#xff0c;因需要考虑到低功耗因素&#xff0c;串联电阻就必须很大&#xff0c;…

CS61a-2020fall学习笔记

此项目含有我对cs61a2020秋季学期大部分lab&#xff0c;discussion&#xff0c;project和homework的解答。从2021/06/22-2021/11/26&#xff0c;暑假太贪玩了&#xff0c;导致一共用时5个月才完成。 课程主页 github项目地址 使用ok时为了不提交邮箱&#xff0c;可在命令行后…

HDLbits---Exams/m2014 q6

HDLbits—Exams/m2014 q6 状态转移图完成就好了不需要任何解题思路 module top_module (input clk,input reset, // synchronous resetinput w,output z); parameter A 3b000,B 3b001,C 3b010,D 3b011,E 3b100,F 3b101;reg[2:0]state,next_state;always(posedge clk)beg…

北京数码视讯s905l固件_数码视讯Q6联通版S905L芯片第三方刷机免拆卡刷固件

数码视讯Q6联通版S905L芯片第三方刷机免拆卡刷固件分享 固件介绍&#xff1a; 1、不带ROOT权限&#xff0c;适用于数码视讯Q6联通版S905L芯片。 2、调出原厂固件屏蔽的wifi&#xff0c;开放原厂固件屏蔽的市场安装和u盘安装apk&#xff1b; 3、无开机广告&#xff0c;无系统更新…

华为凌霄子母路由 Q6参数 华为凌霄子母路由 Q6怎么样

华为凌霄子母路由 Q6 共有电力线版以及网线版两种版本可选&#xff0c;售价分别为 899 元和 1699 元。 华为凌霄子母路由 Q6 更多使用感受和评价 http://huawei.adiannao.cn 具体配置参数为&#xff0c;华为凌霄子母路由 Q6 搭载全新华为专利 —— 凌霄 PLC 技术&#xff0c;…

北京数码视讯s905l固件_数码视讯Q6联通版S905L芯片第三方刷机固件

数码视讯Q6联通版S905L芯片第三方刷机固件是一款数码视讯Q6联通版适用于S905L芯片通用机顶盒不带ROOT权限刷机包&#xff0c;刷机后请务必恢复出厂一次&#xff0c;让盒子env退出刷机模式。建议在专业人员指导下刷机。 固件介绍&#xff1a; 1、不带ROOT权限&#xff0c;适用于…