你知道 go 切片的 copy 和 view 吗 
                                                    
                        
                    
                    
  
                    
                    语义理解切片
go语言中的切片是go语言的一个特色,从语义上来说,切片就是把一个整体的东西切分成小的部分,那么对于语言中的切片也是同理。
举个例子看如下代码:
package main
import "fmt"
func main() {
    arr := [...]int{0, 1, 2, 3, 4, 5, 6, 7}
    fmt.Println("arr[2:6]:", arr[2:6])  // 从下标2到下标6
    fmt.Println("arr[:6]:", arr[:6])    // 从下标0到下标6
    fmt.Println("arr[2:]:", arr[2:])    // 从下标2到最后
    fmt.Println("arr[:]:", arr[:])  // 全部
}
输出结果为:
arr[2:6]: [2 3 4 5]
arr[:6]: [0 1 2 3 4 5]
arr[2:]: [2 3 4 5 6 7]
arr[:]: [0 1 2 3 4 5 6 7]
这里可以很明确的看出,我们想要arr 数组的哪一部分,我们就切哪一部分。
当然,如果仅仅知道切片是这么用的当然还不够,我们应该更加深入的理解,如:
对原数组的 copy  还是 view 。
对于go语言的数组,copy 和 view 是同时都存在的。
- copy 就是使用这个数组的时候我将这个数组拷贝一份,这样对于数组的增删改,是不会改变原数组的值的
 - view 由数组执行切片所返回的对象是一个view,即视图,若我们在视图上操作数组,会改变原数组,
 
copy场景
package  main
import  (
  "fmt"
)
func  updateArr(arr [5]int)  {
  arr[0]  =  100
  fmt.Println("修改后的arr:", arr)
}
func  main()  {
  arr3  :=  [...]int{2,  4,  5,  6,  7}
  fmt.Println("原来的:", arr3)
  updateArr(arr3)
  fmt.Println("再次查看原始的:", arr3)
}
输出结果:
原来的: [2 4 5 6 7]
修改后的arr: [100 4 5 6 7]
再次查看原始的: [2 4 5 6 7]
如上代码可以看到,我们在updateArr 里面修改了下标为0的值,但是我们输出原始数组的时候,并没有变。这就是对数组copy。
view场景
func  updateArr(arr []int)  {
  arr[0]  =  100
  fmt.Println("修改后的arr:", arr)
}
func  main()  {
  arr3  :=  [...]int{2,  4,  5,  6,  7}
  fmt.Println("原来的:", arr3)
  // 使用切片
  updateArr(arr3[:])
  fmt.Println("再次查看原始的:", arr3)
}
输出结果:
原来的: [2 4 5 6 7]
修改后的arr: [100 4 5 6 7]
再次查看原始的: [100 4 5 6 7]
为什么view能够改变原数组
虽然Slice本身是值类型,但是它内部使用了对数组的指针引用,所以修改切片数据,会将数组原有数据修改掉。
当然,在理解上面的同时,一定要知道go是如何定义一个切片的
var b []int
所以,在 updateArr 这个函数传参的时候 arr []int 是传切片进去。不然会报错。
本作品采用《CC 协议》,转载必须注明作者和本文链接
                      本帖由系统于 6年前 自动加精
            
                
          
            
          
          
                关于 LearnKu
              
                    
                    
                    
 
推荐文章: