go 切片使用小结

时间:2022-05-06
本文章向大家介绍go 切片使用小结,主要内容包括其使用实例、应用技巧、基本知识点总结和需要注意事项,具有一定的参考价值,需要的朋友可以参考一下。
最新项目使用go语言开发,因此有机会结识了go语言。在写代码时,无意间发现了同事代码的一个bug。今天拿来一起学习一下。

首先go语言有个强大的基本数据结构,那就是切片,切片相对于数组更加灵活,动态扩展性极好。而go 语言有个内置函数,叫append,函数原型如下:func append(slice []Type, elems ...Type) []Type,即appen可以对切片追加元素。

下面简单的举个例子,例子可能会有点硬,希望大家理解。

由于消息交互,需要将元素为int 类型的切片,转换为 元素为指针类型的切片。 因此有了如下操作:(错误方式为原始代码bug方式简化版)

package main

import "fmt"

func main()  {



    intSlince := make([]int, 0 )

    intSlince = append(intSlince,9)

    intSlince = append(intSlince,8)



    intSlince = append(intSlince,7)



    for _ , v := range intSlince{

        fmt.Println("before convert:",v)

    }





    intPtrSlinceA := make([]*int, 0)



    //错误方式

    //转换切片元素为指针类型

    for _ , v := range intSlince{

        intPtrSlinceA = append(intPtrSlinceA, &v)

    }



    //打印切片元素

    for _ , v := range intPtrSlinceA{

        fmt.Println("wrong:",*v)

    }



    intPtrSlinceB := make([]*int, 0)

    //正确方式

    //转换切片元素为指针类型

    for i:= range intSlince{

        intPtrSlinceB = append(intPtrSlinceB, &intSlince[i])

    }



    //打印切片元素

    for _ , v := range intPtrSlinceB{

        fmt.Println("Right:",*v)

    }



}

运行结果如下:

C:/Gobingo.exe run E:/go_code/gopath/src/console/append.go

before convert: 9

before convert: 8

before convert: 7

wrong: 7

wrong: 7

wrong: 7

Right: 9

Right: 8

Right: 7

消息转化后,发送出去,在接收端,发现消息数据异常。 那么转化前好的,接收时却是异常的,说明转化有问题,再仔细分析上面那段代码。

遍历整个切片,将元素赋值给临时变量v,并把v的地址追加到新的切片中去,动作完成。问题出现在 当把第一个值赋值给临时变量v,并把v的地址追加到新的切片时,那么新的切片里存放的是v地址, 值是原切片第一个元素的值, 而当第二个元素变量赋值时,新的切片的第二个元素的地址 仍然是临时变量v的地址,只不过这个地址存放的值,变成原切片第二个元素的值了,同理,新切片的第一个元素也是v的地址,因此第一个元素里面存放的值,也跟着变了。 造成的后果就是,转化后,新切片中所有元素里面存放的值,都变成了原切片最后一个元素的值。

那如何做才能够解决问题呢?其实我们可以按照正确做法那样书写代码,直接用下标的方式去找对应地址,这样就不会出错了。