feat: slice 包新增优先级切片
This commit is contained in:
parent
34ca7f07d2
commit
93e63b1ace
|
@ -0,0 +1,151 @@
|
|||
package slice
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"sort"
|
||||
)
|
||||
|
||||
// NewPriority 创建一个优先级切片
|
||||
func NewPriority[V any](lengthAndCap ...int) *Priority[V] {
|
||||
p := &Priority[V]{}
|
||||
if len(lengthAndCap) > 0 {
|
||||
var length = lengthAndCap[0]
|
||||
var c int
|
||||
if len(lengthAndCap) > 1 {
|
||||
c = lengthAndCap[1]
|
||||
}
|
||||
p.items = make([]*PriorityItem[V], length, c)
|
||||
}
|
||||
return p
|
||||
}
|
||||
|
||||
// Priority 是一个优先级切片
|
||||
type Priority[V any] struct {
|
||||
items []*PriorityItem[V]
|
||||
}
|
||||
|
||||
// Len 返回切片长度
|
||||
func (slf *Priority[V]) Len() int {
|
||||
return len(slf.items)
|
||||
}
|
||||
|
||||
// Cap 返回切片容量
|
||||
func (slf *Priority[V]) Cap() int {
|
||||
return cap(slf.items)
|
||||
}
|
||||
|
||||
// Clear 清空切片
|
||||
func (slf *Priority[V]) Clear() {
|
||||
clear(slf.items)
|
||||
}
|
||||
|
||||
// Append 添加元素
|
||||
func (slf *Priority[V]) Append(v V, priority int) {
|
||||
slf.items = append(slf.items, NewPriorityItem[V](v, priority))
|
||||
slf.sort()
|
||||
}
|
||||
|
||||
// Appends 添加元素
|
||||
func (slf *Priority[V]) Appends(priority int, vs ...V) {
|
||||
for _, v := range vs {
|
||||
slf.Append(v, priority)
|
||||
}
|
||||
slf.sort()
|
||||
}
|
||||
|
||||
// Get 获取元素
|
||||
func (slf *Priority[V]) Get(index int) *PriorityItem[V] {
|
||||
return slf.items[index]
|
||||
}
|
||||
|
||||
// GetValue 获取元素值
|
||||
func (slf *Priority[V]) GetValue(index int) V {
|
||||
return slf.items[index].Value()
|
||||
}
|
||||
|
||||
// GetPriority 获取元素优先级
|
||||
func (slf *Priority[V]) GetPriority(index int) int {
|
||||
return slf.items[index].Priority()
|
||||
}
|
||||
|
||||
// Set 设置元素
|
||||
func (slf *Priority[V]) Set(index int, value V, priority int) {
|
||||
before := slf.items[index]
|
||||
slf.items[index] = NewPriorityItem[V](value, priority)
|
||||
if before.Priority() != priority {
|
||||
slf.sort()
|
||||
}
|
||||
}
|
||||
|
||||
// SetValue 设置元素值
|
||||
func (slf *Priority[V]) SetValue(index int, value V) {
|
||||
slf.items[index].v = value
|
||||
}
|
||||
|
||||
// SetPriority 设置元素优先级
|
||||
func (slf *Priority[V]) SetPriority(index int, priority int) {
|
||||
slf.items[index].p = priority
|
||||
slf.sort()
|
||||
}
|
||||
|
||||
// Action 直接操作切片,如果返回值不为 nil,则替换切片
|
||||
func (slf *Priority[V]) Action(action func(items []*PriorityItem[V]) []*PriorityItem[V]) {
|
||||
if replace := action(slf.items); replace != nil {
|
||||
slf.items = replace
|
||||
slf.sort()
|
||||
}
|
||||
}
|
||||
|
||||
// Range 遍历切片,如果返回值为 false,则停止遍历
|
||||
func (slf *Priority[V]) Range(action func(index int, item *PriorityItem[V]) bool) {
|
||||
for i, item := range slf.items {
|
||||
if !action(i, item) {
|
||||
break
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// RangeValue 遍历切片值,如果返回值为 false,则停止遍历
|
||||
func (slf *Priority[V]) RangeValue(action func(index int, value V) bool) {
|
||||
slf.Range(func(index int, item *PriorityItem[V]) bool {
|
||||
return action(index, item.Value())
|
||||
})
|
||||
}
|
||||
|
||||
// RangePriority 遍历切片优先级,如果返回值为 false,则停止遍历
|
||||
func (slf *Priority[V]) RangePriority(action func(index int, priority int) bool) {
|
||||
slf.Range(func(index int, item *PriorityItem[V]) bool {
|
||||
return action(index, item.Priority())
|
||||
})
|
||||
}
|
||||
|
||||
// String 返回切片字符串
|
||||
func (slf *Priority[V]) String() string {
|
||||
var vs []V
|
||||
for _, item := range slf.items {
|
||||
vs = append(vs, item.Value())
|
||||
}
|
||||
return fmt.Sprint(vs)
|
||||
}
|
||||
|
||||
// sort 排序
|
||||
func (slf *Priority[V]) sort() {
|
||||
if len(slf.items) <= 1 {
|
||||
return
|
||||
}
|
||||
sort.Slice(slf.items, func(i, j int) bool {
|
||||
return slf.items[i].Priority() < slf.items[j].Priority()
|
||||
})
|
||||
for i := 0; i < len(slf.items); i++ {
|
||||
if i == 0 {
|
||||
slf.items[i].prev = nil
|
||||
slf.items[i].next = slf.items[i+1]
|
||||
} else if i == len(slf.items)-1 {
|
||||
slf.items[i].prev = slf.items[i-1]
|
||||
slf.items[i].next = nil
|
||||
} else {
|
||||
slf.items[i].prev = slf.items[i-1]
|
||||
slf.items[i].next = slf.items[i+1]
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,34 @@
|
|||
package slice
|
||||
|
||||
// NewPriorityItem 创建一个优先级切片元素
|
||||
func NewPriorityItem[V any](v V, priority int) *PriorityItem[V] {
|
||||
return &PriorityItem[V]{v: v, p: priority}
|
||||
}
|
||||
|
||||
// PriorityItem 是一个优先级切片元素
|
||||
type PriorityItem[V any] struct {
|
||||
next *PriorityItem[V]
|
||||
prev *PriorityItem[V]
|
||||
v V
|
||||
p int
|
||||
}
|
||||
|
||||
// Value 返回元素值
|
||||
func (p *PriorityItem[V]) Value() V {
|
||||
return p.v
|
||||
}
|
||||
|
||||
// Priority 返回元素优先级
|
||||
func (p *PriorityItem[V]) Priority() int {
|
||||
return p.p
|
||||
}
|
||||
|
||||
// Next 返回下一个元素
|
||||
func (p *PriorityItem[V]) Next() *PriorityItem[V] {
|
||||
return p.next
|
||||
}
|
||||
|
||||
// Prev 返回上一个元素
|
||||
func (p *PriorityItem[V]) Prev() *PriorityItem[V] {
|
||||
return p.prev
|
||||
}
|
|
@ -0,0 +1,14 @@
|
|||
package slice_test
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"github.com/kercylan98/minotaur/utils/slice"
|
||||
"testing"
|
||||
)
|
||||
|
||||
func TestPriority_Append(t *testing.T) {
|
||||
var s = slice.NewPriority[string]()
|
||||
s.Append("name_1", 2)
|
||||
s.Append("name_2", 1)
|
||||
fmt.Println(s)
|
||||
}
|
Loading…
Reference in New Issue