Compare commits
	
		
			16 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 6b0395e54a | |||
| 385dad461c | |||
| 96e65f3550 | |||
| cc2ead6c4e | |||
| efc99a9658 | |||
| 7988676059 | |||
| 3901e3934f | |||
| abeecfd797 | |||
| 5735640ab6 | |||
| 26ea846e93 | |||
| 0592add8ca | |||
| ab6e6684f8 | |||
| 55928776b4 | |||
| 84cc7a9e05 | |||
| cdb1cf3e03 | |||
| 4f3cf3f537 | 
| @ -1,7 +1,5 @@ | |||||||
| package slices | package slices | ||||||
|  |  | ||||||
| type EqualityComparator[T comparable] func(a, b T) bool |  | ||||||
|  |  | ||||||
| func DefaultEqualityComparator[T comparable](a, b T) bool { | func DefaultEqualityComparator[T comparable](a, b T) bool { | ||||||
| 	return a == b | 	return a == b | ||||||
| } | } | ||||||
|  | |||||||
							
								
								
									
										22
									
								
								contains.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								contains.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,22 @@ | |||||||
|  | package slices | ||||||
|  |  | ||||||
|  | func IndexOf[T comparable](slice []T, value T) int { | ||||||
|  | 	return IndexOfCmp(slice, value, DefaultEqualityComparator[T]) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func IndexOfCmp[T comparable](slice []T, value T, cmp EqualityComparator[T]) int { | ||||||
|  | 	for i, v := range slice { | ||||||
|  | 		if cmp(v, value) { | ||||||
|  | 			return i | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return -1 | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func Contains[T comparable](slice []T, value T) bool { | ||||||
|  | 	return ContainsCmp(slice, value, DefaultEqualityComparator[T]) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func ContainsCmp[T comparable](slice []T, value T, cmp EqualityComparator[T]) bool { | ||||||
|  | 	return IndexOfCmp(slice, value, cmp) != -1 | ||||||
|  | } | ||||||
							
								
								
									
										15
									
								
								count.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								count.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,15 @@ | |||||||
|  | package slices | ||||||
|  |  | ||||||
|  | func Count[T comparable](slice []T, value T) int { | ||||||
|  | 	return CountCmp(slice, value, DefaultEqualityComparator[T]) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func CountCmp[T comparable](slice []T, value T, cmp EqualityComparator[T]) int { | ||||||
|  | 	c := 0 | ||||||
|  | 	for _, v := range slice { | ||||||
|  | 		if cmp(v, value) { | ||||||
|  | 			c++ | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return c | ||||||
|  | } | ||||||
							
								
								
									
										11
									
								
								each.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								each.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,11 @@ | |||||||
|  | package slices | ||||||
|  |  | ||||||
|  | func Each[T any](slice []T, f Consumer[T]) { | ||||||
|  | 	EachIndex(slice, func(_ int, v T) { f(v) }) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func EachIndex[T any](slice []T, f IndexedConsumer[T]) { | ||||||
|  | 	for i, v := range slice { | ||||||
|  | 		f(i, v) | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										67
									
								
								filter.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								filter.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,67 @@ | |||||||
|  | package slices | ||||||
|  |  | ||||||
|  | func Filter[T any](slice []T, f FilterFunc[T]) []T { | ||||||
|  | 	ret := make([]T, 0, len(slice)) | ||||||
|  | 	for _, v := range slice { | ||||||
|  | 		if f(v) { | ||||||
|  | 			ret = append(ret, v) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return ret | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func FindFirst[T any](slice []T, f FilterFunc[T]) (T, bool) { | ||||||
|  | 	for _, v := range slice { | ||||||
|  | 		if f(v) { | ||||||
|  | 			return v, true | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return *new(T), false | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func FindFirstIndex[T any](slice []T, f FilterFunc[T]) (int, bool) { | ||||||
|  | 	for i, v := range slice { | ||||||
|  | 		if f(v) { | ||||||
|  | 			return i, true | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return -1, false | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func FindLastIndex[T any](slice []T, f FilterFunc[T]) (int, bool) { | ||||||
|  | 	for i := len(slice); i >= 0; i-- { | ||||||
|  | 		if f(slice[i]) { | ||||||
|  | 			return i, true | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 	return -1, false | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func Not[T any](filter FilterFunc[T]) FilterFunc[T] { | ||||||
|  | 	return func(v T) bool { | ||||||
|  | 		return !filter(v) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func And[T any](filters ...FilterFunc[T]) FilterFunc[T] { | ||||||
|  | 	return func(v T) bool { | ||||||
|  | 		for _, filter := range filters { | ||||||
|  | 			if !filter(v) { | ||||||
|  | 				return false | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return true | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func Or[T any](filters ...FilterFunc[T]) FilterFunc[T] { | ||||||
|  | 	return func(v T) bool { | ||||||
|  | 		for _, filter := range filters { | ||||||
|  | 			if filter(v) { | ||||||
|  | 				return true | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		return false | ||||||
|  | 	} | ||||||
|  | } | ||||||
							
								
								
									
										9
									
								
								flat.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								flat.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,9 @@ | |||||||
|  | package slices | ||||||
|  |  | ||||||
|  | func Flat[T any](s ...[]T) []T { | ||||||
|  | 	out := make([]T, 0) | ||||||
|  | 	for _, v := range s { | ||||||
|  | 		out = append(out, v...) | ||||||
|  | 	} | ||||||
|  | 	return out | ||||||
|  | } | ||||||
							
								
								
									
										6
									
								
								go.mod
									
									
									
									
									
								
							
							
						
						
									
										6
									
								
								go.mod
									
									
									
									
									
								
							| @ -1,5 +1,5 @@ | |||||||
| module git.milar.in/milarin/slices | module git.tordarus.net/tordarus/slices | ||||||
|  |  | ||||||
| go 1.19 | go 1.23 | ||||||
|  |  | ||||||
| require git.milar.in/milarin/channel v0.0.7 | require git.tordarus.net/tordarus/gmath v0.0.7 | ||||||
|  | |||||||
							
								
								
									
										4
									
								
								go.sum
									
									
									
									
									
								
							
							
						
						
									
										4
									
								
								go.sum
									
									
									
									
									
								
							| @ -1,2 +1,2 @@ | |||||||
| git.milar.in/milarin/channel v0.0.7 h1:cVKtwgH/EE7U+XTHcoFCClJ4LR349KanzjX9xKwRcNg= | git.tordarus.net/tordarus/gmath v0.0.7 h1:tR48idt9AUL0r556ww3ZxByTKJEr6NWCTlhl2ihzYxQ= | ||||||
| git.milar.in/milarin/channel v0.0.7/go.mod h1:We83LTI8S7u7II3pD+A2ChCDWJfCkcBUCUqii9HjTtM= | git.tordarus.net/tordarus/gmath v0.0.7/go.mod h1:mO7aPlvNrGVE9UFXEuuACjZgMDsM63l3OcQy6xSQnoE= | ||||||
|  | |||||||
							
								
								
									
										21
									
								
								map.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								map.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,21 @@ | |||||||
|  | package slices | ||||||
|  |  | ||||||
|  | func Map[I, O any](slice []I, mapper Mapper[I, O]) []O { | ||||||
|  | 	ret := make([]O, 0, len(slice)) | ||||||
|  | 	for _, v := range slice { | ||||||
|  | 		ret = append(ret, mapper(v)) | ||||||
|  | 	} | ||||||
|  | 	return ret | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func MapError[I, O any](slice []I, mapper ErrorMapper[I, O]) ([]O, error) { | ||||||
|  | 	ret := make([]O, 0, len(slice)) | ||||||
|  | 	for _, old := range slice { | ||||||
|  | 		new, err := mapper(old) | ||||||
|  | 		if err != nil { | ||||||
|  | 			return nil, err | ||||||
|  | 		} | ||||||
|  | 		ret = append(ret, new) | ||||||
|  | 	} | ||||||
|  | 	return ret, nil | ||||||
|  | } | ||||||
							
								
								
									
										28
									
								
								of.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										28
									
								
								of.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,28 @@ | |||||||
|  | package slices | ||||||
|  |  | ||||||
|  | func Of[T any](values ...T) []T { | ||||||
|  | 	return values | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // OfMap returns a slice containing the return values of the Unmapper | ||||||
|  | // applied to any key-value pair in m | ||||||
|  | // The order is random | ||||||
|  | func OfMap[K comparable, V, T any](m map[K]V, unmapper MapUnmapper[K, V, T]) []T { | ||||||
|  | 	out := make([]T, 0, len(m)) | ||||||
|  | 	for k, v := range m { | ||||||
|  | 		out = append(out, unmapper(k, v)) | ||||||
|  | 	} | ||||||
|  | 	return out | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // UnmapKey is an Unmapper which returns the map key only | ||||||
|  | // and discards its value. It is supposed to be used with OfMap | ||||||
|  | func UnmapKey[K comparable, V any](key K, _ V) K { | ||||||
|  | 	return key | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // UnmapValue is an Unmapper which returns the map value only | ||||||
|  | // and discards its key. It is supposed to be used with OfMap | ||||||
|  | func UnmapValue[K comparable, V any](_ K, value V) V { | ||||||
|  | 	return value | ||||||
|  | } | ||||||
							
								
								
									
										23
									
								
								reduce.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								reduce.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,23 @@ | |||||||
|  | package slices | ||||||
|  |  | ||||||
|  | import "git.tordarus.net/tordarus/gmath" | ||||||
|  |  | ||||||
|  | func Reduce[T, A any](slice []T, reducer Reducer[A, T]) A { | ||||||
|  | 	res := new(A) | ||||||
|  | 	Each(slice, func(v T) { | ||||||
|  | 		*res = reducer(*res, v) | ||||||
|  | 	}) | ||||||
|  | 	return *res | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func SumReducer[N gmath.Number](a, b N) N { | ||||||
|  | 	return a + b | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func MinReducer[N gmath.Number](a, b N) N { | ||||||
|  | 	return gmath.Min(a, b) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | func MaxReducer[N gmath.Number](a, b N) N { | ||||||
|  | 	return gmath.Max(a, b) | ||||||
|  | } | ||||||
| @ -2,9 +2,15 @@ package slices | |||||||
|  |  | ||||||
| func Reverse[T any](slice []T) []T { | func Reverse[T any](slice []T) []T { | ||||||
| 	s := make([]T, len(slice)) | 	s := make([]T, len(slice)) | ||||||
| 	for i := 0; i < len(slice); i++ { | 	for i := range slice { | ||||||
| 		ri := len(slice) - 1 - i | 		ri := len(slice) - 1 - i | ||||||
| 		s[ri] = slice[i] | 		s[ri] = slice[i] | ||||||
| 	} | 	} | ||||||
| 	return s | 	return s | ||||||
| } | } | ||||||
|  |  | ||||||
|  | func ReverseInPlace[S ~[]T, T any](s S) { | ||||||
|  | 	for i, j := 0, len(s)-1; i < j; i, j = i+1, j-1 { | ||||||
|  | 		s[i], s[j] = s[j], s[i] | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | |||||||
							
								
								
									
										11
									
								
								search.go
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								search.go
									
									
									
									
									
								
							| @ -1,11 +0,0 @@ | |||||||
| package slices |  | ||||||
|  |  | ||||||
| func Search[T any](slice []T, f func(a, b T) T) T { |  | ||||||
| 	if len(slice) == 0 { |  | ||||||
| 		return *new(T) |  | ||||||
| 	} |  | ||||||
|  |  | ||||||
| 	value := slice[0] |  | ||||||
| 	Each(slice, func(v T) { value = f(value, v) }) |  | ||||||
| 	return value |  | ||||||
| } |  | ||||||
							
								
								
									
										52
									
								
								slices.go
									
									
									
									
									
								
							
							
						
						
									
										52
									
								
								slices.go
									
									
									
									
									
								
							| @ -1,52 +0,0 @@ | |||||||
| package slices |  | ||||||
|  |  | ||||||
| import ( |  | ||||||
| 	"git.milar.in/milarin/channel" |  | ||||||
| ) |  | ||||||
|  |  | ||||||
| func IndexOf[T comparable](slice []T, value T) int { |  | ||||||
| 	return IndexOfCmp(slice, value, DefaultEqualityComparator[T]) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func IndexOfCmp[T comparable](slice []T, value T, cmp EqualityComparator[T]) int { |  | ||||||
| 	for i, v := range slice { |  | ||||||
| 		if cmp(v, value) { |  | ||||||
| 			return i |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 	return -1 |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func Contains[T comparable](slice []T, value T) bool { |  | ||||||
| 	return ContainsCmp(slice, value, DefaultEqualityComparator[T]) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func ContainsCmp[T comparable](slice []T, value T, cmp EqualityComparator[T]) bool { |  | ||||||
| 	return IndexOfCmp(slice, value, cmp) != -1 |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func Map[I, O any](slice []I, mapper func(I) O) []O { |  | ||||||
| 	ret := make([]O, 0, len(slice)) |  | ||||||
| 	for _, v := range slice { |  | ||||||
| 		ret = append(ret, mapper(v)) |  | ||||||
| 	} |  | ||||||
| 	return ret |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func MapParallel[I, O any](slice []I, mapper func(I) O) []O { |  | ||||||
| 	return channel.ToSlice(channel.Map(channel.Of(slice...), mapper)) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func MapParallelWithRunner[I, O any](slice []I, runner channel.Runner, mapper func(I) O) []O { |  | ||||||
| 	return channel.ToSlice(channel.MapWithRunner(channel.Of(slice...), runner, mapper)) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func Each[T any](slice []T, f func(T)) { |  | ||||||
| 	EachIndex(slice, func(_ int, v T) { f(v) }) |  | ||||||
| } |  | ||||||
|  |  | ||||||
| func EachIndex[T any](slice []T, f func(int, T)) { |  | ||||||
| 	for i, v := range slice { |  | ||||||
| 		f(i, v) |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
							
								
								
									
										52
									
								
								to.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								to.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,52 @@ | |||||||
|  | package slices | ||||||
|  |  | ||||||
|  | import ( | ||||||
|  | 	"container/list" | ||||||
|  | ) | ||||||
|  |  | ||||||
|  | // Deref returns a slice containing all dereferenced values of s. | ||||||
|  | // The returned slice will be a dereferenced and continuous block of memory. | ||||||
|  | // Nil pointers are ignored. | ||||||
|  | func Deref[T any](s []*T) []T { | ||||||
|  | 	out := make([]T, 0, len(s)) | ||||||
|  | 	Each(s, func(v *T) { | ||||||
|  | 		if v != nil { | ||||||
|  | 			out = append(out, *v) | ||||||
|  | 		} | ||||||
|  | 	}) | ||||||
|  | 	return out | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Ref returns a slice containing pointers to all values of s. | ||||||
|  | // The order in s is preserved. | ||||||
|  | func Ref[T any](s []T) []*T { | ||||||
|  | 	out := make([]*T, 0, len(s)) | ||||||
|  | 	Each(s, func(v T) { | ||||||
|  | 		out = append(out, &v) | ||||||
|  | 	}) | ||||||
|  | 	return out | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // ToList returns a list.List containing all values of s | ||||||
|  | func ToList[T any](s []T) *list.List { | ||||||
|  | 	l := list.New() | ||||||
|  | 	Each(s, func(value T) { l.PushBack(value) }) | ||||||
|  | 	return l | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // ToMap returns a map containing all values of s. | ||||||
|  | // The map key-value pairs are determined by mapper | ||||||
|  | func ToMap[T any, K comparable, V any](s []T, mapper MapMapper[T, K, V]) map[K]V { | ||||||
|  | 	m := map[K]V{} | ||||||
|  | 	Each(s, func(value T) { | ||||||
|  | 		k, v := mapper(value) | ||||||
|  | 		m[k] = v | ||||||
|  | 	}) | ||||||
|  | 	return m | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // ToStructMap returns a struct{} map containing all values of s as keys. | ||||||
|  | // It is a shorthand for ToMap(s, func(value T) (T, struct{}) { return value, struct{}{} }) | ||||||
|  | func ToStructMap[T comparable](s []T) map[T]struct{} { | ||||||
|  | 	return ToMap(s, func(value T) (T, struct{}) { return value, struct{}{} }) | ||||||
|  | } | ||||||
							
								
								
									
										14
									
								
								types.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								types.go
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,14 @@ | |||||||
|  | package slices | ||||||
|  |  | ||||||
|  | type EqualityComparator[T comparable] = func(a, b T) bool | ||||||
|  | type FilterFunc[T any] = func(T) bool | ||||||
|  | type Consumer[T any] = func(T) | ||||||
|  | type IndexedConsumer[T any] = func(int, T) | ||||||
|  |  | ||||||
|  | type Mapper[I, O any] = func(I) O | ||||||
|  | type ErrorMapper[I, O any] = func(I) (O, error) | ||||||
|  |  | ||||||
|  | type MapMapper[T any, K comparable, V any] = func(T) (K, V) | ||||||
|  | type MapUnmapper[K comparable, V, T any] = func(K, V) T | ||||||
|  |  | ||||||
|  | type Reducer[A, T any] = func(acc A, value T) A | ||||||
		Reference in New Issue
	
	Block a user
	