go之切片

一、概念

  • 关于切片
1、切片是对数组一个连续片段的引用,所以切片是一个引用类型
2、切片是数组一样可以索引,可以通过len函数获取切片的数据长度。(数组也可以通过len获取)
3、切片是一个长度可变的数组
4、切片有容量的概念,这是数组所没有的,go 提供了计算容量的函数cap(),可以计算切片容量。0 <=len(s) <=cap(s)

优点:
    因为切片是引用,所以它不需要额外的内存并且比使用数组更有效率,所以在go代码中切片比数组更常用
  • 定义
# 方式一
var slice []type
slice = make([]type,slice_len)
// 如:
//     var s []int
//     s = make([]int,0)


# 方式二
slice := make([]type,slice_len)
// 如: 
//     s := make([]int,0) 


# 方式三
slice := []type{}
// 如
//   s := []int{1,2,3}

二、切片的声明方式

  • 切片的声明与内存创建
package main

import "fmt"

func main(){
	var slice []string
	// make(类型,长度,容量。。。)
	slice = make([]string,3)  // make 内存创建,只用于 切片,map,chan类型

	slice[0] = "j"
	slice[1] = "m"
	slice[2] = "z"
	fmt.Println(slice)

}

  • 关于make
1、引用类型 可以理解为指针,声明引用类型并不会并不会分配内存。
2、与值类型不同,值类型的变量在声明时会分配内存并存储初始值
3、make 就是用于对引用类型分配内存使用的,并返回类型本身
4、目前make函数只用于 切片,map和chan

三、切片的增删改查

package main

import "fmt"

func main(){
	var slice []string
	// make(类型,长度,容量。。。)
	slice = make([]string,0)  // make 内存创建,只用于 切片,map,chan类型

	// 增
	slice = append(slice,"j")
	slice = append(slice,"m","z")
	arr := []string{"j","l","y"}  // 简短声明切片
	slice = append(slice, arr...) // 将切片打撒
	fmt.Println(slice) // [j m z j l y]


	// 改
	// 重新赋值即为改
	slice[0] = "j1"
	slice[1] = "m2"
	slice[2] = "z3"
	fmt.Println(slice) // [j1 m2 z3 j l y]

	// 查
	fmt.Println(slice[3])  // j
	fmt.Println(slice[2:4])  // 输入下标2~4直接的数据,包头不包尾,不包含下标4   // [z3 j]
	fmt.Println(slice[3:])  // 输出下标3之后的所有数据,包含下标3				// [j l y]
	fmt.Println(slice[:3]) // 输出 下标3之前的所有数据,不包含下标3              // [j1 m2 z3]
	for index,v := range slice{
		fmt.Printf("slice[%d]=%v\n",index,v)
	}
	/*
	slice[0]=j1
	slice[1]=m2
	slice[2]=z3
	slice[3]=j
	slice[4]=l
	slice[5]=y
	 */



	// 删
	// go 的切片删除是通过append 来实现的
	slice1 := []string{}
	// 删除slice[3]的数据
	slice1 = append(slice1,slice[:3]...)   // 包头不包尾,不包含下标3的
	slice1 = append(slice1,slice[4:]...)

	fmt.Println(slice1) //[j1 m2 z3 l y]
	}

四、切片之函数使用

  • 引用类型之切片函数使用
package main

import "fmt"

func main() {
	var slice []int
	// make(类型,长度,容量。。。)
	slice = make([]int, 3) // make 内存创建,只用于 切片,map,chan类型

	fmt.Println(slice) // [0,0,0]

	slice = append(slice, 4)
	fmt.Println(slice) // [0,0,0,4]

	saveSlice(slice)
	fmt.Println(slice)
}

func saveSlice(s []int){
	s[0] = 111
}

[0 0 0]
[0 0 0 4]
[111 0 0 4]
  • 为什么函数没有返回值,却改变类切片的值
切片是引用类型,你可以理解为指针
切片在传参的过程不是值拷贝的过程
切片传参实际传的是引用

五、切片之拷贝

  • 引用的复制
package main

import "fmt"

func main() {
	s := []int{1,2,3,4}

	var s1 []int
	s1 = make([]int,len(s))
	s1 = s  // 将 s 复制给 s1

	/*
	s1 := s  // 与上面是一样的效果
	 */

	s[0] = 111
	fmt.Println(s1)
}

  • 为什么切片s的修改,s1 也发生类变化
1、切片其实本质还是对数组的引用
2、赋值操作其实还是对引用的拷贝

简单点:s1 只是对s引用的拷贝,不是值的拷贝
  • 值的拷贝一
package main

import "fmt"

func main() {
	s := []int{1,2,3,4}

	var s1 []int
	s1 = make([]int,len(s))
	for index := range s{
		s1[index] = s[index]
	}
	s[0] = 1111
	fmt.Println(s1)
}
  • 值的拷贝二
package main

import "fmt"

func main() {
	s := []int{1,2,3,4}

	var s1 []int
	s1 = make([]int,0)
	s1 = append(s1,s...)

	s[0] = 111
	fmt.Println(s1)
}

六、切片之copy

  • go 内部函数copy
package main

import "fmt"

func main() {
	s := []int{1,2,3,4}

	s1 := make([]int,len(s))

	copy(s1,s)
	s[0] = 111
	fmt.Println(s1)

}

输出结果
[1 2 3 4]

七、注意

切片的简短声明方式
s := []int{}  // var s []int;s = make([]int,0)   一样的

如果喜欢看小说,请到183小说网

posted @ 2019-03-31 19:39  xiaobaiskill  阅读(219)  评论(0编辑  收藏  举报