golang-切片

2019-12-17  本文已影响0人  爱吃豆包
package container

import "fmt"

/**
    切片
    切片本身没有值, 是对底层array的一个view(视图)
    所有的切片都是在原arr数组的范围中进行切出数据的!
    但是只能向后扩展,不能像前扩展!

    切片里面,声明的 var a []int 没有赋值上默认值, 默认是 nil,但是为nil的情况下,也可以直接往里面添加元素!

 */

/**
    切片是引用传递, 会直接修改原有的值
 */

// 修改其中的某个值
func updateSlice(s []int){
    s[0] = 100
}


func main() {

    arr := [...]int{0,1,2,3,4,5,6,7,8,9}

    // arr[2:6] =  [2 3 4 5]
    fmt.Println("arr[2:6] = ", arr[2:6])

    // arr[:6] =  [0 1 2 3 4 5]
    fmt.Println("arr[:6] = ", arr[:6])

    // arr[2:] =  [2 3 4 5 6 7 8 9]
    fmt.Println("arr[2:] = ", arr[2:])

    // arr[:] =  [0 1 2 3 4 5 6 7 8 9]
    fmt.Println("arr[:] = ", arr[:])


    /**
        通过 s1 := arr[2:6] 声明的切片  这个例子的结论:

        (1)s1 := arr[2:6] 想当于就是在原有的 arr 数组基础上,进行的一个部分内存地址的引用的指向,然后声明出的 s1 的这个变量!
        修改了 s1 里面的值,也会对原有的 arr 数组产生修改,因为他们使用的都是相同内存地址!

        (2)数组里面的每一个值在内存里面都会有一个唯一的内存地址标记!

     */


    // 自动类型,获取一个数组的部分值,转换为切片
    s1 := arr[2:6]
    updateSlice(s1)
    // s1 =  [100 1 2 3 4 5 6 7 8 9]
    fmt.Println("s1 = ", s1)
    fmt.Println("arr = ", arr)

    /**
        此时 s1 里面并没有 6~8 之间的下坐标数, 所以按照常理来说是没有值的!
        但是此时却是有值: s2 =  [8 9],
        为什么会有值?

        推论:
            (1)这 s2 =  [8 9] 的值是原arr数组的值。
            (2)这 s2 =  [8 9] 的值,是我Go语言获取到 s1 的最后一个值之后,针对最后一个值进行的加1操作!
                也就是说 s1 切片里面最后一个值是 5,那么我Go语言获取到这个 5 之后,在这个值上面往后加1,然后5的后面是6!
        (1)正确
        (2)错误

        这两个论证点哪一个是真?

        但是继续测试  s2 =  [8 9] // 报错 , 结果报错了,说切片超出范围!
        由此得知:我 s2 还是针对原arr数组获取的值!也就是说我的切片,切出的值都是基于原arr数组的范围!
        所以第一个论证点是正确的!

        所有的切片都是在原arr数组的范围中进行切出数据的!

        如果说 s1 := arr[2:6] 是这个范围,那么 s2 在s1上切片,s2 := s1[1:3] 是要出错的!因为s2受限于s1,
        但是 s2 := arr[6:8] 是受限于 原arr数组!
        也就是说 只能向后扩展,不能向前扩展!



     */
    s2 := s1[6:8]
    // s2 =  [8 9] // 报错
    fmt.Println("s2 = ", s2)

    // s1=[100 3 4 5],len(s1)=4, cap(s1)=8
    /**
        cap 表示总共有多少个可看的容量, 相当于就是从起始位置算起,只能看后面的,不能看前面的!
        也就是说 s1 是从下坐标2开始的,就只能从2开始算起,有多少个可以看的元素!
      */
    fmt.Printf("s1=%v,len(s1)=%d, cap(s1)=%d\n",s1,len(s1),cap(s1))


    /**
        切片添加元素的时候,是基于原arr元素的,
        如果超出了原arr数组的长度,Go语言会复制一份,在重新创建一个新的数组,新的数组的长度会比原arr数组要大,
        但是新数组的元素如果是旧数组的,那么就会继续使用原数组的元素的内存地址
        如果没有超出原arr数组,就把最后面的值给替换掉,那么对原arr数组本身存在数据会产生影响也会进行修改,因为内存地址相同
     */

    s3 := arr[4:9]
    fmt.Println("s3 = ", s3)
    // 往slice切片里面添加一个 元素
    s4 := append(s3, 23)
    fmt.Println("s4 = ", s4)
    fmt.Println("arr = ", arr)

    // make 分配函数
    /**
        make() 只用于映射、切片和程道,不返回指针。要明确的得到指针用 new() 分配
        make在分配的时候,不能超过总共大小的元素空间!
        比如 make([]int, 5, 10) 在分配的时候,第二个参数的不能大于第三个参数的值!
     */
    //创建一个初始元素个数为5的数组切片,元素初始值为0,并预留10个元素的存储空间
    // 相当于就是初始化5个大小的数组切片,然后共10个空间位置
    makeA := make([]int, 5, 10)    // len(makeA)=5, cap(makeA)=10
    fmt.Println("makeA = ", makeA)

    // 复制值,把 s1 的值复制到 makeA 中
    copy(makeA, s1)
    fmt.Println("复制之后的makeA = ", makeA)

    // 删除操作
    // 复制之后的makeA =  [100 3 4 5 0]
    // 删除makeA中的第3个元素,下坐标也就是2
    makeA = append(makeA[:2], makeA[3:]...)
    fmt.Println("删除元素后的 makeA = ", makeA) // [100 3 5 0]

}

上一篇 下一篇

猜你喜欢

热点阅读