/array

develop golang,similar to javascript es6+

Primary LanguageGoMIT LicenseMIT

javascript 味的 golang 数组

概述

初衷

golang本身并没有提供太多数组相关的操作api,所以诞生了此工具包。开发过前端的朋友对es6的语法并不陌生,所以本工具包模拟es6+与js的常用方法来实现了这个工具包来操作数组。

安装

requirement
>= go 1.18
to install
go get github.com/butoften/array

常用方法

import (
	"github.com/butoften/array"
)
Reverse
  • Reverse 方法用于反转数组
  • 注意:此方法会改变原数组,要在不改变原始数组的情况下反转数组中的元素,使用 toReversed()。
fmt.Printf("----- Reverse Test Start -----\n")
originArr := []Test{
  {
    id:   1,
    name: "A",
  },
  {
    id:   2,
    name: "C",
  },
}
array.Reverse(&originArr)
fmt.Printf("originArr: %v\n", originArr) //[{2 C} {1 A}]
fmt.Printf("----- Reverse Test End -----\n")
ToReversed
  • 反转并返回新的数组
fmt.Printf("----- ToReversed Test Start -----\n")
originArr := []Test{
  {
    id:   1,
    name: "A",
  },
  {
    id:   2,
    name: "C",
  },
}
newArr := array.ToReversed(originArr)
fmt.Printf("originArr: %v\n", originArr) //[{1 A} {2 C}]
fmt.Printf("newArr: %v\n", newArr)       //[{2 C} {1 A}]
fmt.Printf("----- ToReversed Test End -----\n")
Map
  • 返回一个新数组,数组中的元素为原始数组元素调用函数处理后的值

  • 方法按照原始数组元素顺序依次处理元素。

  • 不会改变原始数组

fmt.Printf("----- Map Test Start -----\n")
objArr := []Test{
    Test{
        id:   1,
        name: "A",
    },
    Test{
        id:   2,
        name: "C",
    },
}
newArr := array.Map[Test, TestNew](objArr, func(item Test, index int) TestNew {
    return TestNew{
        Test: item,
        age:  1,
    }
})
fmt.Printf("newArr: %v\n", newArr) //[{{1 A} 1} {{2 C} 1}]
fmt.Printf("objArr: %v\n", objArr) //[{1 A} {2 C}]
fmt.Printf("----- Map Test End -----\n")
ForEach
  • 遍历数组,可以直接修改数组每一项的内容

  • 方法按照原始数组元素顺序依次处理元素。

  • 会改变原始数组

fmt.Printf("----- ForEach Test Start -----\n")
objArr := []Test{
    {
        id:   1,
        name: "A",
    },
    {
        id:   2,
        name: "C",
    },
}
fmt.Printf("objArr: %v\n", objArr) //[{1 A} {2 C}]
array.ForEach(objArr, func(item *Test, index int) {
    item.id += 1
})
fmt.Printf("objArr: %v\n", objArr) //[{2 A} {3 C}]
fmt.Printf("----- ForEach Test End -----\n\n")
Every
  • 用于检测数组所有元素是否都符合指定条件(通过函数提供)

  • 如果数组中检测到有一个元素不满足,则整个表达式返回 false,都满足时,返回true

  • 注:如果是空数组,直接返回false ,这里与js里不一样。

  • 不会改变原始数组

fmt.Printf("----- Every Test Start -----\n")
originArr := []int{1, 2, 4, 5}
res := array.Every[int](originArr, func(item, index int) bool {
    return item > 2
})
fmt.Printf("res: %v\n", res) //false
res = array.Every[int](originArr, func(item, index int) bool {
    return item > 0
})
fmt.Printf("res: %v\n", res) //true
fmt.Printf("----- Every Test End -----\n\n")
Some
  • 用于检测数组中的元素是否满足指定条件(函数提供),只要有一个满足条件,就返回true

  • 如果没有满足条件的元素,则返回false

  • 如果是空数组,直接返回false

  • 不会改变原始数组

fmt.Printf("----- Some Test Start -----\n")
originArr := []int{1, 2, 4, 5}
res := array.Some(originArr, func(item, index int) bool {
    return item > 2
})
fmt.Printf("res: %v\n", res) //true
res = array.Some(originArr, func(item, index int) bool {
    return item > 5
})
fmt.Printf("res: %v\n", res) //false
fmt.Printf("----- Some Test End -----\n\n")
Push

方法可向数组的末尾添加一个或多个元素,并返回新的长度。 注意:此方法改变数组的长度!

fmt.Printf("----- Push Test Start -----\n")
originArr := []int{3, 44, 38}
fmt.Printf("originArr: %v\n", originArr)
newLen := array.Push(&originArr, 1)
fmt.Printf("newLen: %v\n", newLen)
newLen = array.Push(&originArr, 2)
fmt.Printf("newLen: %v\n", newLen)
newLen = array.Push(&originArr, 3, 5, 6, 7)
fmt.Printf("originArr: %v\n", originArr)
fmt.Printf("newLen: %v\n", newLen)
fmt.Printf("----- Push Test End -----\n\n")
Pop
  • last, ok = array.Pop()
  • Pop() 方法用于删除数组的最后一个元素并返回删除的元素。
  • 注意:此方法改变数组的长度!
  • 空数组 Pop会失败 ok为false
fmt.Printf("----- Pop Test Start -----\n")
originArr := []int{3, 44, 8}
fmt.Printf("originArr: %v\n", originArr)
last, ok := array.Pop(&originArr)
fmt.Printf("last: %v-%v\n", last, ok)    //8-true
fmt.Printf("originArr: %v\n", originArr) //[3 44]

originArr = []int{} //空数组 pop会失败
fmt.Printf("originArr: %v\n", originArr)
last, ok = array.Pop(&originArr)
fmt.Printf("last: %v-%v\n", last, ok)    //0-false
fmt.Printf("originArr: %v\n", originArr) //[]
fmt.Printf("----- Pop Test End -----\n\n")
Shift
  • first, ok := array.Shift()
  • Shift() 方法用于把数组的第一个元素从其中删除,并返回第一个元素的值。
  • 此方法改变数组的长度!
  • 空数组Shift会失败,ok为false
fmt.Printf("----- Shift Test Start -----\n")
originArr := []int{3, 44, 38}
fmt.Printf("originArr: %v\n", originArr) //[3 44 38]
first, ok := array.Shift(&originArr)
fmt.Printf("originArr: %v\n", originArr) //[44 38]
fmt.Printf("first: %v-%v\n", first, ok)  //3 true
originArr = []int{}
fmt.Printf("originArr: %v\n", originArr) //[]
first, ok = array.Shift(&originArr)
fmt.Printf("originArr: %v\n", originArr) //[]
fmt.Printf("first: %v-%v\n", first, ok)  //0 false
fmt.Printf("----- Shift Test End -----\n\n")
Unshift
  • UnShift() 方法可向数组的开头添加一个或更多元素,并返回新的长度

  • 此方法改变数组的长度!

fmt.Printf("----- UnShift Test Start -----\n")
originArr := []string{"Banana", "Orange", "Apple", "Mango"}
fmt.Printf("originArr: %v\n", originArr)
length := array.UnShift(&originArr, "Lemon", "Pineapple") //[Banana Orange Apple Mango]
fmt.Printf("originArr: %v\n", originArr)            //[Lemon Pineapple Banana Orange Apple Mango]
fmt.Printf("length: %v\n", length)                  //6
fmt.Printf("----- UnShift Test End -----\n\n")
Slice
  • 从已有的数组中返回选定区间的新元素数组,返回类型为array.Array,可以继续使用此工具包的各种方法

  • 此方法不会对源数组产生影响(原生切片因扩容规则:不扩容的情况下,会对源切片产生影响)

  • 如果你不喜欢此方法你依然可以使用原生切片截取方式[:]来操作,但要注意扩容规则

  • 因为js里slice方法支持起始与结束值为负,所以本工具包也实现了相同的算法,请查看下面非常规案例

fmt.Printf("----- Slice Test Start -----\n")
a := []int{1, 2, 3, 4, 5}
b := a[0:1]
fmt.Printf("a: %v\n", a) //[1 2 3 4 5]
fmt.Printf("b: %v\n", b) //[1]
b = append(b, 6, 7, 8, 9)
fmt.Printf("a: %v\n", a) //[1 6 7 8 9] 因为整体cap不变,没有扩容,所以b切片在append时,a切片也受到了影响
fmt.Printf("b: %v\n", b) //[1 6 7 8 9]

arr := []int{1, 2, 3, 4, 5}
newArr := array.Slice(arr, 0, 1)
fmt.Printf("arr: %v\n", arr)       //[1 2 3 4 5]
fmt.Printf("newArr: %v\n", newArr) //[1]
newArr = append(newArr, 6, 7)
array.Push(&newArr, 8, 9)
fmt.Printf("arr: %v\n", arr)       //[1 2 3 4 5] array.Slice方法由于重新初始切片,所以实现了彼此互不影响
fmt.Printf("newArr: %v\n", newArr) //[1 6 7 8 9]

c := newArr[0:1] //使用原生切片截取方式 但是要注意扩容规则
c = append(c, 66, 77, 88, 99)
fmt.Printf("原生[:] 截取并append后 newArr: %v\n", newArr) //[1 66 77 88 99] 受扩容规则影响
fmt.Printf("原生[:] 截取并append后 c: %v\n", c)           //[1 66 77 88 99]

//以下是非常规操作案例
d := []int{1, 2, 3, 4, 5}
e := array.Slice(d, 0, -1)
fmt.Printf("e: %v\n", e) //[1 2 3 4]
e = array.Slice(d, 0, 20)
fmt.Printf("e: %v\n", e) //[1 2 3 4 5]
e = array.Slice(d, -20, 20)
fmt.Printf("e: %v\n", e) //[1 2 3 4 5]
e = array.Slice(d, -1, 20)
fmt.Printf("e: %v\n", e) //[5]
e = array.Slice(d, -5, 20)
fmt.Printf("e: %v\n", e) //[1 2 3 4 5]
e = array.Slice(d, -1, -2)
fmt.Printf("e: %v\n", e) //[]
e = array.Slice(d, -2, -1)
fmt.Printf("e: %v\n", e) //[4]

e = array.Slice(d, -20, -1)
fmt.Printf("e: %v\n", e) //[1 2 3 4]
fmt.Printf("----- Slice Test End -----\n\n")
Splice
  • Splice(index int, howMany int, args ...T)
  • @param index 规定从何处添加或删除元素,该参数是插入元素或删除元素的起始下标,必须是整数
  • @param howMany 规定应该删除多少元素
    • howMany 为0时,表示不删除,如果 args有值则在index处插入args
    • howMany 为负数时,相当于0,不删除
    • howMany 为正数时,表示删除数量,删除后,如果 args有值则在index处插入args
  • @param args 要在index处添加的多个元素
  • 用于添加或删除数组中的元素
  • 会改变原始数组
  • 返回的是含有被删除的元素的数组
fmt.Printf("----- Splice Test Start -----\n")
arr := []int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}
delArr := array.Splice(&arr, 3, 0, 11, 12, 13)
fmt.Printf("delArr: %v\n", delArr) //[]
fmt.Printf("arr: %v\n", arr)       //[1 2 3 11 12 13 4 5 6 7 8 9 10]

arr = []int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}
delArr = array.Splice(&arr, 3, 2, 11, 12, 13)
fmt.Printf("delArr: %v\n", delArr) //[4 5]
fmt.Printf("arr: %v\n", arr)       //[1 2 3 11 12 13 6 7 8 9 10]

arr = []int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}
delArr = array.Splice(&arr, 3, 3, 11, 12)
fmt.Printf("delArr: %v\n", delArr) //[4 5 6]
fmt.Printf("arr: %v\n", arr)       //[1 2 3 11 12 7 8 9 10]

arr = []int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}
delArr = array.Splice(&arr, 3, 4, 11, 12)
fmt.Printf("delArr: %v\n", delArr) //[4 5 6 7]
fmt.Printf("arr: %v\n", arr)       //[1 2 3 11 12 8 9 10]

arr = []int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}
delArr = array.Splice(&arr, -3, 5, 11, 12)
fmt.Printf("delArr: %v\n", delArr) //[8 9 10]
fmt.Printf("arr: %v\n", arr)       //[1 2 3 4 5 6 7 11 12]

arr = []int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}
delArr = array.Splice(&arr, -3, -5, 11, 12)
fmt.Printf("delArr: %v\n", delArr) //[]
fmt.Printf("arr: %v\n", arr)       //[1 2 3 4 5 6 7 11 12 8 9 10]

arr = []int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}
delArr = array.Splice(&arr, 0, 5, 11, 12)
fmt.Printf("delArr: %v\n", delArr) //[1 2 3 4 5]
fmt.Printf("arr: %v\n", arr)       //[11 12 6 7 8 9 10]

arr = []int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}
delArr = array.Splice(&arr, 0, 50, 11, 12)
fmt.Printf("delArr: %v\n", delArr) //[1 2 3 4 5 6 7 8 9 10]
fmt.Printf("arr: %v\n", arr)       //[11 12]

arr = []int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}
delArr = array.Splice(&arr, 0, 0)
fmt.Printf("delArr: %v\n", delArr) //[]
fmt.Printf("arr: %v\n", arr)       //[1 2 3 4 5 6 7 8 9 10]

arr = []int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}
delArr = array.Splice(&arr, 50, 11, 12)
fmt.Printf("delArr: %v\n", delArr) //[]
fmt.Printf("arr: %v\n", arr)       //[1 2 3 4 5 6 7 8 9 10 12]

arr = []int{1, 2, 3, 4, 5, 6, 7, 8, 9, 10}
delArr = array.Splice(&arr, 10, 50, 11, 12)
fmt.Printf("delArr: %v\n", delArr) //[]
fmt.Printf("arr: %v\n", arr)       //[1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12]
fmt.Printf("----- Splice Test End -----\n\n")
Empty
  • 切片清空

  • 只清len不清cap

fmt.Printf("----- Empty Test Start -----\n")
arr := []int{3, 44, 38}
fmt.Printf("arr: %v-%v-%v\n", arr, len(arr), cap(arr)) //arr: [3 44 38]-3-3
array.Empty(&arr)
fmt.Printf("arr: %v-%v-%v\n", arr, len(arr), cap(arr)) //arr: []-0-3
fmt.Printf("----- Empty Test End -----\n\n")
BrokenEmpty
  • 切片断开式清空
  • len与cap同时清空,断开底层数组
fmt.Printf("----- BrokenEmpty Test Start -----\n")
arr := []int{3, 44, 38}
fmt.Printf("arr: %v-%v-%v\n", arr, len(arr), cap(arr)) //arr: [3 44 38]-3-3
array.BrokenEmpty(&arr)
fmt.Printf("arr: %v-%v-%v\n", arr, len(arr), cap(arr)) //arr: []-0-0
fmt.Printf("----- BrokenEmpty Test End -----\n\n")
Find
  • 根据回调函数进行搜索
  • 返回结果为 res, exist 其中res为目标结果 ,exist 为bool
  • 从前向后遍历
type Test struct {
	id   int
	name string
}

func main() {
    fmt.Printf("----- Find Test Start -----\n")
	var arr []Test
	array.Push(&arr, Test{
		id:   1,
		name: "A",
	})
	array.Push(&arr, Test{
		id:   2,
		name: "B",
	})

	res, exist := array.Find(arr, func(item Test, key int) bool {
		return item.name == "B" && item.id == 2
	})
	if exist {
		fmt.Printf("res: %v\n", res) //res: {2 B}
	} else {
		fmt.Printf("not found: %v\n", res)
	}
	fmt.Printf("----- Find Test End -----\n\n")
}
FindLast

与Find类似,不同的是,从后向前遍历

fmt.Printf("----- FindLast Test Start -----\n")
	var objArr []Test
	array.Push(&objArr, Test{
		id:   1,
		name: "A",
	})
	array.Push(&objArr, Test{
		id:   2,
		name: "C",
	})
	res, exist := array.FindLast(objArr, func(item Test, key int) bool {
		return item.name == "C" && item.id == 2
	})
	if exist {
		fmt.Printf("res: %v\n", res) //res: {2 C}
	} else {
		fmt.Printf("not found: %v\n", res)
	}
	fmt.Printf("----- FindLast Test End -----\n\n")
FindIndex
  • FindIndex()返回符合传入回调函数条件的第一个元素索引位置

  • 如果没有符合条件的元素返回 -1

  • 从前向后遍历

fmt.Printf("----- FindIndex Test Start -----\n")
ages := []int{3, 10, 18, 20}
index := array.FindIndex(ages, func(item, index int) bool {
    return item == 18
})
fmt.Printf("index: %v\n", index) //index: 2
fmt.Printf("----- FindIndex Test End -----\n\n")
FindLastIndex
  • 与FindIndex类似,不同的是,从后向前遍历

  • 如果没有符合条件的元素返回 -1

fmt.Printf("----- FindLastIndex Test Start -----\n")
ages := []int{3, 10, 18, 20}
index := array.FindLastIndex(ages, func(item, index int) bool {
    return item > 10
})
fmt.Printf("index: %v\n", index) //index: 3
fmt.Printf("----- FindLastIndex Test End -----\n\n")
Filter
  • 根据回调函数里的条件进行过滤
  • 返回结果依然是一个数组,如果没有匹配项,则返回空数组
type Test struct {
	id   int
	name string
}

func main() {
	fmt.Printf("----- Filter Test Start -----\n")
	var arr []Test
	array.Push(&arr, Test{
		id:   1,
		name: "A",
	})
	array.Push(&arr, Test{
		id:   2,
		name: "B",
	})

	resFilter := array.Filter(arr, func(item Test, key int) bool {
		return item.name == "A"
	})
	fmt.Printf("resFilter: %v\n", resFilter) //resFilter: [{1 A}]
	fmt.Printf("----- Filter Test End -----\n\n")
}
Flat
  • 根据指定深度递归地将所有子数组元素拼接到新的数组中

  • 返回新的数组

  • depth 默认值为1

fmt.Printf("----- Flat Test Start -----\n")
type AnySlice = []any
originArr := AnySlice{
  1,
  AnySlice{
    2,
    AnySlice{
      3,
      AnySlice{4},
    },
    5,
  },
}
fmt.Printf("originArr: %v\n", originArr) // [1 [2 [3 [4]] 5]]
newArr := array.Flat(originArr)
fmt.Printf("newArr : %v\n", newArr) // [1 2 [3 [4]] 5]
newArr = array.Flat(originArr, 1)
fmt.Printf("newArr : %v\n", newArr) // [1 2 [3 [4]] 5]
newArr = array.Flat(originArr, 2)
fmt.Printf("newArr : %v\n", newArr) // [1 2 3 [4] 5]
newArr = array.Flat(originArr, 3)
fmt.Printf("newArr : %v\n", newArr) // [1 2 3 4 5]

fmt.Printf("----- Flat Test End -----\n")

排序

Sort 原生排序

golang原生sort.Slice排序封装

type Test struct {
	id   float32
	name string
}
func main(){
  fmt.Printf("----- Sort Test Start -----\n")
	arrTest := []TestFloat{
		{
			id:   3.2,
			name: "A1",
		},
		{
			id:   4.2,
			name: "A2",
		},
		{
			id:   38.9,
			name: "A3",
		},
		{
			id:   5.4,
			name: "A4",
		},
		{
			id:   38.7,
			name: "A5",
		},
		{
			id:   38.5,
			name: "A6",
		},
	}
	fmt.Printf("arr: %v\n", arrTest)
	//升序
	array.Sort(&arrTest, func(a TestFloat, b TestFloat) bool {
		return a.id < b.id
	})
	fmt.Printf("升序 Sort: %v\n", arrTest)
	//降序
	array.Sort(&arrTest, func(a TestFloat, b TestFloat) bool {
		return a.id > b.id
	})
	fmt.Printf("降序 Sort: %v\n", arrTest)
	fmt.Printf("----- Sort Test End -----\n\n")
}
BubbleSort 冒泡排序
fmt.Printf("----- BubbleSort Test Start -----\n")
arr := []int{3, 44, 38, 5, 15, 36, 26, 27, 2, 46, 4, 19, 50, 48}
fmt.Printf("arr: %v\n", arr)
//升序
array.BubbleSort(&arr, func(a int, b int) bool {
    return a < b
})
fmt.Printf("升序 arr: %v\n", arr)
fmt.Printf("----- BubbleSort Test End -----\n\n")
SelectSort 选择排序
fmt.Printf("----- SelectSort Test Start -----\n")
arr := []int{3, 44, 38, 5, 15, 36, 26, 27, 2, 46, 4, 19, 50, 48}
fmt.Printf("arr: %v\n", arr)
//升序
array.SelectSort(&arr, func(a int, b int) bool {
    return a < b
})
fmt.Printf("升序 arr: %v\n", arr)
fmt.Printf("----- SelectSort Test End -----\n\n")
InsertSort 插入排序
fmt.Printf("----- InsertSort Test Start -----\n")
arr := []int{3, 44, 38, 5, 15, 36, 26, 27, 2, 46, 4, 19, 50, 48}
fmt.Printf("arr: %v\n", arr)
//升序
array.InsertSort(&arr, func(a int, b int) bool {
    return a < b
})
fmt.Printf("升序 arr: %v\n", arr)
fmt.Printf("----- InsertSort Test End -----\n\n")
ShellSort 希尔排序
fmt.Printf("----- ShellSort Test Start -----\n")
arr := []int{3, 44, 38, 5, 15, 36, 26, 27, 2, 46, 4, 19, 50, 48}
fmt.Printf("arr: %v\n", arr)
//升序
array.ShellSort(&arr, func(a int, b int) bool {
    return a < b
})
fmt.Printf("升序 arr: %v\n", arr)
fmt.Printf("----- ShellSort Test End -----\n\n")
MergeSort 归并排序
fmt.Printf("----- MergeSort Test Start -----\n")
arr := []int{3, 44, 38, 5, 15, 36, 26, 27, 2, 46, 4, 19, 50, 48}
fmt.Printf("arr: %v\n", arr)
//升序
array.MergeSort(&arr, func(a int, b int) bool {
    return a < b
})
fmt.Printf("升序 arr: %v\n", arr)
fmt.Printf("----- MergeSort Test End -----\n\n")
QuickSort 快速排序
fmt.Printf("----- QuickSort Test Start -----\n")
arr := []int{3, 44, 38, 5, 15, 36, 26, 27, 2, 46, 4, 19, 50, 48}
fmt.Printf("arr: %v\n", arr)
//升序
array.QuickSort(&arr, func(a int, b int) bool {
    return a < b
})
fmt.Printf("升序 arr: %v\n", arr)
fmt.Printf("----- QuickSort Test End -----\n\n")