go切片和指针切片示例详解

 更新时间:2024年04月15日 09:00:35   作者:香吧香  
在Go语言中,切片(Slice)和指针的切片(即切片中每个元素都是指向某种数据类型的指针)是两个不同的概念,它们各自具有特定的用途和优势,这篇文章主要介绍了go切片和指针切片,需要的朋友可以参考下

在Go语言中,切片(Slice)和指针的切片(即切片中每个元素都是指向某种数据类型的指针)是两个不同的概念,它们各自具有特定的用途和优势。

切片(Slice)

  切片是对数组的一个连续片段的引用,它提供了对数组元素集合的抽象表示。切片底层数据结构都是数组,它包含三个关键部分:指向数组起始元素的指针、切片长度和切片容量。切片长度是指切片当前包含的元素个数,而切片容量是指从切片的起始元素到底层数组的最后一个元素的个数。

  切片的一个重要特性是它提供了对底层数组的动态视图。这意味着你可以通过切片来访问、修改和操作数组的一部分或全部元素,而无需复制整个数组。此外,切片还可以方便地进行扩容和缩容操作,以满足不同场景下的需求。

  示例

package main  
import "fmt"  
func main() {  
    // 定义一个数组  
    array := [5]int{1, 2, 3, 4, 5}  
    // 创建一个切片,引用数组的前三个元素  
    slice := array[:3]  
    // 打印切片元素  
    fmt.Println(slice) // 输出: [1 2 3]  
    // 修改切片元素,也会修改底层数组对应位置的元素  
    slice[1] = 100  
    fmt.Println(array) // 输出: [1 100 3 4 5]  
}

指针的切片

  指针的切片是一个切片,但其元素是指针,指向某种数据类型的实例。这意味着每个元素都是一个地址,通过这个地址可以间接访问和操作实际的数据。指针的切片常用于需要存储大量数据且希望避免数据复制的场景,或者当需要在切片中存储可变大小的对象时。

  使用指针的切片可以节省内存空间,因为只需要存储指针而不是实际的数据。同时,通过指针可以方便地修改原始数据。然而,这也带来了额外的复杂性和风险,因为需要小心处理指针的解引用和内存管理。

  示例

package main  
import "fmt"  
type Person struct {  
    Name string  
    Age  int  
}  
func main() {  
    // 创建几个Person实例的指针  
    person1 := &Person{Name: "Alice", Age: 30}  
    person2 := &Person{Name: "Bob", Age: 25}  
    // 创建一个指针的切片,存储这些指针  
    people := []*Person{person1, person2}  
    // 通过指针修改Person实例的属性  
    people[0].Age = 31  
    // 打印修改后的Person实例  
    fmt.Println(people[0].Name, people[0].Age) // 输出: Alice 31  
}

结构体切片与结构体指针切片的区别

  • 内存占用:

    • 结构体切片:每个元素都是结构体的一个完整副本,因此内存占用较大。

    • 结构体指针切片:每个元素只是一个指向结构体的指针,内存占用较小。但是,这并不意味着整体内存占用会小,因为还需要考虑实际结构体对象所占用的内存。

  • 修改元素:

    • 结构体切片:修改切片中的一个元素,将直接修改该元素的值,不会影响其他切片或原始结构体对象。

    • 结构体指针切片:修改切片中的一个指针指向的元素,将影响所有指向该元素的指针。如果多个切片或变量指向同一个结构体对象,修改该对象的内容将影响所有引用。

  • 初始化与赋值:

    • 结构体切片:可以直接初始化并赋值。

    • 结构体指针切片:需要先初始化结构体对象,然后将对象的地址赋值给切片。

  • nil与空切片:

    • 对于结构体指针切片,nil切片和空切片(长度为0的切片)是不同的。nil切片没有分配底层数组,而空切片分配了底层数组但长度为0。

    • 对于结构体切片,通常不会讨论nil切片,因为切片总是与底层数组相关联。

能否直接遍历结构体切片并赋值给结构体指针切片

  不能直接遍历结构体切片并赋值给结构体指针切片。因为结构体切片中的元素是结构体的值,而结构体指针切片中的元素是指向结构体的指针。需要遍历结构体切片,并分别为每个元素创建指针,然后将这些指针添加到结构体指针切片中。

示例说明

  假设我们有一个Person结构体:

type Person struct {  
    Name string  
    Age  int  
}

  结构体切片的使用

// 创建并初始化一个Person结构体切片  
peopleSlice := []Person{  
    {"Alice", 30},  
    {"Bob", 25},  
}  
// 修改切片中的一个元素  
peopleSlice[0].Age = 31  
// 遍历并打印切片中的元素  
for _, person := range peopleSlice {  
    fmt.Println(person.Name, person.Age)  
}

结构体指针切片的使用

// 创建并初始化一些Person结构体对象  
alice := Person{"Alice", 30}  
bob := Person{"Bob", 25}  
// 创建一个Person指针切片,并将结构体对象的地址添加到切片中  
peoplePtrSlice := []*Person{&alice, &bob}  
// 修改切片中的一个指针指向的元素  
peoplePtrSlice[0].Age = 31 // 这将改变alice的年龄,因为peoplePtrSlice[0]指向alice  
// 遍历并打印切片中的元素(通过解引用指针)  
for _, personPtr := range peoplePtrSlice {  
    fmt.Println(personPtr.Name, personPtr.Age)  
}  
// 如果想要遍历结构体切片并赋值给结构体指针切片,你需要这样做:  
peopleSlice := []Person{  
    {"Charlie", 28},  
    {"David", 35},  
}  
// 初始化一个空的Person指针切片,长度与peopleSlice相同  
peoplePtrSlice = make([]*Person, len(peopleSlice))  
// 遍历peopleSlice,为peoplePtrSlice分配新的指针  
for i, person := range peopleSlice {  
    // 创建person的一个副本,并获取其地址,然后赋值给peoplePtrSlice  
    peoplePtrSlice[i] = &Person{Name: person.Name, Age: person.Age}  
}  
// 现在peoplePtrSlice包含了指向peopleSlice中元素副本的指针

  在上面的示例中,peoplePtrSlice是通过遍历peopleSlice并创建每个元素的副本的地址来初始化的。如果你想要peoplePtrSlice中的指针指向peopleSlice中的相同对象(而不是副本),你需要确保这些对象是通过new函数或通过&操作符在堆上分配的,并且它们的地址被添加到peoplePtrSlice中。但是,在大多数情况下,你可能不希望这样做,因为这会导致切片之间共享相同的对象,从而可能引起意外的副作用。

能否直接遍历结构体指针切片并赋值给结构体切片

  不能直接遍历结构体指针切片并赋值给结构体切片。结构体指针切片中的元素是指向结构体的指针,而结构体切片中的元素是结构体的值。因此,如果你尝试直接将指针切片中的指针赋值给结构体切片,你会得到的是指针的值(即内存地址),而不是结构体对象本身的值。

  要遍历结构体指针切片并将指针指向的结构体对象赋值给结构体切片,你需要对每个指针进行解引用,获取其指向的结构体对象,然后将该对象赋值给结构体切片中的相应位置。

  下面是一个详细的示例说明和分析:

package main  
import (  
    "fmt"  
)  
// 定义Person结构体  
type Person struct {  
    Name string  
    Age  int  
}  
func main() {  
    // 初始化一些Person结构体对象,并获取它们的地址  
    alice := &Person{"Alice", 30}  
    bob := &Person{"Bob", 25}  
    // 创建一个Person指针切片  
    peoplePtrSlice := []*Person{alice, bob}  
    // 创建一个与peoplePtrSlice长度相同的Person结构体切片  
    peopleSlice := make([]Person, len(peoplePtrSlice))  
    // 遍历peoplePtrSlice,解引用指针并将结构体对象赋值给peopleSlice  
    for i, personPtr := range peoplePtrSlice {  
        peopleSlice[i] = *personPtr // 解引用指针,获取结构体对象  
    }  
    // 现在peopleSlice包含了peoplePtrSlice中指针指向的结构体对象的副本  
    fmt.Println(peopleSlice) // 输出: [{Alice 30} {Bob 25}]  
    // 修改peopleSlice中的元素,不会影响peoplePtrSlice或原始结构体对象  
    peopleSlice[0].Age = 31  
    fmt.Println(peopleSlice) // 输出: [{Alice 31} {Bob 25}]  
    fmt.Println(peoplePtrSlice) // 输出: [&{Alice 30} &{Bob 25}],peoplePtrSlice中的元素未改变  
}

  在这个示例中:

  • 我们首先创建了两个Person结构体对象的指针alicebob
  • 然后我们创建了一个包含这两个指针的peoplePtrSlice切片。
  • 接着我们创建了一个与peoplePtrSlice长度相同的空Person结构体切片peopleSlice
  • 在遍历peoplePtrSlice时,我们使用*personPtr来解引用指针,并将得到的结构体对象赋值给peopleSlice中的对应位置。
  • 最后,我们验证了修改peopleSlice中的元素不会影响peoplePtrSlice或原始的结构体对象。

  需要注意的是,这样做会创建结构体对象的副本。如果你只是想引用原始的对象而不是它们的副本,你应该直接使用指针切片,而不是创建结构体切片。如果你确实需要结构体切片,并且想保持对原始对象的引用,那么你需要重新考虑你的数据结构设计,因为结构体切片本身不保存对原始对象的引用。

到此这篇关于go切片和指针切片的文章就介绍到这了,更多相关go切片和指针切片内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Golang实践之Error创建和处理详解

    Golang实践之Error创建和处理详解

    在 C#、Java 等语言中常常使用 try...catch的方式来捕获异常,但是在Golang 对于错误处理有不同的方式,像网上也有很多对 error 处理的最佳实践的文章,其中很多其实就是对 error 的统一封装,使用规范进行约束,本文主要是记录自己对处理 Error 的一些认识和学习
    2023-09-09
  • 利用Go语言实现二叉搜索树

    利用Go语言实现二叉搜索树

    二叉树是一种常见并且非常重要的数据结构,在很多项目中都能看到二叉树的身影,当然它也有很多变种,本文要介绍的是二叉搜索树的实现,希望对大家有所帮助
    2023-07-07
  • Go语言Echo服务器的方法

    Go语言Echo服务器的方法

    这篇文章主要介绍了Go语言Echo服务器的方法,实例分析了Echo服务器的技巧,具有一定参考借鉴价值,需要的朋友可以参考下
    2015-02-02
  • Go中的代码换行问题

    Go中的代码换行问题

    这篇文章主要介绍了Go中的代码换行问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-12-12
  • 使用Go实现伪静态URL重写功能

    使用Go实现伪静态URL重写功能

    在Web开发中,伪静态URL已成为优化网站架构和提升SEO的常用技术手段,伪静态URL是一种介于动态URL和静态URL之间的解决方案,本文给大家介绍了如何使用Go实现伪静态URL重写功能,需要的朋友可以参考下
    2024-08-08
  • Go语言文件读写操作案例详解

    Go语言文件读写操作案例详解

    这篇文章主要为大家介绍了Go语言文件读写操作案例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-09-09
  • golang常见接口限流算法的实现

    golang常见接口限流算法的实现

    本文主要介绍了golang常见接口限流算法的实现,包含固定窗口、滑动窗口、漏桶和令牌桶,具有一定的参考价值,感兴趣的可以了解一下
    2025-03-03
  • 一文带你了解Go语言中time包的时间常用操作

    一文带你了解Go语言中time包的时间常用操作

    在日常开发中,我们避免不了时间的使用,我们可能需要获取当前时间,然后格式化保存,也可能需要在时间类型与字符串类型之间相互转换等。本文将会对 Go time 包里面的常用函数和方法进行介绍,需要的可以参考一下
    2022-12-12
  • golang原生http包实现各种情况的get请求方式

    golang原生http包实现各种情况的get请求方式

    这篇文章主要介绍了golang原生http包实现各种情况的get请求方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-08-08
  • Go语言获取本机逻辑CPU数量的方法

    Go语言获取本机逻辑CPU数量的方法

    这篇文章主要介绍了Go语言获取本机逻辑CPU数量的方法,实例分析了runtime库的操作技巧,需要的朋友可以参考下
    2015-03-03

最新评论