Go 复合数据类型之结构体与自定义类型

Go 复合数据类型之结构体与自定义类型

go语言是面向对象的吗

文章目录

  • Go 复合数据类型之结构体与自定义类型
    • 一、类型别名和自定义类型
      • 1.1 类型定义(Type Definition)
        • 简单示例
      • 1.2 类型别名
        • 简单示例
      • 1.3 类型定义和类型别名的区别
    • 二、结构体
      • 2.1 结构体介绍
      • 2.2 结构体的定义
      • 2.3 定义一个空结构体
        • 2.3.1 空结构体介绍
        • 2.3.2 空结构体作用
      • 2.4 使用其他结构体作为自定义结构体中字段的类型
        • 2.4.1 结构体嵌套介绍
        • 2.4.2 结构体嵌套的好处
      • 2.5 匿名结构体
    • 三、结构体变量的声明与初始化
      • 3.1 结构体变量的声明
      • 3.2 零值初始化
      • 3.3 使用顺序复合字面值初始化
      • 3.4 使用键值对初始化
      • 3.5 使用特定的构造函数
    • 四、结构体类型的内存布局
    • 五、结构体字段的可见性
    • 六、结构体的“继承”

一、类型别名和自定义类型

在Go中,可以通过使用关键字type来自定义一个新类型,有两种常见的方法:类型定义(Type Definition)和类型别名(Type Alias)。

1.1 类型定义(Type Definition)

类型定义是通过将一个已有的类型声明为一个新的类型的方式,可以将一个现有类型的特性和方法继承给新类型。这个新类型具有原有类型的所有方法和属性,但它们是不同的类型。

// 使用类型定义创建新类型
type MyInt int

MyInt就是一种新的类型,它具有int的特性。

如果一个新类型是基于某个 Go 原生类型定义的,那么我们就叫 Go 原生类型为新类型底层类型(Underlying Type)它被用来判断两个类型本质上是否相同(Identical)。比如上面这个例子中类型 int 就是类型 MyInt 的底层类型。

简单示例

这里我们定义两个类型,新类型 T1 是基于 Go 原生类型 int 定义的新自定义类型,而新类型 T2 则是基于刚刚定义的类型 T1,定义的新类型:

type T1 int 
type T2 T1  

在上面例子中,虽然 T1T2 是不同类型,但因为它们的底层类型都是类型 int,所以它们在本质上是相同的。而本质上相同的两个类型,它们的变量可以通过显式转型进行相互赋值,相反,如果本质上是不同的两个类型,它们的变量间连显式转型都不可能,更不要说相互赋值了。

比如下面这个代码示例:

type T1 int
type T2 T1
type T3 stringfunc main() {var n1 T1var n2 T2 = 5n1 = T1(n2)  // okvar s T3 = "hello"n1 = T1(s) // 错误:cannot convert s (type T3) to type T1
}

这段代码中,T1 和 T2 本质上是相同的类型,所以我们可以将 T2 变量 n2 的值,通过显式转型赋值给 T1 类型变量 n1。而类型 T3 的底层类型为类型 string,与 T1/T2 的底层类型不同,所以它们本质上就不是相同的类型。这个时候,如果我们把 T3 类型变量 s 赋值给 T1 类型变量 n1,编译器就会给出编译错误的提示。

1.2 类型别名

类型别名是在已有类型的基础上创建一个新类型,它们是同一类型的两个名称,所以它们可以互相赋值,它们只是为了提供更清晰的语义或用于解决包之间的命名冲突。

类型别名规定:TypeAlias只是Type的别名,本质上TypeAlias与Type是同一个类型。就像一个孩子小时候有小名、乳名,上学后用学名,英语老师又会给他起英文名,但这些名字都指的是他本人。

type TypeAlias = Type
简单示例
type T = string var s string = "hello" 
var t T = s // ok
fmt.Printf("%T\n", t) // string

因为类型 T 是通过类型别名的方式定义的,T 与 string 实际上是一个类型,所以这里,使用 string 类型变量 s 给 T 类型变量 t 赋值的动作,实质上就是同类型赋值。另外我们也可以看到,通过 Printf 输出的变量 t 的类型信息也是 string

1.3 类型定义和类型别名的区别

类型别名与类型定义表面上看只有一个等号的差异,我们通过下面的这段代码来理解它们之间的区别。

//类型定义
type NewInt int//类型别名
type MyInt = intfunc main() {var a NewIntvar b MyIntfmt.Printf("type of a:%T\n", a) //type of a:main.NewIntfmt.Printf("type of b:%T\n", b) //type of b:int
}

结果显示a的类型是main.NewInt,表示main包下定义的NewInt类型。b的类型是intMyInt类型只会在代码中存在,编译完成时并不会有MyInt类型。

二、结构体

2.1 结构体介绍

  1. 结构体是一种自定义的复合数据类型,用于组合不同类型的值,以创建更复杂的数据结构。每个结构体可以包含一个或多个字段,每个字段都有自己的名称和类型。可以封装多个基本数据类型。结构体的字段可以是基本数据类型,也可以是其他自定义结构体类型。
  2. 英文名称struct。 也就是我们可以通过struct关键字来定义结构体类型。
  3. Go语言中通过struct来实现面向对象。

2.2 结构体的定义

使用typestruct关键字来定义结构体,通常遵循以下格式:

type T struct {Field1 T1Field2 T2... ...FieldN Tn
}

其中:

  • T为类型名:标识自定义结构体的名称,在同一个包内不能重复。
  • Field1Field2为字段名:表示结构体字段名。结构体中的字段名必须唯一。
  • T1T2 为字段类型:表示结构体字段的具体类型。

根据上面的这个定义,我们会得到一个名为 T 的结构体类型,定义中 struct 关键字后面的大括号包裹的内容就是一个类型字面值。我们看到这个类型字面值由若干个字段(field)聚合而成,每个字段有自己的名字与类型,并且在一个结构体中,每个字段的名字应该都是唯一的。

比如这里,我就用对现实世界的书(Book)进行抽象定义了一个结构体:

type Book struct {Title string              // 书名Pages int                 // 书的页数Indexes int               // 书的索引
}

这样我们就拥有了一个Book的结构体类型,它有TitlePagesIndexes三个字段,分别表示书名、页数和索引。这样我们使用这个Book结构体就能够很方便的在程序中表示和存储书信息了。

	var book Bookbook.Title = "The Go Programming Language"book.Pages = 800

我们通过.来访问结构体的字段(成员变量),例如book.Titlebook.Pages等。

2.3 定义一个空结构体

2.3.1 空结构体介绍

空结构体,**也就是没有包含任何字段的结构体类型,**就像下面示例代码这样:

type Empty struct{} // Empty是一个不包含任何字段的空结构体类型
2.3.2 空结构体作用

空结构体类型在Go语言中有一些特殊的用途,尽管它们不包含任何字段,但由于其特殊性质,可以用于以下几种情况:

  1. 占位符:空结构体可以用作占位符,表示某个位置需要一个值,但实际上并不需要存储任何数据。这在某些数据结构中很有用,例如,用于实现集合(set)或映射(map)的数据结构,你可能只关心元素的存在而不关心元素的值。使用空结构体可以在占用很少内存的同时实现这一目标。
  2. 通道信号:在并发编程中,空结构体常常用于通道(channel)的信号传递。你可以在通道中发送一个空结构体来表示某种信号,而不必发送任何实际的数据。这在同步和通信方面非常有用。
  3. 数据占位符:有时候,某些数据结构需要占据某个位置,但不需要实际的数据,此时可以使用空结构体。
  4. 函数参数:如果一个函数需要传递一个不需要使用的参数,你可以将其定义为空结构体类型,以避免引入不必要的数据。

尽管空结构体类型没有实际的数据存储,但由于其在内存中的大小为0字节,因此使用空结构体可以节省内存,特别是在需要大量实例的情况下,这种节省可以显著减少内存开销。

我们来看下面示例代码这样:

var s Empty
println(unsafe.Sizeof(s)) // 0

我们看到,输出的空结构体类型变量的大小为 0,也就是说,空结构体类型变量的内存占用为 0。基于空结构体类型内存零开销这样的特性,我们在日常 Go 开发中会经常使用空结构体类型元素,作为一种“事件”信息进行 Goroutine 之间的通信,就像下面示例代码这样:

var c = make(chan Empty) // 声明一个元素类型为Empty的channel
c<-Empty{}               // 向channel写入一个“事件”

这种以空结构体为元素类建立的 channel,是目前能实现的、内存占用最小的 Goroutine 间通信方式。

2.4 使用其他结构体作为自定义结构体中字段的类型

2.4.1 结构体嵌套介绍
  • 结构体嵌套是指在一个结构体内部嵌入另一个结构体,将一个结构体类型的字段嵌套到另一个结构体中,以创建复合结构体。

比如,这段代码,这里结构体类型 Book 的字段 Author 的类型,就是另外一个结构体类型 Person:

type Person struct {Name stringPhone stringAddr string
}type Book struct {Title stringAuthor Person... ...
}

如果我们要访问 Book 结构体字段 Author 中的 Phone 字段,我们可以这样操作:

var book Book 
println(book.Author.Phone)

同时,Go 还提供了一种更为简便的定义方法,那就是我们可以无需提供字段的名字,只需要使用其类型就可以了,以上面的 Book 结构体定义为例,我们可以用下面的方式提供一个等价的定义:

type Book struct {Title stringPerson... ...
}

以这种方式定义的结构体字段,我们叫做嵌入字段(Embedded Field)。我们也可以将这种字段称为匿名字段,或者把类型名看作是这个字段的名字。如果我们要访问 Person 中的 Phone 字段,我们可以通过下面两种方式进行:

var book Book 
println(book.Person.Phone) // 将类型名当作嵌入字段的名字
println(book.Phone)        // 支持直接访问嵌入字段所属类型中字段

第一种方式显然是通过把类型名当作嵌入字段的名字来进行操作的,而第二种方式更像是一种“语法糖”,我们可以“绕过”Person 类型这一层,直接访问 Person 中的字段。

Go 语言不支持这种在结构体类型定义中,递归地放入其自身类型字段的定义方式。面。比如下面这段代码:

type T struct {t T  ... ...
}

面对上面的示例代码,编译器就会给出“invalid recursive type T”的错误信息。

不过,虽然我们不能在结构体类型 T 定义中,拥有以自身类型 T 定义的字段,但是你可以在结构体中包含自身类型的指针、切片和map类型的字段,这是允许的。这种方式通常用于构建递归数据结构或树状结构。比如这样:

type T struct {t  *T           // okst []T          // okm  map[string]T // ok
}     

这是因为指针、切片和map这些类型并没有真正包含自身类型T,只是间接引用了结构体类型 T,所以不会造成递归定义的问题。

2.4.2 结构体嵌套的好处

使用其他结构体作为字段类型的好处包括:

  1. 模块化:将数据模型分解成更小的结构体,每个结构体关注特定的数据和功能,有助于提高代码的模块化性。
  2. 可读性:结构体嵌套提供了更自然的数据组织方式,使代码更容易阅读和理解。
  3. 代码复用:可以重复使用现有的结构体,避免在不同地方定义相同的字段。
  4. 扩展性:你可以轻松地添加、修改或扩展数据模型,而无需对所有相关代码进行大规模更改。

2.5 匿名结构体

在定义一些临时数据结构等场景下还可以使用匿名结构体。

package mainimport ("fmt"
)func main() {var user struct{Name string; Age int}user.Name = "Jarvis"user.Age = 18fmt.Printf("%#v\n", user)
}

三、结构体变量的声明与初始化

3.1 结构体变量的声明

只有当结构体实例化时,才会真正地分配内存。也就是必须实例化后才能使用结构体的字段。

可以使用标准变量声明语句,或者是短变量声明语句声明一个结构体类型的变量:

type Book struct {...
}var book Book // 标准变量声明
var book = Book{}
book := Book{} //短变量声明

3.2 零值初始化

零值初始化说的是使用结构体的零值作为它的初始值。“零值”指的是一个类型的默认值。对于 Go 原生类型来说,这个默认值也称为零值。Go 结构体类型由若干个字段组成,当这个结构体类型变量的各个字段的值都是零值时,我们就说这个结构体类型变量处于零值状态。

**结构体类型的变量通常都要被赋予适当的初始值后,才会有合理的意义。**通常不具有或者很难具有合理的意义,比如通过下面代码得到的零值 book 变量就是这样:

var book Book // book为零值结构体变量

想象一下,一本书既没有书名,也没有作者、页数、索引等信息,那么通过 Book 类型对这本书的抽象就失去了实际价值。所以对于像 Book 这样的结构体类型,使用零值初始化并不是正确的选择。

那么采用零值初始化的零值结构体变量就真的没有任何价值了吗?恰恰相反。如果一种类型采用零值初始化得到的零值变量,是有意义的,而且是直接可用的,这种类型为“零值可用”类型。可以说,定义零值可用类型是简化代码、改善开发者使用体验的一种重要的手段

在 Go 语言标准库和运行时的代码中,有很多践行“零值可用”理念的好例子,最典型的莫过于 sync 包的 Mutex 类型了。Mutex 是 Go 标准库中提供的、用于多个并发 Goroutine 之间进行同步的互斥锁。我们来看下下面这段代码:

var mu sync.Mutex
mu.Lock()
mu.Unlock()

Go 标准库的设计者很贴心地将 sync.Mutex 结构体的零值状态,设计为可用状态,这样开发者便可直接基于零值状态下的 Mutex 进行 lock 与 unlock 操作,而且不需要额外显式地对它进行初始化操作了。

Go 标准库中的 bytes.Buffer 结构体类型,也是一个零值可用类型的典型例子,这里我演示了 bytes.Buffer 类型的常规用法:

var b bytes.Buffer
b.Write([]byte("Hello, Go"))
fmt.Println(b.String()) // 输出:Hello, Go

可以看到,我们不需要对 bytes.Buffer 类型的变量 b 进行任何显式初始化,就可以直接通过处于零值状态的变量 b,调用它的方法进行写入和读取操作。

3.3 使用顺序复合字面值初始化

对于我们前面的 Book 类型,确实不能设计为零值可用类型,因为它们的零值并非有效值。对于这类类型,我们需要对它的变量进行显式的初始化后,才能正确使用。在日常开发中,对结构体类型变量进行显式初始化的最常用方法就是使用复合字面值。

最简单的对结构体变量进行显式初始化的方式,就是按顺序依次给每个结构体字段进行赋值,比如下面的代码:

	type Book struct {Title string              // 书名Pages int                 // 书的页数Indexes int               // 书的索引}var book = Book{"The Go Programming Language", 700, 9}

我们依然可以用这种方法给结构体的每一个字段依次赋值,但这种方法也有很多问题:

  1. 当结构体类型定义中的字段顺序发生变化,或者字段出现增删操作时,我们就需要手动调整该结构体类型变量的显式初始化代码,让赋值顺序与调整后的字段顺序一致。
  2. 当一个结构体的字段较多时,这种逐一字段赋值的方式实施起来就会比较困难,而且容易出错,开发人员需要来回对照结构体类型中字段的类型与顺序,谨慎编写字面值表达式。
  3. 一旦结构体中包含非导出字段,那么这种逐一字段赋值的方式就不再被支持了,编译器会报错:
type T struct {F1 intF2 stringf3 intF4 intF5 int
}var t = T{11, "hello", 13} // 错误:implicit assignment of unexported field 'f3' in T literalvar t = T{11, "hello", 13, 14, 15} // 错误:implicit assignment of unexported field 'f3' in T literal

事实上,Go语言并不推荐我们按照字段顺序对一个结构体变量进行显示初始化。而是要带上字段名,甚至 Go 官方还在提供的 go vet 工具中专门内置了一条检查规则:“composites”,用来静态检查代码中结构体变量初始化是否使用了这种方法,一旦发现,就会给出警告。

那么我们应该用哪种形式的复合字面值给结构体变量赋初值呢?

3.4 使用键值对初始化

  • 使用键值对对结构体进行初始化时,键对应结构体的字段,值对应该字段的初始值。

Go 推荐我们用“field:value”形式的复合字面值,对结构体类型变量进行显式初始化,**这种方式可以降低结构体类型使用者和结构体类型设计者之间的耦合,**这也是 Go 语言的惯用法。这里,我们用“field:value”形式复合字面值,对上面的类型 T 的变量进行初始化看看:

var t = T{F2: "hello",F1: 11,F4: 14,
}

我们看到,使用这种“field:value”形式的复合字面值对结构体类型变量进行初始化,非常灵活。和之前的顺序复合字面值形式相比,“field:value”形式字面值中的字段可以以任意次序出现。未显式出现在字面值中的结构体字段(比如上面例子中的 F5)将采用它对应类型的零值。

复合字面值作为结构体类型变量初值被广泛使用,即便结构体采用类型零值时,我们也会使用复合字面值的形式:

t := T{}

而比较少使用 new 这一个 Go 预定义的函数来创建结构体变量实例:

tp := new(T)

这里值得我们注意的是,我们不能用从其他包导入的结构体中的未导出字段,来作为复合字面值中的 field。这会导致编译错误,因为未导出字段是不可见的。

那么,如果一个结构体类型中包含未导出字段,并且这个字段的零值还不可用时,我们要如何初始化这个结构体类型的变量呢?又或是一个结构体类型中的某些字段,需要一个复杂的初始化逻辑,我们又该怎么做呢?这时我们就需要使用一个特定的构造函数,来创建并初始化结构体变量了。

3.5 使用特定的构造函数

通过专用构造函数进行结构体类型变量创建、初始化大多都符合这种模式:

func NewT(field1, field2, ...) *T {... ...
}

这里,NewT 是结构体类型 T 的专用构造函数,它的参数列表中的参数通常与 T 定义中的导出字段相对应,返回值则是一个 T 指针类型的变量。T 的非导出字段在 NewT 内部进行初始化,一些需要复杂初始化逻辑的字段也会在 NewT 内部完成初始化。这样,我们只要调用 NewT 函数就可以得到一个可用的 T 指针类型变量了。

其实,使用特定的构造函数创建并初始化结构体变量的例子,并不罕见。在 Go 标准库中就有很多,其中 time.Timer 这个结构体就是一个典型的例子,它的定义如下:

// $GOROOT/src/time/sleep.go
type runtimeTimer struct {pp       uintptrwhen     int64period   int64f        func(interface{}, uintptr) arg      interface{}seq      uintptrnextwhen int64status   uint32
}type Timer struct {C <-chan Timer runtimeTimer
}

我们看到,Timer 结构体中包含了一个非导出字段 r,r 的类型为另外一个结构体类型 runtimeTimer。这个结构体更为复杂,而且我们一眼就可以看出来,这个 runtimeTimer 结构体不是零值可用的,那我们在创建一个 Timer 类型变量时就没法使用显式复合字面值的方式了。这个时候,Go 标准库提供了一个 Timer 结构体专用的构造函数 NewTimer,它的实现如下:

// $GOROOT/src/time/sleep.go
func NewTimer(d Duration) *Timer {c := make(chan Time, 1)t := &Timer{C: c,r: runtimeTimer{when: when(d),f:    sendTime,arg:  c,},}startTimer(&t.r)return t
}

我们看到,NewTimer 这个函数只接受一个表示定时时间的参数 d,在经过一个复杂的初始化过程后,它返回了一个处于可用状态的 Timer 类型指针实例。

四、结构体类型的内存布局

Go 结构体类型是既数组类型之后,第二个将它的元素(结构体字段)一个接着一个以“平铺”形式,存放在一个连续内存块中的。下图是一个结构体类型 T 的内存布局:

WechatIMG183

我们看到,结构体类型 T 在内存中布局是非常紧凑的,Go 为它分配的内存都用来存储字段了,没有被 Go 编译器插入的额外字段。我们可以借助标准库 unsafe 包提供的函数,获得结构体类型变量占用的内存大小,以及它每个字段在内存中相对于结构体变量起始地址的偏移量:

var t T
unsafe.Sizeof(t)      // 结构体类型变量占用的内存大小
unsafe.Offsetof(t.Fn) // 字段Fn在内存中相对于变量t起始地址的偏移量

不过,上面这张示意图是比较理想的状态,真实的情况可能就没那么好了:

WechatIMG184

在真实情况下,虽然 Go 编译器没有在结构体变量占用的内存空间中插入额外字段,但结构体字段实际上可能并不是紧密相连的,中间可能存在“缝隙”。这些“缝隙”同样是结构体变量占用的内存空间的一部分,它们是 Go 编译器插入的“填充物(Padding)”。

那么,Go 编译器为什么要在结构体的字段间插入“填充物”呢?这其实是内存对齐的要求。所谓内存对齐,指的就是各种内存对象的内存地址不是随意确定的,必须满足特定要求。

对于各种基本数据类型来说,它的变量的内存地址值必须是其类型本身大小的整数倍,比如,一个 int64 类型的变量的内存地址,应该能被 int64 类型自身的大小,也就是 8 整除;一个 uint16 类型的变量的内存地址,应该能被 uint16 类型自身的大小,也就是 2 整除。

这些基本数据类型的对齐要求很好理解,那么像结构体类型这样的复合数据类型,内存对齐又是怎么要求的呢?是不是它的内存地址也必须是它类型大小的整数倍呢?

实际上没有这么严格。对于结构体而言,它的变量的内存地址,只要是它最长字段长度与系统对齐系数两者之间较小的那个的整数倍就可以了。但对于结构体类型来说,我们还要让它每个字段的内存地址都严格满足内存对齐要求。

这么说依然比较绕,我们来看一个具体例子,计算一下这个结构体类型 T 的对齐系数:

type T struct {b bytei int64u uint16
}

计算过程是这样的:

WechatIMG185

我们简单分析一下,整个计算过程分为两个阶段。第一个阶段是对齐结构体的各个字段。

首先,我们看第一个字段 b 是长度 1 个字节的 byte 类型变量,这样字段 b 放在任意地址上都可以被 1 整除,所以我们说它是天生对齐的。我们用一个 sum 来表示当前已经对齐的内存空间的大小,这个时候 sum=1;

接下来,我们看第二个字段 i,它是一个长度为 8 个字节的 int64 类型变量。按照内存对齐要求,它应该被放在可以被 8 整除的地址上。但是,如果把 i 紧邻 b 进行分配,当 i 的地址可以被 8 整除时,b 的地址就无法被 8 整除。这个时候,我们需要在 b 与 i 之间做一些填充,使得 i 的地址可以被 8 整除时,b 的地址也始终可以被 8 整除,于是我们在 i 与 b 之间填充了 7 个字节,此时此刻 sum=1+7+8;

再下来,我们看第三个字段 u,它是一个长度为 2 个字节的 uint16 类型变量,按照内存对其要求,它应该被放在可以被 2 整除的地址上。有了对其的 i 作为基础,我们现在知道将 u 与 i 相邻而放,是可以满足其地址的对齐要求的。i 之后的那个字节的地址肯定可以被 8 整除,也一定可以被 2 整除。于是我们把 u 直接放在 i 的后面,中间不需要填充,此时此刻,sum=1+7+8+2。

现在结构体 T 的所有字段都已经对齐了,我们开始第二个阶段,也就是对齐整个结构体。

我们前面提到过,结构体的内存地址为 min(结构体最长字段的长度,系统内存对齐系数)的整数倍,那么这里结构体 T 最长字段为 i,它的长度为 8,而 64bit 系统上的系统内存对齐系数一般为 8,两者相同,我们取 8 就可以了。那么整个结构体的对齐系数就是 8。

这个时候问题就来了!为什么上面的示意图还要在结构体的尾部填充了 6 个字节呢?

我们说过结构体 T 的对齐系数是 8,那么我们就要保证每个结构体 T 的变量的内存地址,都能被 8 整除。如果我们只分配一个 T 类型变量,不再继续填充,也可能保证其内存地址为 8 的倍数。但如果考虑我们分配的是一个元素为 T 类型的数组,比如下面这行代码,我们虽然可以保证 T[0]这个元素地址可以被 8 整除,但能保证 T[1]的地址也可以被 8 整除吗?

var array [10]T

我们知道,数组是元素连续存储的一种类型,元素 T[1]的地址为 T[0]地址 +T 的大小 (18),显然无法被 8 整除,这将导致 T[1]及后续元素的地址都无法对齐,这显然不能满足内存对齐的要求。

问题的根源在哪里呢?问题就在于 T 的当前大小为 18,这是一个不能被 8 整除的数值,如果 T 的大小可以被 8 整除,那问题就解决了。于是我们才有了最后一个步骤,我们从 18 开始向后找到第一个可以被 8 整除的数字,也就是将 18 圆整到 8 的倍数上,我们得到 24,我们将 24 作为类型 T 最终的大小就可以了。

为什么会出现内存对齐的要求呢?这是出于对处理器存取数据效率的考虑。在早期的一些处理器中,比如 Sun 公司的 Sparc 处理器仅支持内存对齐的地址,如果它遇到没有对齐的内存地址,会引发段错误,导致程序崩溃。我们常见的 x86-64 架构处理器虽然处理未对齐的内存地址不会出现段错误,但数据的存取性能也会受到影响。

从这个推演过程中,你应该已经知道了,Go 语言中结构体类型的大小受内存对齐约束的影响。这样一来,不同的字段排列顺序也会影响到“填充字节”的多少,从而影响到整个结构体大小。比如下面两个结构体类型表示的抽象是相同的,但正是因为字段排列顺序不同,导致它们的大小也不同:

type T struct {b bytei int64u uint16
}type S struct {b byteu uint16i int64
}func main() {var t Tprintln(unsafe.Sizeof(t)) // 24var s Sprintln(unsafe.Sizeof(s)) // 16
}

所以,你在日常定义结构体时,一定要注意结构体中字段顺序,尽量合理排序,降低结构体对内存空间的占用。

另外,前面例子中的内存填充部分,是由编译器自动完成的。不过,有些时候,为了保证某个字段的内存地址有更为严格的约束,我们也会做主动填充。比如 runtime 包中的 mstats 结构体定义就采用了主动填充:

// $GOROOT/src/runtime/mstats.go
type mstats struct {... ...// Add an uint32 for even number of size classes to align below fields// to 64 bits for atomic operations on 32 bit platforms._ [1 - _NumSizeClasses%2]uint32 // 这里做了主动填充last_gc_nanotime uint64 // last gc (monotonic time)last_heap_inuse  uint64 // heap_inuse at mark termination of the previous GC... ...
}

通常我们会通过空标识符来进行主动填充,因为填充的这部分内容我们并不关心。

五、结构体字段的可见性

结构体中字段大写开头表示可公开访问,小写表示私有(仅在定义当前结构体的包中可访问)。

六、结构体的“继承”

Go语言中使用结构体也可以实现其他编程语言中面向对象的继承。

//Animal 动物
type Animal struct {name string
}func (a *Animal) move() {fmt.Printf("%s会动!\n", a.name)
}//Dog 狗
type Dog struct {Feet    int8*Animal //通过嵌套匿名结构体实现继承
}func (d *Dog) wang() {fmt.Printf("%s会汪汪汪~\n", d.name)
}func main() {d1 := &Dog{Feet: 4,Animal: &Animal{ //注意嵌套的是结构体指针name: "乐乐",},}d1.wang() //乐乐会汪汪汪~d1.move() //乐乐会动!
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/95859.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

day-65 代码随想录算法训练营(19)图论 part 04

463.岛屿的周长 分析&#xff1a; 1.陆地的旁边是海面&#xff0c;存在周长2.陆地在边界上&#xff0c;存在周长 思路一&#xff1a;深度优先遍历 1.通过记录访问情况来访问数据 class Solution { public:int direct[4][2]{{0,1},{0,-1},{1,0},{-1,0}};int res0;void dfs(…

集群服务器

文章目录 项目名:实现集群服务器技术栈通过这项目你学到(或者复习到)实现功能编码环境json环境muduo库boost库MySql数据库登录mysql&#xff1a;查看mysql服务开启了没有&#xff1f;mysql的服务器及开发包库chat&#xff0c;表 allgroup friend groupuser offlinemessage user…

从零开始学习线性回归:理论、实践与PyTorch实现

文章目录 &#x1f966;介绍&#x1f966;基本知识&#x1f966;代码实现&#x1f966;完整代码&#x1f966;总结 &#x1f966;介绍 线性回归是统计学和机器学习中最简单而强大的算法之一&#xff0c;用于建模和预测连续性数值输出与输入特征之间的关系。本博客将深入探讨线性…

Mac上protobuf环境构建-java

参考文献 getting-started 官网pb java介绍 maven protobuf插件 简单入门1 简单入门2 1. protoc编译器下载安装 https://github.com/protocolbuffers/protobuf/releases?page10 放入.zshrc中配置环境变量  ~/IdeaProjects/test2/ protoc --version libprotoc 3.12.1  …

Javascript文件上传

什么是文件上传 文件上传包含两部分&#xff0c; 一部分是选择文件&#xff0c;包含所有相关的界面交互。一部分是网络传输&#xff0c;通过一个网络请求&#xff0c;将文件的数据携带过去&#xff0c;传递到服务器中&#xff0c;剩下的&#xff0c;在服务器中如何存储&#xf…

基于SpringBoot的电影评论网站

目录 前言 一、技术栈 二、系统功能介绍 电影信息管理 电影评论回复 电影信息 用户注册 三、核心代码 1、登录模块 2、文件上传模块 3、代码封装 前言 随着信息技术在管理上越来越深入而广泛的应用&#xff0c;管理信息系统的实施在技术上已逐步成熟。本文介绍了电影评…

Pyhon-每日一练(1)

&#x1f308;write in front&#x1f308; &#x1f9f8;大家好&#xff0c;我是Aileen&#x1f9f8;.希望你看完之后&#xff0c;能对你有所帮助&#xff0c;不足请指正&#xff01;共同学习交流. &#x1f194;本文由Aileen_0v0&#x1f9f8; 原创 CSDN首发&#x1f412; 如…

Vue中如何进行分布式路由配置与管理

Vue中的分布式路由配置与管理 随着现代Web应用程序的复杂性不断增加&#xff0c;分布式路由配置和管理成为了一个重要的主题。Vue.js作为一种流行的前端框架&#xff0c;提供了多种方法来管理Vue应用程序的路由。本文将深入探讨在Vue中如何进行分布式路由配置与管理&#xff0…

Spring Boot中的@Controller使用教程

一 Controller使用方法&#xff0c;如下所示&#xff1a; Controller是SpringBoot里最基本的组件&#xff0c;他的作用是把用户提交来的请求通过对URL的匹配&#xff0c;分配个不同的接收器&#xff0c;再进行处理&#xff0c;然后向用户返回结果。下面通过本文给大家介绍Spr…

Java方法:重复使用的操作可以写成方法哦

&#x1f451;专栏内容&#xff1a;Java⛪个人主页&#xff1a;子夜的星的主页&#x1f495;座右铭&#xff1a;前路未远&#xff0c;步履不停 目录 一、方法的概念1、什么是方法&#xff1f;2、方法的定义3、方法调用的过程 二、方法重载1、重载的概念2、方法签名 在日常生活中…

计算机网络 快速了解网络层次、常用协议、常见物理设备。 掌握程序员必备网络基础知识!!!

文章目录 0 引言1 基础知识的定义1.1 计算机网络层次1.2 网络供应商 ISP1.3 猫、路由器、交换机1.4 IP协议1.5 TCP、UDP协议1.6 HTTP、HTTPS、FTP协议1.7 Web、Web浏览器、Web服务器1.8 以太网和WLAN1.9 Socket &#xff08;网络套接字&#xff09; 2 总结 0 引言 在学习的过程…

【Java-LangChain:使用 ChatGPT API 搭建系统-2】语言模型,提问范式与 Token

第二章 语言模型&#xff0c;提问范式与 Token 在本章中&#xff0c;我们将和您分享大型语言模型&#xff08;LLM&#xff09;的工作原理、训练方式以及分词器&#xff08;tokenizer&#xff09;等细节对 LLM 输出的影响。我们还将介绍 LLM 的提问范式&#xff08;chat format…

postgresql-物化视图

postgresql-物化视图 物化视图创建物化视图刷新物化视图修改物化视图删除物化视图 物化视图 创建物化视图 postgresql使用create materialized view 语句创建视图 create materialized view if not exists name as query [with [NO] data];-- 创建一个包含员工统计信息的物化…

java遇到的问题

java遇到的问题 Tomcat与JDK版本问题 当使用Tomcat10的版本用于springmvc借用浏览器调试时&#xff0c;使用JDK8浏览器会报异常。 需要JDK17&#xff08;可以配置多个JDK环境&#xff0c;切换使用&#xff09;才可以使用&#xff0c;配置为JAVA_HOME路径&#xff0c;否则&a…

【AI视野·今日Robot 机器人论文速览 第四十七期】Wed, 4 Oct 2023

AI视野今日CS.Robotics 机器人学论文速览 Wed, 4 Oct 2023 Totally 40 papers &#x1f449;上期速览✈更多精彩请移步主页 Interesting: &#x1f4da;基于神经网络的多模态触觉感知, classification, position, posture, and force of the grasped object多模态形象的解耦(f…

苹果系统_安装matplotlib__pygame,以pycharm导入模块

为了更便捷、连贯的进行python编程学习&#xff0c;尽量在开始安装python软件时&#xff0c;将编辑器、模块一并安装好&#xff0c;这样能避免以后版本冲突的问题。小白在开始安装pycharm、pip、matplotlib往往会遇到一些问题&#xff0c;文中列示其中部分bug&#xff0c;供大家…

VL53L5CX驱动开发(1)----驱动TOF进行区域检测

VL53L5CX驱动开发----1.驱动TOF进行区域检测 闪烁定义视频教学样品申请源码下载主要特点硬件准备技术规格系统框图应用示意图区域映射生成STM32CUBEMX选择MCU 串口配置IIC配置X-CUBE-TOF1串口重定向代码配置Tera Term配置演示结果 闪烁定义 VL53L5CX是一款先进的飞行感应&…

Reset信号如何同步?

首先来复习一个更加基础的概念&#xff1a;同步reset和异步reset。 同步reset&#xff08;synchronous reset&#xff09;是说&#xff0c;当reset信号为active的时候&#xff0c;寄存器在下一个时钟沿到来后被复位&#xff0c;时钟沿到来之前寄存器还是保持其之前的值。 异步…

xxl-job的原理(2)—调度中心管理注册信息

一、调度中心管理注册信息 1.JobApiController 执行器调用调度中心的url来实现注册、下线、回调等操作&#xff1b;其主要的实现类是JobApiController&#xff0c;调用/api/registry接口注册执行器信息&#xff0c;调用/api/registryRemove接口下线执行器信息&#xff0c;调用…

操作系统和进程相关的认识

目录 冯诺依曼体系结构 冯诺依曼体系结构五大组成部分 为什么数据只能通过存储器进行输入和输出 操作系统 概念一&#xff1a;访问操作系统的请求都是通过系统调用完成的 操作系统如何管理用户信息 概念二&#xff1a;先描述&#xff0c;再组织。 进程的概念 在认识进行相关的知…