11-GoLang切片

——引言:无论是C语言中的数组还是Go语言中的数组,数组的长度一旦确定就不能改变, 但在实际开发中我们可能事先不能确定数组的长度, 为了解决这类问题Go语言中推出了一种新的数据类型切片

  • 什么是切片?
    1.切片简单理解就是一个可变长度的数组底层的实现原理就是一个结构体, 结构体中有一个指针指向了一个数组
    2.本质上所有的数据都是保存在指向的数组中的
    先看一遍这个图,留个印象,一会儿再看一遍


emmmmm,再看下底层源码

//切片源码
    type slice struct{
      array unsafe.Pointer // 指向底层数组指针
      len int // 切片长度(保存了多少个元素)
      cap int // 切片容量(可以保存多少个元素)
    }
  • 如何创建切片
    1.通过数组来创建
    格式: [起始位置:结束位置], 从起始位置开始截取, 直到结束位置, 但是不包括结束位置
    注意: 截取了多少个元素, len就等于几
    容量等于数组的长度 - 起始位置
package main
import "fmt"
func main() {
    // 1.通过数组来创建
    // 1.创建一个数组
    //                         0  1  2  3  4
    var ages [5]int = [5]int{1, 3, 5, 7, 9}
    // 2.通过数组创建切片
    // [起始位置:结束位置], 从起始位置开始截取, 直到结束位置, 但是不包括结束位置
    var sce01 []int = ages[0:2]
    var sce02 []int = ages[2:4]
    fmt.Println(sce01)   //[1 3]
    fmt.Println(sce02)   //[5 7]
    // 注意点: 如果只写了起始位置, 没有写结束位置, 那么会截取到最后
    var sce03 []int = ages[0:]
    var sce04 []int = ages[2:]
    fmt.Println(sce03)   //[1 3 5 7 9]
    fmt.Println(sce04)   //[5 7 9]
    // 注意点: 如果只写了结束位置, 没有写开始位置, 那么会从开始截取到指定的结束位置
    var sce05 []int = ages[:2]
    fmt.Println(sce05)  //[1 3]
    // 注意点: 如果只写了:, 那么就是从开始截取到末尾
    var sce06 []int = ages[:]
    fmt.Println(sce06)   //[1 3 5 7 9]
    // 计算切片当前保存数据的个数
    fmt.Println(len(sce06))   //5
    // 计算切片总共可以保存数据的个数
    fmt.Println(cap(sce06))   //5

    var ages1 [5]int = [5]int{1,3,5,7,8}
    var sce1 []int = ages1[2:4]
    var sce2 []int = ages1[0:]
    var sce3 []int = ages1[2:]
    var sce4 []int = ages1[:4]
    fmt.Println(sce1)
    fmt.Println(sce2)
    fmt.Println(sce3)
    fmt.Println(sce4)
    fmt.Println(len(sce4))
    fmt.Println(cap(sce4))
}

2.通过make函数来创建

  • 格式: make([]数据类型, 长度, 容量)
  • 内部会先创建一个数组,让切片指向数组
  • 不指定容量时切片容量等于长度
// 2.通过make函数创建切片

    // 第一个参数: 告诉系统要存储什么类型的数据
    // 注意点: 如果是创建切片一定要在传入的数据类型前面写上[]
    // 第二个参数: 告诉系统创建出来的切片len等于多少
    // 第三个参数: 告诉系统创建出来的切片cap等于多少
    // 注意点: 第三个参数可以省略, 如果省略切片的容量就等于切片的长度(等于第二个参数)
    var sce []int = make([]int, 2, 5)
    //var sce []int = make([]int, 2)  //省略时默认容量cap=len,即2
    fmt.Println(sce)   //[0 0],初始化为0值
    fmt.Println(len(sce))  //2
    fmt.Println(cap(sce))  //5

3.通过Go提供的语法糖来创建

// 3.通过Go提供的语法糖来创建
    // 一定要注意[]里面没有值就是切片
    // 通过Go提供的语法糖来创建len等于cap
    var sce []int = []int{1, 3, 5} // 相当于make([]int, 3)
    fmt.Println(sce)         //[1 3 5]
    fmt.Println(len(sce))    //3
    fmt.Println(cap(sce))    //3
    var sce6 []int = []int{2,4,6}   
    fmt.Println(sce6)        //[2 4 6]
    fmt.Println(len(sce6))   //3
    fmt.Println(cap(sce6))   //3
    sce7:=[]int{1,1,1}  
    fmt.Println(sce7)        //[1 1 1]

切片和数组

  • 切片和数组最直观的区别
    []中没有数字就是切片, []中有数字就是数组(或者省略格式“...”)
  • 数组:
// 数组
    var ages1 [3]int = [3]int{1, 3, 5} // 这里是数组
    var ages2 = [3]int{1, 3, 5} // 这里是数组
    ages3 := [3]int{1, 3, 5} // 这里是数组
    ages4  := [...]int{1, 3, 5} // 这里写...代表是定义一个数组
    fmt.Println(ages1,ages2,ages3,ages4)
  • 切片:
// 2.切片
    var ages5 []int = []int{1, 3, 5} // 省略元素个数, 这里不是数组是切片
    var ages6 = []int{1, 3, 5} // 省略元素个数, 这里不是数组是切片
    ages7 := []int{1, 3, 5} // 省略元素个数, 这里不是数组是切片
    fmt.Println(ages5,ages6,ages7)
  • 二维数组
// 二维数组
    // 定义一个数组, 数组中存储的又是数组
    var values1 [2][3]int = [2][3]int{
        {1, 3, 5},
        {2, 4, 5},
    }
    var values2 = [2][3]int{
        {1, 3, 5},
        {2, 4, 5},
    }
    values3 := [2][3]int{
        {1, 3, 5},
        {2, 4, 5},
    }
    values4 := [...][3]int{
        {1, 3, 5},
        {2, 4, 5},
    }
    fmt.Println(values1,values2,values3,values4)
  • 切片存数组
// 切片
    // 定义一个切片, 切片中保存的是数组
    var values01 [][3]int = [][3]int{
        {1, 3, 5},
        {2, 4, 5},
    }
    var values02 = [][3]int{
        {1, 3, 5},
        {2, 4, 5},
    }
    values03 := [][3]int{
        {1, 3, 5},
        {2, 4, 5},
    }
    fmt.Println(values01,values02,values03)
  • 切片存切片
//  定义一个切片, 切片中保存的是切片
    var values11 [][]int = [][]int{
        {1, 3, 5},
        {2, 4, 5},
    }
    var values12 = [][]int{
        {1, 3, 5},
        {2, 4, 5},
    }
    values13 := [][]int{
        {1, 3, 5},
        {2, 4, 5},
    }
    fmt.Println(values11,values12,values13)
  • 注意点:
    只要数组的[]中没有编写数字, 那么这个就不是一个数组, 这就是一个切片

切片的基本使用

  • 1.可以像使用普通数组那样使用切片
    var sce []int = make([]int, 2, 5)
    fmt.Println(sce)
    sce[0] = 1
    sce[1] = 2
    // 注意点: 如果像使用数组一样使用切片, 那么索引的长度不能超过切片的len
    //sce[2] = 999  //报错  超过len
  • 2.可以通过预先定义好的函数来使用切片
    append(切片, 数据)

  • 3.append函数的作用:
    在len后面追加数据

    var sce []int = make([]int, 2, 5)
    fmt.Println(sce)
    sce[0] = 1
    sce[1] = 2
    //用append函数可以在len后面追加切片的元素
    sce = append(sce, 3)
    fmt.Println(sce)   //[1 2 3]
    fmt.Printf("%p\n", sce)  //0xc04206a0c0,切片是地址传递,结构体中有指针指向数组
    sce = append(sce, 4)
    fmt.Printf("%p\n", sce)  //0xc04206a0c0
    sce = append(sce, 5)
    fmt.Printf("%p\n", sce)  //0xc04206a0c0
    fmt.Println(sce)         //[1 2 3 4 5]
    fmt.Println(cap(sce))    //5
    // 超过容量
    sce = append(sce, 6)
    fmt.Printf("%p\n", sce)   //0xc042088000  超过原本容量会重新分配内存空间,地址会变
    fmt.Println(sce)          //[1 2 3 4 5 6]
    fmt.Println(cap(sce))     //10   会扩容成原本容量的两倍
  • 4.容量
    如果通过append函数追加数据之后超过了原有的容量, 那么系统内部会自动按照当前容量*2的方式重新定义一个数组作为切片保存数据的模型
    var sce []int = make([]int, 1024, 1024)
    fmt.Printf("%p\n", sce)   //0xc042072000
    fmt.Println(cap(sce))   //1024
    sce = append(sce, 1025)
    fmt.Printf("%p\n", sce)   //0xc042078000
    fmt.Println(cap(sce))   //1280
  • 5.为什么append函数要返回一个切片
    因为在扩容的时候会重新定义一个新的切片, 所以需要返回一个切片

切片的高级用法(假装很高级)

  • 切片的增删查改
// 1.定义一个切片
    sce := []int{1, 3, 5, 7, 9}

    // 2.修改切片中的值
    // 切片名称[索引] = 值
    fmt.Println(sce)
    sce[1] = 666
    fmt.Println(sce)

    // 3.给切片增加数据
    fmt.Println(sce)
    sce = append(sce, 666)
    fmt.Println(sce)

    // 4.查询切片中的数据
    for _, value := range sce {
        if value == 3 {
            fmt.Println("包含数字3")
        }
    }


    //// 5.删除切片中指定的元素
    // {1, 3, 5, 7, 9}
    index := 2 // 需要删除元素的索引
    //注意点: 不仅能够通过数组生成切片, 还可以通过切片生成切片
    //sce = sce[:index] // {1, 3}
    //fmt.Println(sce)
    //sce = sce[index + 1:] //{7, 9}
    //fmt.Println(sce)
    //发现规律: 将第二次截取的切片添加到第一次截取的切片后面即可
    //{1, 3, 7, 9}
    //...是可变参数
    sce = append(sce[:index], sce[index + 1:]...)
    fmt.Println(sce)
  • 通过数组生成切片和切片生成切片的步骤及注意点(关注地址问题)
// 1.创建一个数组
    var arr [5]int = [5]int{1, 3, 5, 7, 9} // 数组
    // Go语言中不能直接通过数组名获取数组的地址,与C语言不同
    //fmt.Printf("%p\n", arr)    //不能这样写,必须有取地址符&
    fmt.Printf("%p\n", &arr)//   0xc04206a0f0
    fmt.Printf("%p\n", &arr[0])//0xc04206a0f0

    // 2.通过数组创建一个切片
    var sce0 []int = arr[:]
    // 直接打印sce0打印的是sce中指针保存的地址
    // 也就是底层指向的那个数组的地址
    fmt.Printf("%p\n", sce0)   //0xc04206a0f0
    //打印&sce0才是切片自己的地址
    fmt.Printf("%p\n", &sce0)  //0xc0420504a0

    // 3.通过切片创建一个切片
    // 通过切片创建一个切片, 新的切片和老的切片底层指向同一个数组
    var sce2 []int = sce0[:]
    fmt.Printf("%p\n", sce0)   //0xc04206a0f0

    // 4.会直接修改指向的同一个数组的值
    //arr[1] = 666
    //sce[1] = 666
    sce2[1] = 666
    fmt.Println(arr)
    fmt.Println(sce)
    fmt.Println(sce2)

注意点:切片可以再生成新的切片, 两个切片底层指向同一个数组
现在再看一遍这个图,是不是醍醐灌顶


  • 通过数组生成切片补充
    • 格式: 数组名称[low: high, max]
     //                  0  1  2  3  4
     var arr [5]int = [5]int{1, 3, 5, 7, 9}
     //不指定切片长度
     var sce0 []int = arr[2:4]
     fmt.Println(sce0) // [5 7]
     fmt.Println(len(sce0)) // 2  high-low = 切片长度
     fmt.Println(cap(sce0)) // 3  数组以前的长度 - 截取的开始位置 = 容量
    
    
     // 总结: 如果没有指定第三个参数, 那么切片的容量就等于 数组的容量 - 第一个参数
     //       如果指定了第三个参数, 那么切片的容量就等于 第三个参数 - 第一个参数
     //       切片的第三个参数max,必须大于或等于第二个参数且不能超过数组的长度(越界)
     //指定切片长度
     var sce []int = arr[2:4:5]
     fmt.Println(sce)
     fmt.Println(len(sce))
     fmt.Println(cap(sce))
    
    • 仅仅定义, 没有初始化的数组是可以使用的, 但是仅仅定义但是没有初始化的切片是不能使用的(零值问题)
    // 1.定义一个数组变量
     var arr [3]int
     // 2.使用数组变量
     //arr[0] = 1
     arr[1] = 3
     arr[2] = 5
     // 3.打印使用之后的结果
     fmt.Println(arr)    //[0 3 5] 数组会自动给元素赋零值
    
     //1.定义一个切片变量
     var sce []int
     //对切片进行初始化
     sce = make([]int, 3, 5)  //会内建数组,所以也会初始化为零值
    
     //2.使用切片变量
     //赋值
     sce[0] = 1
     sce[1] = 3
     sce[2] = 5
     //追加
     sce = append(sce, 2)
     sce = append(sce, 4)
     sce = append(sce, 6)
    
     // 3.打印使用之后的结果
     fmt.Println(sce)   //[1 3 5 2 4 6]
     fmt.Println(cap(sce))   //10 append自动扩容两倍
    
    • 计算切片长度(与数组对比)
    var arr [3]int = [3]int{1, 3, 5}
     // 第一种方式: 推荐方式
     length := len(arr)
     fmt.Println(length)
     // 第二种方式: 不推荐方式
     size := unsafe.Sizeof(arr)
     fmt.Println(size)
     eleSize := unsafe.Sizeof(arr[0])
     fmt.Println(eleSize)
     fmt.Println(size/eleSize)
    
     var sce []int = []int{1, 3, 5, 7, 9}
     // 方式一: 推荐方式
     length2 := len(sce)
     fmt.Println(length2)
    
     // 方式二: 注意点, 切片不能通过sizeof来计算
     // 由于切片是一个结构体内部指向了一个数组, 所以通过sizeof计算的是切片本身结构体的大小, 而不是切片的长度
     //size := unsafe.Sizeof(sce)
     //fmt.Println(size)
    

可变参数

  • 可变参数底层其实就是一个切片

  • 使用数组传递参数时必须明确参数长度

package main
import "fmt"
func main()  {
    arr := [3]int{10, 20, 30}
    res := sum2(arr)
    fmt.Println(res)
    sce := []int{10, 20, 30}
    result := sum1(sce)
    fmt.Println(result)
}
func sum2(nums [3]int) int{
    // 1.获取数组的长度
    length := len(nums)
    var res int // 保存计算结果
    // 2.遍历数组
    for i := 0; i < length; i++ {
        res += nums[i]
    }
    return res
}
func sum1(nums []int) int{
    // 2.遍历切片
    var res int
    for _, value := range nums{
        res += value
    }
    return res
}
  • 使用可变参数(切片)时不需要对参数长度进行计算
package main
import "fmt"
func main()  {
    res:=sum( 10, 20, 30)
    fmt.Println(res)
    res2:=sum02(3.14, 10, 20, 30)
    fmt.Println(res2)
}
// 注意点: 如果函数编写了可变参数, 那么可变参数只能放在形参列表的最后
func sum( nums ...int) int{
    var res int
    for _,value := range nums {
        res += value
    }
    return res
}
func sum02(value float32, nums ...int) int{
    var res int
    for _,value := range nums {
        res += value
    }
    return res
}
  • 对照两段代码,再次验证了可变参数的底层实现实际上是切片

copy函数

package main
import "fmt"
func main() {
    sce1 := []int{1, 3, 4} // 源切片
    sce12 := []int{1, 3, 4, 5} // 源切片
    sce13 := []int{1, 3} // 源切片
    sce2 := []int{0, 0, 0} // 目标切片
    sce02 := []int{0, 0, 0} // 目标切片
    sce03 := []int{0, 0, 0} // 目标切片
    // 第一个参数:目标切片
    // 第二个参数:源切片
    // 会将源切片中的数据拷贝到目标切片中
    // 拷贝的时候会以目标切片为主(源切片元素更多则舍去,元素更少则补0)
    copy(sce2, sce1)
    copy(sce02, sce12)
    copy(sce03, sce13)
    fmt.Println(sce2)    //[1 3 4]
    fmt.Println(sce02)   //[1 3 4]
    fmt.Println(sce03)   //[1 3 0]
}

切片补充

  • 切片和数组不同, 切片不支持== !=操作
    var arr1 [3]int = [3]int{1, 3, 5}
     var arr2 [3]int = [3]int{1, 3, 5}
     res1 := arr1 == arr2
     fmt.Println(res1)   //true


    var sce1 []int = []int{1, 3, 5}
    var sce2 []int = []int{1, 3, 5}
    res2 := sce1 == sce2    //会报错,切记切片是地址拷贝
    fmt.Println(res2)
  • 在Go语言中字符串的底层实现就是切片, 所以可以通过字符串来生成切片
   var str string = "www.doudou.com"
    //str[0] = 'M'    //这样操作会报错,不可以直接修改字符串
    //sce := str[:]
    sce := str[4:]
    fmt.Printf("%T\n", sce)   //string   此切片的数据类型
    fmt.Println(sce)                 //doudou.com

    var str2 string = "www.doudou.com"
    sce2 := make([]byte, 3)       //byte用于存储一个字符
    copy(sce2, str2)
    fmt.Println(sce2)  //[119 119 119]  输出ASCLL码
    fmt.Printf("%c\n", sce2[0]) //w
    fmt.Printf("%c\n", sce2[1]) //w
    fmt.Printf("%c\n", sce2[2]) //w
    sce3 := make([]byte, len(str2))
    sce0 :=make([]byte,len(str2))
    copy(sce3, str2)
    copy(sce0, str2)
    sce3[0] ='M'
    sce0[0] ='N'
//用切片可以间接修改字符串
    fmt.Printf("%s\n", sce3)  //Mww.doudou.com
    fmt.Printf("%s\n", sce0)  //Nww.doudou.com

数组和切片做函数参数的区别(值传递和地址传递)

  • 数组作为函数的参数, 是值传递, 在函数内修改形参, 不会影响函数外的实参
  • 切片作为函数的参数, 是地址传递(指针), 在函数内修改形参, 会影响到函数外的实参
func main() {
    var arr [3]int = [3]int{1, 3, 5}
    fmt.Println(arr)   //[1 3 5]
    change1(arr)       
    fmt.Println(arr)   //[1 3 5]

    var sce []int = []int{1, 3, 5}
    fmt.Println(sce)    //[1 3 5] 
    change2(sce)        
    fmt.Println(sce)    //[1 666 5]
}
func change2(nums []int)  {
    nums[1] = 666
}
func change1(nums [3]int)  {
    nums[1] = 666
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,864评论 6 494
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,175评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,401评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,170评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,276评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,364评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,401评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,179评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,604评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,902评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,070评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,751评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,380评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,077评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,312评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,924评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,957评论 2 351