Go语言之切片

简介: Go语言之切片

切片


切片(slice)是一个拥有相同类型元素的可变长度的序列。它是基于数组类型做的封装。它非常灵活,支持自动扩容。

与数组的区别:

1、定义写法类似

切片:var name []string

数组:var name [5]string 数组必须指定长度数字

2、相互转换

切片可以通过数组转换而来,转换成切片之后,就可以扩容了

package main
import "fmt"
func main() {
  var a = [5]int{1,2,3,4,5}
  fmt.Printf("a的类型是%T\n",a)
  var b = a[0:3]
  fmt.Printf("b的类型是%T\n",b)
}


输出结果:

a的类型是[5]int
b的类型是[]int


切片不能直接转换成数组


切片的构造


1、直接声明

package main
import "fmt"
func main() {
  e := []int{}
  fmt.Printf("%T",e)
}


输出结果:

[]int


2、通过数组转换

package main
import "fmt"
func main() {
  var a = [5]int{1,2,3,4,5}
  fmt.Printf("a的类型是%T\n",a)
  var b = a[0:3]
  fmt.Printf("b的类型是%T\n",b)
}


输出结果:

a的类型是[5]int
b的类型是[]int


上面说了,切片的本质是对底层数组的封装,它包含了三个信息:底层数组的指针、切片的长度(len)和切片的容量(cap)

package main
import "fmt"
func main() {
  a := []int{1,2,3,4,5,6,7,8}
  s1 :=a[0:5]
  fmt.Printf("s1的长度是%v\n",len(s1))
  fmt.Printf("s1的容量是%v\n",cap(s1))
  s2 := a[3:6]
  fmt.Printf("s2的长度是%v\n",len(s2))
  fmt.Printf("s2的容量是%v\n",cap(s2))
}


输入结果:

s1的长度是5
s1的容量是8
s2的长度是3
s2的容量是5


这个结果是正确的,看下图:

1675236645934.jpg

主要说下容量,容量的意思是当长度大于容量,将会报错,并长度最大就是容量的大小


3、通过make函数构造

package main
import "fmt"
func main() {
  d := make([]int,5,10)
  fmt.Println(d)
  fmt.Printf("d的类型是%T\n",d)
}


输出结果:

[0 0 0 0 0]
d的类型是[]int


切片的类型


nil切片:var a []int,这种写法是没有初始化的,未声明内存,不能直接使用

普通切片:var a []int{} 这种写法后面带{},代表初始化切片,申请内存


只要申请了内存的切片就不是nil


package main
import "fmt"
func main() {
  var a []int
  var b = []int{}
  fmt.Printf("a的类型是%T,b的类型是%T\n",a,b)
  fmt.Printf("a的值是%v,b的值是%v\n",a,b)
  if a == nil{
  fmt.Println("a是nil")
  }
  if b == nil {
  fmt.Println("b是一个nil")
  }
}


输出结果:

a的类型是[]int,b的类型是[]int
a的值是[],b的值是[]
a是nil

从上面结果看出,值都hi是一样的,但是只有anil

再说一遍,只要申请了内存就不是nil


切片的拷贝


1、共用底层内存数据

a := make([]int,3) //使用make定义一个切片类型
  b := a             //声明b==a
  b[0] = 100     //赋值切片b第一个元素值为100
  fmt.Println(a)
  fmt.Println(b)


输出结果:

[100 0 0]
  [100 0 0]


换一种写法

package main
import "fmt"
func main() {
  //var a =make([]int,5)
  var a  = []int{1,2,3,4,5}
  b :=a
  b[1] =100
  fmt.Println(a)
  fmt.Println(b)
}


输出结果:

[1 100 3 4 5]
[1 100 3 4 5]


下面拷贝完成后,会生成两个内存位置,不是共享

//切片的拷贝,拷贝完成后,会生成两个内存位置,不是共享
     j :=[]int{1,2,3,4,5}
     k :=make([]int,5,5)
     copy(k,j)
     k[0] = 100
     fmt.Println(j,k)
     /*
     输出结果:[1 2 3 4 5] [100 2 3 4 5]
     */


切片的遍历


//切片的遍历,与数组一样
  c := []int{1,3,5,7}
  for index,value  := range c{
    fmt.Println(index,value)
  }

输出结果:

0 1
  1 3
  2 5
  3 7


切片的追加


//切片的追加
  f :=[]int{1,2,3}
  g :=[]int{4,5,6}
  h :=append(f,g...) //使用append参数可解决这个问题,第二个切片后面需要加...
  fmt.Println(f,g,h)
  /*
  输出结果:
  [1 2 3] [4 5 6] [1 2 3 4 5 6]
  */


切片的删除


也是append参数,把不要的信息不添加到组合中即可

//切片的删除,比如现在要删除一个切片的数据
i := []string{"北京","上海","广州","深圳"}
i =append(i[0:2],i[3:]...)  //从北京到上海组合起来,再从深圳到最后组合起来
fmt.Println(i)
 输出结果:  [北京 上海 深圳]     //间接的删除了广州
相关文章
|
9天前
|
存储 监控 算法
员工上网行为监控中的Go语言算法:布隆过滤器的应用
在信息化高速发展的时代,企业上网行为监管至关重要。布隆过滤器作为一种高效、节省空间的概率性数据结构,适用于大规模URL查询与匹配,是实现精准上网行为管理的理想选择。本文探讨了布隆过滤器的原理及其优缺点,并展示了如何使用Go语言实现该算法,以提升企业网络管理效率和安全性。尽管存在误报等局限性,但合理配置下,布隆过滤器为企业提供了经济有效的解决方案。
46 8
员工上网行为监控中的Go语言算法:布隆过滤器的应用
|
29天前
|
存储 Go 索引
go语言中数组和切片
go语言中数组和切片
40 7
|
29天前
|
Go 开发工具
百炼-千问模型通过openai接口构建assistant 等 go语言
由于阿里百炼平台通义千问大模型没有完善的go语言兼容openapi示例,并且官方答复assistant是不兼容openapi sdk的。 实际使用中发现是能够支持的,所以自己写了一个demo test示例,给大家做一个参考。
|
29天前
|
程序员 Go
go语言中结构体(Struct)
go语言中结构体(Struct)
102 71
|
28天前
|
存储 Go 索引
go语言中的数组(Array)
go语言中的数组(Array)
106 67
|
1月前
|
Go 索引
go语言for遍历数组或切片
go语言for遍历数组或切片
101 62
|
4天前
|
算法 安全 Go
Go 语言中实现 RSA 加解密、签名验证算法
随着互联网的发展,安全需求日益增长。非对称加密算法RSA成为密码学中的重要代表。本文介绍如何使用Go语言和[forgoer/openssl](https://github.com/forgoer/openssl)库简化RSA加解密操作,包括秘钥生成、加解密及签名验证。该库还支持AES、DES等常用算法,安装简便,代码示例清晰易懂。
32 12
|
29天前
|
存储 Go
go语言中映射
go语言中映射
38 11
|
1月前
|
Go
go语言for遍历映射(map)
go语言for遍历映射(map)
35 12
|
30天前
|
Go 索引
go语言使用索引遍历
go语言使用索引遍历
30 9