• Golang学习笔记——Slice


    切片和数组很类似,甚至你可以理解成数组的子集。但是`切片有一个数组所没有的特点,那就是切片的长度是可变的`。

    严格地讲,切片有`容量(capacity)`和`长度(length)`两个属性。

    首先我们来看一下切片的定义。切片有两种定义方式,一种是先声明一个变量是切片,然后使用内置函数make去初始化这个切片。另外一种是通过取数组切片来赋值。

     1 package main
     2 
     3     import (
     4         "fmt"
     5     )
     6 
     7     func main() {
     8         var x = make([]float64, 5)
     9         fmt.Println("Capcity:", cap(x), "Length:", len(x))
    10         var y = make([]float64, 5, 10)
    11         fmt.Println("Capcity:", cap(y), "Length:", len(y))
    12 
    13         for i := 0; i < len(x); i++ {
    14             x[i] = float64(i)
    15         }
    16         fmt.Println(x)
    17 
    18         for i := 0; i < len(y); i++ {
    19             y[i] = float64(i)
    20         }
    21         fmt.Println(y)
    22     }

    输出结果为

    Capcity: 5 Length: 5
    Capcity: 10 Length: 5
    [0 1 2 3 4]
    [0 1 2 3 4]

    上面我们首先用make函数定义切片x,这个时候x的容量是5,长度也是5。然后使用make函数定义了切片y,这个时候y的容量是10,长度是5。然后我们再分别为切片x和y的元素赋值,最后输出。

    所以使用make函数定义切片的时候,有`两种方式`,一种`只指定长度,这个时候切片的长度和容量是相同的`。另外一种是`同时指定切片长度和容量`。虽然切片的容量可以大于长度,但是`赋值的时候要注意最大的索引仍然是len(x)-1`。否则会报索引超出边界错误。

    另外一种是通过数组切片赋值,采用`[low_index:high_index]`的方式获取数值切片,其中切片元素`包括low_index的元素`,但是`不包括high_index的元素`。

     1 package main
     2 
     3     import (
     4         "fmt"
     5     )
     6 
     7     func main() {
     8         var arr1 = [5]int{1, 2, 3, 4, 5}
     9         var s1 = arr1[2:3]
    10         var s2 = arr1[:3]
    11         var s3 = arr1[2:]
    12         var s4 = arr1[:]
    13         fmt.Println(s1)
    14         fmt.Println(s2)
    15         fmt.Println(s3)
    16         fmt.Println(s4)
    17     }

    输出结果为

    [3]
    [1 2 3]
    [3 4 5]
    [1 2 3 4 5]


    在上面的例子中,我们还省略了low_index或high_index。如果省略了low_index,那么等价于从索引0开始;如果省略了high_index,则默认high_index等于len(arr1),即切片长度。

    这里为了体现切片的长度可以变化,我们看一下下面的例子:

     1 package main
     2 
     3     import (
     4         "fmt"
     5     )
     6 
     7     func main() {
     8         var arr1 = make([]int, 5, 10)
     9         for i := 0; i < len(arr1); i++ {
    10             arr1[i] = i
    11         }
    12         fmt.Println(arr1)
    13 
    14         arr1 = append(arr1, 5, 6, 7, 8)
    15         fmt.Println("Capacity:", cap(arr1), "Length:", len(arr1))
    16         fmt.Println(arr1)
    17     }

    输出结果为

    [0 1 2 3 4]
    Capacity: 10 Length: 9
    [0 1 2 3 4 5 6 7 8]

    这里我们初始化arr1为容量10,长度为5的切片,然后为前面的5个元素赋值。然后输出结果。然后我们再使用Go内置方法append来为arr1追加四个元素,这个时候再看一下arr1的容量和长度以及切片元素,我们发现切片的长度确实变了。

    另外我们再用`append`方法给arr1多追加几个元素,试图超过arr1原来定义的容量大小。

     1 package main
     2 
     3     import (
     4         "fmt"
     5     )
     6 
     7     func main() {
     8         var arr1 = make([]int, 5, 10)
     9         for i := 0; i < len(arr1); i++ {
    10             arr1[i] = i
    11         }
    12 
    13         arr1 = append(arr1, 5, 6, 7, 8, 9, 10)
    14         fmt.Println("Capacity:", cap(arr1), "Length:", len(arr1))
    15         fmt.Println(arr1)
    16     }

    输出结果为

    Capacity: 20 Length: 11
    [0 1 2 3 4 5 6 7 8 9 10]

    我们发现arr1的长度变为11,因为元素个数现在为11个。另外我们发现arr1的容量也变了,变为原来的两倍。这是因为`Go在默认的情况下,如果追加的元素超过了容量大小,Go会自动地重新为切片分配容量,容量大小为原来的两倍`。

    上面我们介绍了,可以`使用append函数给切片增加元素`,现在我们再来介绍一个`copy函数用来从一个切片拷贝元素到另一个切片`。

     1 package main
     2 
     3     import (
     4         "fmt"
     5     )
     6 
     7     func main() {
     8         slice1 := []int{1, 2, 3, 4, 5, 6}
     9         slice2 := make([]int, 5, 10)
    10         copy(slice2, slice1)
    11         fmt.Println(slice1)
    12         fmt.Println(slice2)
    13     }

    输出结果

    [1 2 3 4 5 6]
    [1 2 3 4 5]

    在上面的例子中,我们将slice1的元素拷贝到slice2,因为slice2的长度为5,所以最多拷贝5个元素。

    总结一下,数组和切片的区别就在于`[]`里面是否有数字或者`...`。因为数值长度是固定的,而切片是可变的。

  • 相关阅读:
    CF1132G
    CF1129B
    CF1131G
    CF1109D
    CF1110H
    CF1106F
    my.cnf 配置
    mysql 导入导出
    mysql 批量删除表数据
    国内开源镜像站
  • 原文地址:https://www.cnblogs.com/cuibin/p/6752328.html
Copyright © 2020-2023  润新知