深入解析Go语言编程中slice切片结构

作者:goldensun 时间:2024-04-25 15:01:23 

数组转换成切片


a := [10]int{}
fmt.Println(a)
 s1 := a[:10]  //取前10个元素 [5:]取 5-最后的元素
 fmt.Println(s1)


slice测试


a := []byte{'a', 'b', 'c', 'd', 'e', 'f', 'h'}
sa := a[2:5]
fmt.Println(string(sa))
sd1 := a[3:5]
fmt.Println(string(sd1)) //看看效果


我们看到这样的是slice_a指向Array_ori 其实是从c指向到k 我们用fmt.Println(cap(slice_a)) 结果肯定不是3

深入解析Go语言编程中slice切片结构

自己动手试一下下边这个
 

  
        a := []byte{'a', 'b', 'c', 'd', 'e', 'f', 'h'}
 sa := a[2:5]
fmt.Println(string(sa))
 s := sa[1:3]
 fmt.Println(string(s))
 s2 := sa[3:5]
 fmt.Println(string(s2))


slice是指向底层的数组,如果多个slice指向同一个的时候,其中一个改变了,其他的都改变。试一下下边这个
 


a := []int{1, 2, 3, 4, 5}
s1 := a[2:5]
s2 := a[1:3]
fmt.Println(s1, s2)
s1[0] = 9
fmt.Println(s1, s2)

切片是引用类型,即如果赋值切片给另一个切片,它们都指向同一底层数组。例如,如果某函数取切片参量,对其元素的改动会显现在调用者中,类似于传递一个底层数组的指针。因此 Read 函数可以接受切片参量,而不需指针和计数;切片的长度决定了可读数据的上限。这里是 os 包的 File 型的 Read 方法的签名:


  func (file *File) Read(buf []byte) (n int, err os.Error)


此方法返回读入字节数和可能的错误值。要读入一个大的缓冲 b 的首32字节, 切片(动词)缓冲。


      n, err := f.Read(buf[0:32])


这种切片常用且高效。实际上,先不管效率,此片段也可读缓冲的首32字节。


      var n int
      var err os.Error
      for i := 0; i < 32; i++ {
          nbytes, e := f.Read(buf[i:i+1])  // Read one byte.
          if nbytes == 0 || e != nil {
              err = e
              break
          }
          n += nbytes
      }


只要还在底层数组的限制内,切片的长度可以改变,只需赋值自己。切片的容量,可用内部函数 cap 取得,给出此切片可用的最大长度。下面的函数给切片添值。如果数据超过容量,切片重新分配,返回结果切片。此函数利用了 len 和 cap 对 nil 切片合法、返回0的事实。


Apppend的用法
 


a := make([]int, 3, 6)
 fmt.Printf("%p", a)
 a = append(a, 1, 2, 3)
 fmt.Printf("%v %p\n", a, a)
 a = append(a, 1, 2, 3)
 fmt.Printf("%v %p\n", a, a)


我们必须返回切片,因为尽管 Append 可以改变 slice 的元素, 切片自身(持有指针、长度和容量的运行态数据结构)是值传递的。添加切片的主意很有用,因此由内置函数 append 实现。


  func Append(slice, data[]byte) []byte {
      l := len(slice)
      if l + len(data) > cap(slice) {  // reallocate
          // Allocate double what's needed, for future growth.
          newSlice := make([]byte, (l+len(data))*2)
          // Copy data (could use bytes.Copy()).
          for i, c := range slice {
              newSlice[i] = c
          }
          slice = newSlice
      }
      slice = slice[0:l+len(data)]
      for i, c := range data {
          slice[l+i] = c
      }
      return slice
  }

当slice中append追加的元素超过了指向的容量,就会重新指向一个新的底层数组,所以一个底层数组的改变,不会带动其他的改变,试一下下边的代码


a := []int{1, 2, 3, 4, 5}
s1 := a[2:5]
s2 := a[1:3]
fmt.Println(s1, s2)
s2 = append(s2, 1, 2, 2, 3, 3, 4, 5)
s1[0] = 9
fmt.Println(s1, s2)


copy
这是一个拷贝的函数,下边的代码是从s2拷贝到s1然后我们会看到结果是[7 8 9 4 5]
如果是copy(s2,s1) 我们看到的结果是[1 2 3]
 


s1 := []int{1, 2, 3, 4, 5}
s2 := []int{7, 8, 9}
copy(s1, s2)
fmt.Println(s1)

标签:Go,切片
0
投稿

猜你喜欢

  • Host localhost is not allowed to conn

    2011-02-23 12:41:00
  • ASP动态生成的javascript表单验证代码

    2008-10-13 20:11:00
  • JavaScript修改作用域外变量的方法

    2024-04-10 16:12:01
  • python 安装移动复制第三方库操作

    2022-01-20 06:43:09
  • Python3 批量扫描端口的例子

    2021-12-20 05:17:37
  • Python实现自动签到脚本功能

    2022-07-24 21:53:40
  • 在IPython中执行Python程序文件的示例

    2023-02-20 09:06:14
  • Sqlserver事务备份和还原的实例代码(必看)

    2024-01-23 18:44:14
  • 如何在不支持数据库的ASP空间里获取更大的数据库权限?

    2010-06-18 19:35:00
  • Python最长回文子串问题

    2023-10-02 16:13:21
  • Pthon批量处理将pdb文件生成dssp文件

    2021-10-07 13:11:04
  • django从请求到响应的过程深入讲解

    2021-08-09 19:41:06
  • Python统计列表中的重复项出现的次数的方法

    2023-10-12 08:06:15
  • 如何创建并使用一个断开连接的记录集的数据访问页?

    2009-11-14 20:50:00
  • 细化解析:MySQL 数据库中对XA事务的限制

    2009-01-14 11:59:00
  • Linux下利用python实现语音识别详细教程

    2022-07-13 15:11:55
  • TensorFlow卷积神经网络AlexNet实现示例详解

    2022-06-14 21:17:30
  • Python内建类型float源码学习

    2022-05-05 17:07:37
  • 简单谈谈centos7中配置php

    2024-03-12 02:42:33
  • 出现“不能执行已释放的Script代码”错误的原因及解决办法

    2024-04-19 10:02:13
  • asp之家 网络编程 m.aspxhome.com