您现在的位置是:首页 >技术杂谈 >05-修建数据殿堂:Golang struct的艺术架构网站首页技术杂谈

05-修建数据殿堂:Golang struct的艺术架构

花崽oyf 2024-10-28 00:01:03
简介05-修建数据殿堂:Golang struct的艺术架构

在这里插入图片描述

?个人主页:个人主页
?系列专栏:Golang基础
?Go(又称Golang)是由Google开发的开源编程语言。它结合了静态类型的安全性和动态语言的灵活性,拥有高效的并发编程能力和简洁的语法。Go被设计用于构建可扩展、高性能的软件系统,具有优秀的内存管理和快速的编译速度,适用于Web开发、系统编程和云计算等领域。

struct类型的声明

Go语言中,和C或者其他语言一样,可以声明新的类型,作为其它类型的属性或字段的容器。
例如,可以创建一个自定义类型 person 代表一个人的实体。
这个实体拥有属性:姓名和年龄。
这样的类型称之 struct
如下代码所示:

type person struct {
	name string
	age int
}

声明一个 struct 如此简单,上面的类型包含有两个字段:

  • 一个 string 类型的字段name,用来保存用户名称这个属性
  • 一个 int 类型的字段age,用来保存用户年龄这个属性

使用 struct 看下面的代码:

type person struct {
	name string
	age int
}
var P person // P现在就是person类型的变量了
P.name = "张三" // 赋值"张三"给P的name属性.
P.age = 22 // 赋值"22"给变量P的age属性
fmt.Printf("my name is %s", P.name) // 访问P的name属性.

除了上面这种P的声明使用之外,还有另外几种声明使用方式:
1.按照顺序提供初始化值

P := person{"张三", 22}

2.通过 field:value 的方式初始化,这样可以任意顺序

P := person{age:22, name:"张三"}

3.当然也可以通过 new 函数分配一个指针,此处P的类型为 *person

P := new(person)

看一个完整的使用 struct 的例子:

package main
import "fmt"

// 声明一个新的类型
type person struct {
	name string
	age int
}
// 比较两个人的年龄,返回年龄大的那个人,并且返回年龄差
// struct也是传值的
func Older(p1, p2 person) (person, int) {
	if p1.age>p2.age { // 比较p1和p2这两个人的年龄
		return p1, p1.age-p2.age
	}
	return p2, p2.age-p1.age
}
func main() {
	var tom person
	// 赋值初始化
	tom.name, tom.age = "Tom", 18
	// 两个字段都写清楚的初始化
	bob := person{age:25, name:"Bob"}
	// 按照struct定义顺序初始化值
	paul := person{"Paul", 43}
	tb_Older, tb_diff := Older(tom, bob)
	tp_Older, tp_diff := Older(tom, paul)
	bp_Older, bp_diff := Older(bob, paul)
	fmt.Printf("Of %s and %s, %s is older by %d years
",
		tom.name, bob.name, tb_Older.name, tb_diff)
	fmt.Printf("Of %s and %s, %s is older by %d years
",
		tom.name, paul.name, tp_Older.name, tp_diff)
	fmt.Printf("Of %s and %s, %s is older by %d years
",
		bob.name, paul.name, bp_Older.name, bp_diff)
}

struct 的匿名字段

定义的时候是字段名与其类型一一对应,实际上Go支持只提供类型,而不写字段名的方式,也就是匿名字段,也称为嵌入字段。
当匿名字段是一个 struct 的时候,那么这个 struct 所拥有的全部字段都被隐式地引入了当前定义的这个 struct
看一个例子,让上面说的这些更具体化:

package main
import "fmt"

type Human struct {
	name string
	age int
	weight int
}
type Student struct {
	Human // 匿名字段,那么默认Student就包含了Human的所有字段
	speciality string
}
func main() {
	// 初始化一个学生
	mark := Student{Human{"Mark", 25, 120}, "Computer Science"}
	// 访问相应的字段
	fmt.Println("His name is ", mark.name)
	fmt.Println("His age is ", mark.age)
	fmt.Println("His weight is ", mark.weight)
	fmt.Println("His speciality is ", mark.speciality)
	// 修改对应的备注信息
	mark.speciality = "AI"
	fmt.Println("Mark changed his speciality")
	fmt.Println("His speciality is ", mark.speciality)
	// 修改他的年龄信息
	fmt.Println("Mark become old")
	mark.age = 46
	fmt.Println("His age is", mark.age)
	// 修改他的体重信息
	fmt.Println("Mark is not an athlet anymore")
	mark.weight += 60
	fmt.Println("His weight is", mark.weight)
}

看到 Student 访问属性 agename 的时候,就像访问自己所有用的字段一样,匿名字段就是这样,能够实现字段的继承。student 还能访问 Human 这个字段作为字段名,请看下面的代码:

mark.Human = Human{"Marcus", 55, 220}
mark.Human.age -= 1

通过匿名访问和修改字段相当的有用,但是不仅仅是 struct 字段,所有的内置类型和自定义类型都是可以作为匿名字段的。
请看下面的例子:

package main
import "fmt"

type Skills []string
type Human struct {
	name string
	age int
	weight int
}
type Student struct {
	Human // 匿名字段,struct
	Skills // 匿名字段,自定义的类型string slice
	int // 内置类型作为匿名字段
	speciality string
}
func main() {
	// 初始化学生Jane
	jane := Student{Human:Human{"Jane", 35, 100}, speciality:"Biology"}
	// 现在访问相应的字段
	fmt.Println("Her name is ", jane.name)
	fmt.Println("Her age is ", jane.age)
	fmt.Println("Her weight is ", jane.weight)
	fmt.Println("Her speciality is ", jane.speciality)
	// 修改他的skill技能字段
	jane.Skills = []string{"anatomy"}
	fmt.Println("Her skills are ", jane.Skills)
	fmt.Println("She acquired two new ones ")
	jane.Skills = append(jane.Skills, "physics", "golang")
	fmt.Println("Her skills now are ", jane.Skills)
	// 修改匿名内置类型字段
	jane.int = 3
	fmt.Println("Her preferred number is", jane.int)
}

从上面例子看出来 struct 不仅仅能够将 struct 作为匿名字段,自定义类型、内置类型都可以作为匿名字段,而且可以在相应的字段上面进行函数操作(如例子中的 append )。
这里有一个问题:如果 human 里面有一个字段叫做 phone ,而 student 也有一个字段叫做 phone ,那么该怎么办呢?
Go里面很简单的解决了这个问题,最外层的优先访问,也就是当通过 student.phone 访问的时候,是访问 student 里面的字段,而不是 human 里面的字段。
这样就允许去重载通过匿名字段继承的一些字段,当然如果想访问重载后对应匿名类型里面的字段,可以通过匿名字段名来访问。请看下面的例子:

package main
import "fmt"

type Human struct {
	name string
	age int
	phone string // Human类型拥有的字段
}
type Employee struct {
	Human // 匿名字段Human
	speciality string
	phone string // 雇员的phone字段
}
func main() {
	Bob := Employee{Human{"Bob", 34, "777-444-XXXX"}, "Designer", "333-222"}
	fmt.Println("Bob's work phone is:", Bob.phone)
	// 如果要访问Human的phone字段
	fmt.Println("Bob's personal phone is:", Bob.Human.phone)
}
风语者!平时喜欢研究各种技术,目前在从事后端开发工作,热爱生活、热爱工作。