gset

集合,即不可重复的一组元素,元素项可以为任意类型。

同时,gset支持可选的并发安全参数选项,支持并发安全的场景。

使用场景

集合操作。

使用方式

  1. import "github.com/gogf/gf/g/container/gset"

接口文档https://godoc.org/github.com/gogf/gf/g/container/gset

使用示例

示例1,基本使用

  1. package main
  2. import (
  3. "github.com/gogf/gf/g/container/gset"
  4. "fmt"
  5. )
  6. func main() {
  7. // 创建一个非并发安全的集合对象
  8. s := gset.New(true)
  9. // 添加数据项
  10. s.Add(1)
  11. // 批量添加数据项
  12. s.Add([]interface{}{1, 2, 3}...)
  13. // 集合数据项大小
  14. fmt.Println(s.Size())
  15. // 集合中是否存在指定数据项
  16. fmt.Println(s.Contains(2))
  17. // 返回数据项slice
  18. fmt.Println(s.Slice())
  19. // 删除数据项
  20. s.Remove(3)
  21. // 遍历数据项
  22. s.Iterator(func(v interface{}) bool {
  23. fmt.Println("Iterator:", v)
  24. return true
  25. })
  26. // 将集合转换为字符串
  27. fmt.Println(s.String())
  28. // 并发安全写锁操作
  29. s.LockFunc(func(m map[interface{}]struct{}) {
  30. m[4] = struct{}{}
  31. })
  32. // 并发安全读锁操作
  33. s.RLockFunc(func(m map[interface{}]struct{}) {
  34. fmt.Println(m)
  35. })
  36. // 清空集合
  37. s.Clear()
  38. fmt.Println(s.Size())
  39. }

执行后,输出结果为:

  1. 3
  2. true
  3. [1 2 3]
  4. Iterator: 1
  5. Iterator: 2
  6. [1 2]
  7. map[1:{} 2:{} 4:{}]
  8. 0

示例2,交差并补集

我们可以使用以下方法实现交差并补集,并返回一个新的结果集合,

  1. func (set *Set) Intersect(others ...*Set) (newSet *Set)
  2. func (set *Set) Diff(others ...*Set) (newSet *Set)
  3. func (set *Set) Union(others ...*Set) (newSet *Set)
  4. func (set *Set) Complement(full *Set) (newSet *Set)
  1. Intersect: 交集,属于set或属于others的元素为元素的集合。
  2. Diff: 差集,属于set且不属于others的元素为元素的集合。
  3. Union: 并集,属于set或属于others的元素为元素的集合。
  4. Complement: 补集,(前提: set应当为full的子集)属于全集full不属于集合set的元素组成的集合。如果给定的full集合不是set的全集时,返回full与set的差集.

通过集合方法我们可以发现,交差并集方法支持多个集合参数进行计算。以下为简化示例,只使用一个参数集合。

  1. package main
  2. import (
  3. "fmt"
  4. "github.com/gogf/gf/g"
  5. "github.com/gogf/gf/g/container/gset"
  6. )
  7. func main() {
  8. s1 := gset.NewFrom(g.Slice{1, 2, 3})
  9. s2 := gset.NewFrom(g.Slice{4, 5, 6})
  10. s3 := gset.NewFrom(g.Slice{1, 2, 3, 4, 5, 6, 7})
  11. // 交集
  12. fmt.Println(s3.Intersect(s1).Slice())
  13. // 差集
  14. fmt.Println(s3.Diff(s1).Slice())
  15. // 并集
  16. fmt.Println(s1.Union(s2).Slice())
  17. // 补集
  18. fmt.Println(s1.Complement(s3).Slice())
  19. }

执行后,输出结果为:

  1. [1 2 3]
  2. [4 5 6 7]
  3. [1 2 3 4 5 6]
  4. [7 4 5 6]

性能测试

https://github.com/gogf/gf/blob/master/g/container/gset/gset_z_bench_test.go

  1. goos: darwin
  2. goarch: amd64
  3. Benchmark_IntSet_Add-4 10000000 277 ns/op 8 B/op 0 allocs/op
  4. Benchmark_IntSet_Contains-4 20000000 60.6 ns/op 0 B/op 0 allocs/op
  5. Benchmark_IntSet_Remove-4 10000000 211 ns/op 0 B/op 0 allocs/op
  6. Benchmark_AnySet_Add-4 5000000 312 ns/op 21 B/op 1 allocs/op
  7. Benchmark_AnySet_Contains-4 30000000 68.2 ns/op 0 B/op 0 allocs/op
  8. Benchmark_AnySet_Remove-4 5000000 267 ns/op 0 B/op 0 allocs/op
  9. Benchmark_StrSet_Add-4 5000000 383 ns/op 20 B/op 1 allocs/op
  10. Benchmark_StrSet_Contains-4 10000000 160 ns/op 7 B/op 0 allocs/op
  11. Benchmark_StrSet_Remove-4 5000000 306 ns/op 7 B/op 0 allocs/op
  12. Benchmark_Unsafe_IntSet_Add-4 10000000 258 ns/op 35 B/op 0 allocs/op
  13. Benchmark_Unsafe_IntSet_Contains-4 20000000 146 ns/op 0 B/op 0 allocs/op
  14. Benchmark_Unsafe_IntSet_Remove-4 10000000 173 ns/op 0 B/op 0 allocs/op
  15. Benchmark_Unsafe_AnySet_Add-4 5000000 355 ns/op 41 B/op 1 allocs/op
  16. Benchmark_Unsafe_AnySet_Contains-4 10000000 150 ns/op 0 B/op 0 allocs/op
  17. Benchmark_Unsafe_AnySet_Remove-4 200000000 11.9 ns/op 0 B/op 0 allocs/op
  18. Benchmark_Unsafe_StrSet_Add-4 5000000 486 ns/op 59 B/op 1 allocs/op
  19. Benchmark_Unsafe_StrSet_Contains-4 5000000 298 ns/op 7 B/op 0 allocs/op
  20. Benchmark_Unsafe_StrSet_Remove-4 10000000 158 ns/op 7 B/op 0 allocs/op