Go 与结构体

「这是我参与11月更文挑战的第16天,活动详情查看:2021最后一次更文挑战

结构体

结构体是将零个或多个任意类型的命名变量组合在一起的聚合数据类型。每个变量都叫做结构体的成员。

1
2
3
4
5
go复制代码type Employee struct {
ID int
Name string
age int
}

Employee就是一个结构体。

定义结构体时要注意

1.如果一个成员变量的首字母大写,则它是可导出的。

2.成员变量的顺序对于结构体的同一性很重要,顺序不同表示不同的结构体。

3.结构体中的成员变量类型不能是结构体本身,但可以是该结构体的指针类型

1
2
3
4
5
6
go复制代码type Node struct {
value int
next *Node
//这样是非法的
//next Node
}

4.结构体的零值由结构体成员的零值构成。

结构体变量的声明

1
2
3
4
5
6
7
8
9
10
11
12
go复制代码type Point struct {
X int
Y int
}

// 方式一
p1 := Point{1, 2} // X=1 Y=2
// 方式二
p2 := Point{X: 3, Y: 4} // X=3 Y=4

// 如果某个成员变量没有指定,那么该成员变量的值为该类型零值
p3 := Point{Y:6} // X=0 Y=6

方式一和方式二不可以混用。并且不能通过方式一绕过不可导出变量无法在其他包中使用的规则

1
2
3
4
5
6
7
8
9
go复制代码// p.go
package p
type T struct { a, b int }

// q.go
package q
import "p"
var _ = p.T{1, 2} // 编译错误
var _ = p.T{a: 1, b: 2} // 编译错误

如果在函数中需要修改结构体变量的内容时,需要传递指针

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
go复制代码type Point struct {
X, Y int
}

func update(p Point, x int) { // 函数接收到的是实惨的一个副本
p.X = x
}

func update2(p *Point, x int) {
p.X = 2
}

func main() {
p := Point{}
update(p, 1)
fmt.Println(p) // {0 0}
update2(&p, 2)
fmt.Println(p) // {2 2}
}

结构体的比较

必须所有的成员变量是可比较的,那这个结构体是可变比较

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
go复制代码type Point struct {
X, Y int
}

type Book struct {
name string
chapters []string
}

func main() {
p1 := Point{X: 1, Y: 2}
p2 := Point{X: 1, Y: 2}
p3 := Point{X: 1, Y: 3}
fmt.Println(p1 == p2, p2 == p3) // true false

b1 := Book{name: "abc", chapters: []string{"1", "2"}}
b2 := Book{name: "abc", chapters: []string{"1", "2"}}
fmt.Print(b1 == b2) // 编译错误,无法比较,因为chapters是slice类型 不能比较
}

结构体嵌套和匿名成员

结构体嵌套的目的是为了复用
Go允许我们定义不带名称的结构体成员(即匿名成员),这样可以快捷访问到匿名成员里的成员变量

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
go复制代码type Point struct {
X, Y int
}

type Circle struct {
Point
Radius int
}

type Wheel struct {
Circle
Spokes int
}

func main() {
var w Wheel
w.X = 8 // 等价于w.Circle.Point.X
w.Y = 8
w.Radius = 5 // 等价于w.Circle.Radius
w.Spokes = 20

fmt.Printf("%#v", w) // main.Wheel{Circle:main.Circle{Point:main.Point{X:8, Y:8}, Radius:5}, Spokes:20}
}

外围的结构体Wheel不仅获得匿名成员Circle的内部成员变量,也会获得Circle的方法

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
go复制代码type Point struct {
X, Y int
}

func (p Point) Print() {
fmt.Printf("%#v", p)
}

type Circle struct {
Point
Radius int
}

func (c Circle) Print() {
fmt.Printf("%#v", c)
}


type Wheel struct {
Circle
Spokes int
}

// func (w Wheel) Print() {
// fmt.Printf("%#v", w)
// }

func main() {
var w Wheel
w.X = 8
w.Y = 8
w.Radius = 5
w.Spokes = 20

fmt.Printf("%#v\n", w) // main.Wheel{Circle:main.Circle{Point:main.Point{X:8, Y:8}, Radius:5}, Spokes:20}
w.Print() // main.Circle{Point:main.Point{X:8, Y:8}, Radius:5}
}

如果在Wheel上再加一个匿名命名成员Band, 它也有Print方法

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
go复制代码type Band struct {
Name string
}

func (b Band) Print() {
fmt.Printf("%#v", b)
}

type Wheel struct {
Circle
Band
Spokes int
}


func main() {
var w Wheel
w.X = 8
w.Y = 8
w.Radius = 5
w.Spokes = 20

fmt.Printf("%#v\n", w) // main.Wheel{Circle:main.Circle{Point:main.Point{X:8, Y:8}, Radius:5}, Band:main.Band{Name:""}, Spokes:20}
w.Print() // 编译报错
w.Circle.Print() // main.Circle{Point:main.Point{X:8, Y:8}, Radius:5}
w.Band.Print() // main.Band{Name:""}
}

如果外围结构体WheelPrint方法,会直接调用该方法;如果没有,但其内部的多个匿名成员都有该方法,需要显示指定调用哪个匿名结构体的Print方法

本文转载自: 掘金

开发者博客 – 和开发相关的 这里全都有

0%