如何在Go语言中正确初始化切片中的结构体,而不使用构造函数。

huangapple go评论104阅读模式
英文:

How to properly init structs in Go inside slices without constructors

问题

Go语言没有构造函数,所以我想知道如何在切片中正确初始化结构体。我很难相信答案会是初始化并复制所有结构体两次。

这是一个示例,其中队列的缓冲区必须是2的幂。

  1. package main
  2. import "fmt"
  3. // minQueueLen is smallest capacity that queue may have.
  4. // Must be power of 2 for bitwise modulus: x % n == x & (n - 1).
  5. const minQueueLen = 16
  6. type Queue[T any] struct {
  7. buf []T
  8. }
  9. func New[T any]() *Queue[T] {
  10. return &Queue[T]{
  11. buf: make([]T, minQueueLen),
  12. }
  13. }
  14. type SomeStruct struct {
  15. q Queue[int]
  16. }
  17. func main() {
  18. someSlice := make([]SomeStruct, 10)
  19. // 现在buf的大小是错误的,因为我们没有使用正确的构造函数进行初始化。
  20. // 初始化这个结构体两次(成千上万次)将非常浪费资源。
  21. fmt.Println("Size of a buf: ", len(someSlice[0].q.buf))
  22. }

以上是一个示例,其中队列的缓冲区必须是2的幂。

英文:

Go doesn't have constructors so I'm wondering how you would properly initialize a struct inside a slice. I have trouble believing that the answer would be to initialize and copy all the structs twice?

  1. package main
  2. import "fmt"
  3. // minQueueLen is smallest capacity that queue may have.
  4. // Must be power of 2 for bitwise modulus: x % n == x & (n - 1).
  5. const minQueueLen = 16
  6. type Queue[T any] struct {
  7. buf []T
  8. }
  9. func New[T any]() *Queue[T] {
  10. return &Queue[T]{
  11. buf: make([]T, minQueueLen),
  12. }
  13. }
  14. type SomeStruct struct {
  15. q Queue[int]
  16. }
  17. func main() {
  18. someSlice := make([]SomeStruct, 10)
  19. // Now buf is the wrong size because we didn't
  20. // get to init it with the proper constructor.
  21. // It would be very wasteful to initialize this
  22. // struct twice (100s of thousands or more).
  23. fmt.Println("Size of a buf: ", len(someSlice[0].q.buf))
  24. }

Here is an example where the buffer of the queue must be a power of 2.

答案1

得分: 0

你从未真正初始化它。分配一个切片只是分配了空间,但并没有初始化各个元素。一旦你有了切片,你可以遍历它并进行初始化:

  1. func main() {
  2. someSlice := make([]SomeStruct, 10)
  3. for i := range someSlice {
  4. someSlice[i].q = *New[int]()
  5. }
  6. fmt.Println("Size of a buf:", len(someSlice[0].q.buf))
  7. }

你也可以这样做:

  1. func (s *SomeStruct) Init() {
  2. s.q = Queue[int]{
  3. buf: make([]int, minQueueLen),
  4. }
  5. }
  6. func main() {
  7. someSlice := make([]SomeStruct, 10)
  8. for i := range someSlice {
  9. someSlice[i].Init()
  10. }
  11. fmt.Println("Size of a buf:", len(someSlice[0].q.buf))
  12. }
英文:

You never really initialized it. Allocating a slice simply allocates the space, but does not initialize individual elements. Once you have the slice, you can loop through it and inialize:

  1. func main() {
  2. someSlice := make([]SomeStruct, 10)
  3. for i:=range someSlice {
  4. someSlice[i].q=*New[int]()
  5. }
  6. fmt.Println("Size of a buf: ", len(someSlice[0].q.buf))
  7. }

You can also do:

  1. func (s *SomeStruct) Init() {
  2. s.q=Queue[int]{
  3. buf: make([]int, minQueueLen),
  4. }
  5. }
  6. func main() {
  7. someSlice := make([]SomeStruct, 10)
  8. for i:=range someSlice {
  9. someSlice[i].Init()
  10. }
  11. fmt.Println("Size of a buf: ", len(someSlice[0].q.buf))
  12. }

huangapple
  • 本文由 发表于 2022年3月3日 11:01:26
  • 转载请务必保留本文链接:https://go.coder-hub.com/71331633.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定