我的Go+语言初体验——GO+实现数据结构之【数组 切片 Map】(1)

简介: 我的Go+语言初体验——GO+实现数据结构之【数组 切片 Map】(1)

随着Go+ 的发布不久,正准备学习GO+,又刚好赶上这次征文的活动

以写促学,今后,我将带大家使用 GO+ 逐步 实现常见的数据结构 ,刚好一边学习GO+一边复习数据结构

今天是第一篇

我们使用 GO+ 来学习数据结构 之 数组、切片与Map

数组基本操作

声明

一维数组

初始化数组长度为1

var linearArray [1]int

根据数组内容自己推算长度


linearArray := [...]int{1,2,3}

对于var和:=的区别在于:var适合全局:=只能试用局部


Golang 中数组的初始值如果不声明,则默认是全部有初值的


比如 linearArray 这个数组声明了长度为 1, Go 的编译器会把这 1 个元素全都初始化为 0


bool 值类型的数组,如果不做赋值操作,则初始值全为 false


我们可以来试验一下

package main
func main() {
  var linearArray [1]int
  print linearArray[0]
}

1.png

package main
func main() {
  var linearArray [1]bool
  print linearArray[0]
}

2.png

二维数组

二维数组声明和其它语言差不多

var twoDimensionalArray [1][2]int

遍历

linearArray := [...]int{1,2,3}
for x <- linearArray, x > 1 {
    print x
}

3.png

另一种遍历方法:

linearArray := [...]int{5, 4, 3, 2, 1, 0}
for index, value := range linearArray {
    println "索引:",index,", 值: ",value
}

4.png

当然你也可以使用最基础的方式遍历


linearArray := [...]int{5, 4, 3, 2, 1, 0}
for i := 0; i < len(linearArray); i++ {
    println linearArray[i]
}

5.png

切片基本操作

数组是按值传递的(即是传递的副本),而切片是引用类型,传递切片的成本非常小,而且不定长,如果很难理解的话,可以暂时把它理解为"可以自定义长度的列表"或者是“可扩容的数组”


语法如下:


  • make([ ]Type, length, capacity)

其长度为切片的容量 capacity

  • make([ ]Type, length)

其长度记为切片的长度 length

  • [ ]Type{}
  • [ ]Type{value1, …, valueN}

使用make()函数创建切片、映射和通道,通过内置的函数 append() 来增加切片的容量。

slice := []int{}
for i := 0; i < 5; i++ {
    slice = append(slice, i)
}
print slice

6.png

此外,和python类似,我们可以这样提取区间数据

slice := []int{}
for i := 0; i < 5; i++ {
    slice = append(slice, i)
}
print slice[1:3]

7.png

提取slice中第一个开始到第三个数值(不包括第三个)


[n:m] 的意思就是取区间 [n,m)


从位置下标的角度上来说 [n-1,m-1) 可能更合适, 为什么呢?


我们可以来实现下这个功能

func ExtractInterval(sliceObject []int, start int, end int) []int {
  new_sliceObject := []int{}
  for i := start - 1; i < end-1; i++ {
  new_sliceObject = append(new_sliceObject, sliceObject[i])
  }
  return new_sliceObject
}
slice := []int{}
for i := 0; i < 5; i++ {
  slice = append(slice, i)
}
print ExtractInterval(slice, 1, 3)

8.png

你可以发现:在我们ExtractInterval方法中所实现的就是提取slice数组中的下标位置是0和1的数据



更多知识点


Go+ 切片的扩容规则


切片的扩容并不会改变原来的切片,它会生成一个容量更大的切片,然后将把原有的元素和新元素一并拷贝到新切片中. 一般的情况下,你可以简单地认为新切片的容量将会是原切片容量的 2 倍.


当原切片的长度大于或等于 1024 时,Go+会以原容量的 1.25 倍作为新容量的基准. 如果容量增加的太快,很容易浪费空间.


如果我们一次追加的元素过多,要大于 2 倍,那么新容量就会以新长度为基准. 比如你现在的切片长度为 5,现在一下往里面添加了 25 个元素,那么 Go+ 会直接创建一个长度为 30 的底层数组,然后把所有的数据拷贝进去.


Go+ 切片其底层数组永远不会改变


当切片需要扩容时,新的切片诞生的同时也会创建出新的底层数组,它只是把原数组的数据拷贝了进来,并未对其做任何的修改.




Map literal(集合)

声明

定义,个人感觉和Python是差不多的

MapLiteral := {"a": 1, "b": 2, "c": 3, "d": 4, "e": 5}

遍历

MapLiteral := {"a": 1, "b": 2, "c": 3, "d": 4, "e": 5}
for k, v := range MapLiteral {
  println "k:[", k, "].v:[", v, "]\n"
}

9.png

删除

delete() 方法可以实现

delete(集合, key)

还可以试一下这样定义集合,顺便试验下delete() 方法

MapLiteral := map[string]int{"a": 1, "b": 2, "c": 3, "d": 4, "e": 5}
delete(MapLiteral, "a")
for k, v := range MapLiteral {
  println "k:[", k, "].v:[", v, "]\n"
}

10.png

修改与增加

可以直接这样实现

MapLiteral := map[string]int{"a": 1, "b": 2, "c": 3, "d": 4, "e": 5}
MapLiteral["a"] = 0
MapLiteral["f"] = 6
for k, v := range MapLiteral {
  println "k:[", k, "].v:[", v, "]\n"
}

11.png


练习


请使用GO+ 实现简单的 冒泡排序

Array := [...]int{123, 343, 143, 524, 462, 5672, 3467}
// 冒泡排序
for i := 0; i < len(Array); i++ {
  for j := 1; j < len(Array)-i; j++ {
  if Array[j] < Array[j-1] {
    Array[j], Array[j-1] = Array[j-1], Array[j]
  }
  }
}
print Array

12.png

目录
相关文章
|
3月前
|
存储 JavaScript Java
(Python基础)新时代语言!一起学习Python吧!(四):dict字典和set类型;切片类型、列表生成式;map和reduce迭代器;filter过滤函数、sorted排序函数;lambda函数
dict字典 Python内置了字典:dict的支持,dict全称dictionary,在其他语言中也称为map,使用键-值(key-value)存储,具有极快的查找速度。 我们可以通过声明JS对象一样的方式声明dict
300 1
|
7月前
|
存储 JSON 安全
Go语言切片,使用技巧与避坑指南
Go语言中的切片(Slice)是动态引用数组的高效数据结构,支持扩容与截取。本文从切片基础、常用操作到高级技巧全面解析,涵盖创建方式、`append`扩容机制、共享陷阱及安全复制等内容。通过代码示例详解切片特性,如预分配优化性能、区分`nil`与空切片、处理多维切片等。掌握这些核心知识点,可编写更高效的Go代码。
257 2
|
3月前
|
存储 Java Go
【Golang】(3)条件判断与循环?切片和数组的关系?映射表与Map?三组关系傻傻分不清?本文带你了解基本的复杂类型与执行判断语句
在Go中,条件控制语句总共有三种if、switch、select。循环只有for,不过for可以充当while使用。如果想要了解这些知识点,初学者进入文章中来感受吧!
187 1
|
6月前
|
数据采集 机器学习/深度学习 存储
Go语言实战案例 - 找出切片中的最大值与最小值
本案例通过实现查找整数切片中的最大值与最小值,帮助初学者掌握遍历、比较和错误处理技巧,内容涵盖算法基础、应用场景及完整代码示例,适合初学者提升编程能力。
|
7月前
|
Go 索引
Go语言中使用切片需要注意什么?
本文详细讲解了Go语言中切片(Slice)的使用方法与注意事项。切片是对数组连续片段的引用,具有灵活的操作方式。文章从定义与初始化、长度与容量、自动扩容、共享底层数组、复制、边界检查、零值到拼接等方面展开,并配以示例代码演示。通过学习,读者可深入了解切片的工作原理及优化技巧,避免常见陷阱,提升编程效率与代码质量。
195 2
|
11月前
|
测试技术 Go API
Go 切片导致 rand.Shuffle 产生重复数据的原因与解决方案
在 Go 语言开发中,使用切片时由于其底层数据共享特性,可能会引发意想不到的 Bug。本文分析了 `rand.Shuffle` 后切片数据重复的问题,指出原因在于切片是引用类型,直接赋值会导致底层数组共享,进而影响原始数据。解决方案是使用 `append` 进行数据拷贝,确保独立副本,避免 `rand.Shuffle` 影响原始数据。总结强调了切片作为引用类型的特性及正确处理方法,确保代码稳定性和正确性。
299 82
|
8月前
|
安全 Go 开发者
Go语言之切片的原理与用法 - 《Go语言实战指南》
切片(slice)是Go语言中用于处理变长数据集合的核心结构,基于数组的轻量级抽象,具有灵活高效的特点。切片本质是一个三元组:指向底层数组的指针、长度(len)和容量(cap)。本文详细介绍了切片的声明与初始化方式、基本操作(如访问、修改、遍历)、长度与容量的区别、自动扩容机制、共享与副本处理、引用类型特性以及常见陷阱。通过理解切片的底层原理,开发者可以更高效地使用这一数据结构,优化代码性能。
292 13
|
7月前
|
Go
【LeetCode 热题100】DP 实战进阶:最长递增子序列、乘积最大子数组、分割等和子集(力扣300 / 152/ 416 )(Go语言版)
本文深入解析三道经典的动态规划问题:**最长递增子序列(LIS)**、**乘积最大子数组** 和 **分割等和子集**。 - **300. LIS** 通过 `dp[i]` 表示以第 `i` 个元素结尾的最长递增子序列长度,支持 O(n²) 动态规划与 O(n log n) 的二分优化。 - **152. 乘积最大子数组** 利用正负数特性,同时维护最大值与最小值的状态转移方程。 - **416. 分割等和子集** 转化为 0-1 背包问题,通过布尔型 DP 实现子集和判断。 总结对比了三题的状态定义与解法技巧,并延伸至相关变种问题,助你掌握动态规划的核心思想与灵活应用!
327 1
|
8月前
|
Go 索引
Go语言数组的定义与操作 - 《Go语言实战指南》
本文介绍了 Go 语言中的数组(Array)相关知识,包括定义、初始化方式(默认、显式、指定索引及自动推导长度)、访问与修改、遍历方法(for 循环和 for range)、值类型特性(复制行为)、多维数组支持以及其与切片的区别。数组是定长且同类型的集合,适合性能敏感场景,但实际开发中更常用动态的切片(slice)。
288 11
|
8月前
|
人工智能 Go
[go]Slice 切片原理
本文详细介绍了Go语言中的切片(slice)数据结构,包括其定义、创建方式、扩容机制及常见操作。切片是一种动态数组,依托底层数组实现,具有灵活的扩容和传递特性。文章解析了切片的内部结构(包含指向底层数组的指针、长度和容量),并探讨了通过`make`创建切片、基于数组生成切片以及切片扩容的规则。此外,还分析了`append`函数的工作原理及其可能引发的扩容问题,以及切片拷贝时需要注意的细节。最后,通过典型面试题深入讲解了切片在函数间传递时的行为特点,帮助读者更好地理解和使用Go语言中的切片。
270 0