最新项目使用go语言开发,因此有机会结识了go语言。在写代码时,无意间发现了同事代码的一个bug。今天拿来一起学习一下。
首先go语言有个强大的基本数据结构,那就是切片,切片相对于数组更加灵活,动态扩展性极好。而go 语言有个内置函数,叫append,函数原型如下:func append(slice []Type, elems ...Type) []Type,即appen可以对切片追加元素。
下面简单的举个例子,例子可能会有点硬,希望大家理解。
由于消息交互,需要将元素为int 类型的切片,转换为 元素为指针类型的切片。 因此有了如下操作:(错误方式为原始代码bug方式简化版)
```go
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:/Go\bin\go.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的地址,因此第一个元素里面存放的值,也跟着变了。 造成的后果就是,转化后,新切片中所有元素里面存放的值,都变成了原切片最后一个元素的值。
那如何做才能够解决问题呢?其实我们可以按照正确做法那样书写代码,直接用下标的方式去找对应地址,这样就不会出错了。
有疑问加站长微信联系(非本文作者))