1
0
mirror of https://github.com/gogf/gf.git synced 2025-04-05 11:18:50 +08:00

container/gmap&gset: fix deadlock when removing values during iterating (#3572)

This commit is contained in:
Lonely 2024-05-21 22:07:28 +08:00 committed by GitHub
parent 15b60462a2
commit 1b23bf495a
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
20 changed files with 173 additions and 30 deletions

View File

@ -45,9 +45,7 @@ func NewAnyAnyMapFrom(data map[interface{}]interface{}, safe ...bool) *AnyAnyMap
// Iterator iterates the hash map readonly with custom callback function `f`.
// If `f` returns true, then it continues iterating; or false to stop.
func (m *AnyAnyMap) Iterator(f func(k interface{}, v interface{}) bool) {
m.mu.RLock()
defer m.mu.RUnlock()
for k, v := range m.data {
for k, v := range m.Map() {
if !f(k, v) {
break
}

View File

@ -46,9 +46,7 @@ func NewIntAnyMapFrom(data map[int]interface{}, safe ...bool) *IntAnyMap {
// Iterator iterates the hash map readonly with custom callback function `f`.
// If `f` returns true, then it continues iterating; or false to stop.
func (m *IntAnyMap) Iterator(f func(k int, v interface{}) bool) {
m.mu.RLock()
defer m.mu.RUnlock()
for k, v := range m.data {
for k, v := range m.Map() {
if !f(k, v) {
break
}

View File

@ -42,9 +42,7 @@ func NewIntIntMapFrom(data map[int]int, safe ...bool) *IntIntMap {
// Iterator iterates the hash map readonly with custom callback function `f`.
// If `f` returns true, then it continues iterating; or false to stop.
func (m *IntIntMap) Iterator(f func(k int, v int) bool) {
m.mu.RLock()
defer m.mu.RUnlock()
for k, v := range m.data {
for k, v := range m.Map() {
if !f(k, v) {
break
}

View File

@ -42,9 +42,7 @@ func NewIntStrMapFrom(data map[int]string, safe ...bool) *IntStrMap {
// Iterator iterates the hash map readonly with custom callback function `f`.
// If `f` returns true, then it continues iterating; or false to stop.
func (m *IntStrMap) Iterator(f func(k int, v string) bool) {
m.mu.RLock()
defer m.mu.RUnlock()
for k, v := range m.data {
for k, v := range m.Map() {
if !f(k, v) {
break
}

View File

@ -47,9 +47,7 @@ func NewStrAnyMapFrom(data map[string]interface{}, safe ...bool) *StrAnyMap {
// Iterator iterates the hash map readonly with custom callback function `f`.
// If `f` returns true, then it continues iterating; or false to stop.
func (m *StrAnyMap) Iterator(f func(k string, v interface{}) bool) {
m.mu.RLock()
defer m.mu.RUnlock()
for k, v := range m.data {
for k, v := range m.Map() {
if !f(k, v) {
break
}

View File

@ -43,9 +43,7 @@ func NewStrIntMapFrom(data map[string]int, safe ...bool) *StrIntMap {
// Iterator iterates the hash map readonly with custom callback function `f`.
// If `f` returns true, then it continues iterating; or false to stop.
func (m *StrIntMap) Iterator(f func(k string, v int) bool) {
m.mu.RLock()
defer m.mu.RUnlock()
for k, v := range m.data {
for k, v := range m.Map() {
if !f(k, v) {
break
}

View File

@ -43,9 +43,7 @@ func NewStrStrMapFrom(data map[string]string, safe ...bool) *StrStrMap {
// Iterator iterates the hash map readonly with custom callback function `f`.
// If `f` returns true, then it continues iterating; or false to stop.
func (m *StrStrMap) Iterator(f func(k string, v string) bool) {
m.mu.RLock()
defer m.mu.RUnlock()
for k, v := range m.data {
for k, v := range m.Map() {
if !f(k, v) {
break
}

View File

@ -110,6 +110,22 @@ func Test_AnyAnyMap_Batch(t *testing.T) {
})
}
func Test_AnyAnyMap_Iterator_Deadlock(t *testing.T) {
gtest.C(t, func(t *gtest.T) {
m := gmap.NewAnyAnyMapFrom(map[interface{}]interface{}{1: 1, 2: "2", "3": "3", "4": 4}, true)
m.Iterator(func(k interface{}, _ interface{}) bool {
if gconv.Int(k)%2 == 0 {
m.Remove(k)
}
return true
})
t.Assert(m.Map(), map[interface{}]interface{}{
1: 1,
"3": "3",
})
})
}
func Test_AnyAnyMap_Iterator(t *testing.T) {
gtest.C(t, func(t *gtest.T) {
expect := map[interface{}]interface{}{1: 1, 2: "2"}

View File

@ -118,6 +118,22 @@ func Test_IntAnyMap_Batch(t *testing.T) {
})
}
func Test_IntAnyMap_Iterator_Deadlock(t *testing.T) {
gtest.C(t, func(t *gtest.T) {
m := gmap.NewIntAnyMapFrom(map[int]interface{}{1: 1, 2: 2, 3: "3", 4: 4}, true)
m.Iterator(func(k int, _ interface{}) bool {
if k%2 == 0 {
m.Remove(k)
}
return true
})
t.Assert(m.Map(), map[int]interface{}{
1: 1,
3: "3",
})
})
}
func Test_IntAnyMap_Iterator(t *testing.T) {
gtest.C(t, func(t *gtest.T) {
expect := map[int]interface{}{1: 1, 2: "2"}

View File

@ -127,6 +127,22 @@ func Test_IntIntMap_Batch(t *testing.T) {
})
}
func Test_IntIntMap_Iterator_Deadlock(t *testing.T) {
gtest.C(t, func(t *gtest.T) {
m := gmap.NewIntIntMapFrom(map[int]int{1: 1, 2: 2, 3: 3, 4: 4}, true)
m.Iterator(func(k int, _ int) bool {
if k%2 == 0 {
m.Remove(k)
}
return true
})
t.Assert(m.Map(), map[int]int{
1: 1,
3: 3,
})
})
}
func Test_IntIntMap_Iterator(t *testing.T) {
gtest.C(t, func(t *gtest.T) {
expect := map[int]int{1: 1, 2: 2}

View File

@ -153,6 +153,22 @@ func Test_IntStrMap_Batch(t *testing.T) {
})
}
func Test_IntStrMap_Iterator_Deadlock(t *testing.T) {
gtest.C(t, func(t *gtest.T) {
m := gmap.NewIntStrMapFrom(map[int]string{1: "1", 2: "2", 3: "3", 4: "4"}, true)
m.Iterator(func(k int, _ string) bool {
if k%2 == 0 {
m.Remove(k)
}
return true
})
t.Assert(m.Map(), map[int]string{
1: "1",
3: "3",
})
})
}
func Test_IntStrMap_Iterator(t *testing.T) {
gtest.C(t, func(t *gtest.T) {
expect := map[int]string{1: "a", 2: "b"}

View File

@ -7,6 +7,7 @@
package gmap_test
import (
"strconv"
"testing"
"github.com/gogf/gf/v2/container/garray"
@ -109,6 +110,23 @@ func Test_StrAnyMap_Batch(t *testing.T) {
})
}
func Test_StrAnyMap_Iterator_Deadlock(t *testing.T) {
gtest.C(t, func(t *gtest.T) {
m := gmap.NewStrAnyMapFrom(map[string]interface{}{"1": "1", "2": "2", "3": "3", "4": "4"}, true)
m.Iterator(func(k string, _ interface{}) bool {
kInt, _ := strconv.Atoi(k)
if kInt%2 == 0 {
m.Remove(k)
}
return true
})
t.Assert(m.Map(), map[string]interface{}{
"1": "1",
"3": "3",
})
})
}
func Test_StrAnyMap_Iterator(t *testing.T) {
gtest.C(t, func(t *gtest.T) {
expect := map[string]interface{}{"a": true, "b": false}

View File

@ -7,6 +7,7 @@
package gmap_test
import (
"strconv"
"testing"
"github.com/gogf/gf/v2/container/garray"
@ -118,6 +119,20 @@ func Test_StrIntMap_Batch(t *testing.T) {
})
}
func Test_StrIntMap_Iterator_Deadlock(t *testing.T) {
gtest.C(t, func(t *gtest.T) {
m := gmap.NewStrIntMapFrom(map[string]int{"1": 1, "2": 2, "3": 3, "4": 4}, true)
m.Iterator(func(k string, _ int) bool {
kInt, _ := strconv.Atoi(k)
if kInt%2 == 0 {
m.Remove(k)
}
return true
})
t.Assert(m.Size(), 2)
})
}
func Test_StrIntMap_Iterator(t *testing.T) {
gtest.C(t, func(t *gtest.T) {
expect := map[string]int{"a": 1, "b": 2}

View File

@ -7,6 +7,7 @@
package gmap_test
import (
"strconv"
"testing"
"github.com/gogf/gf/v2/container/garray"
@ -117,6 +118,20 @@ func Test_StrStrMap_Batch(t *testing.T) {
})
}
func Test_StrStrMap_Iterator_Deadlock(t *testing.T) {
gtest.C(t, func(t *gtest.T) {
m := gmap.NewStrStrMapFrom(map[string]string{"1": "1", "2": "2", "3": "3", "4": "4"}, true)
m.Iterator(func(k string, _ string) bool {
kInt, _ := strconv.Atoi(k)
if kInt%2 == 0 {
m.Remove(k)
}
return true
})
t.Assert(m.Size(), 2)
})
}
func Test_StrStrMap_Iterator(t *testing.T) {
gtest.C(t, func(t *gtest.T) {
expect := map[string]string{"a": "a", "b": "b"}

View File

@ -53,9 +53,7 @@ func NewFrom(items interface{}, safe ...bool) *Set {
// Iterator iterates the set readonly with given callback function `f`,
// if `f` returns true then continue iterating; or false to stop.
func (set *Set) Iterator(f func(v interface{}) bool) {
set.mu.RLock()
defer set.mu.RUnlock()
for k := range set.data {
for _, k := range set.Slice() {
if !f(k) {
break
}

View File

@ -45,9 +45,7 @@ func NewIntSetFrom(items []int, safe ...bool) *IntSet {
// Iterator iterates the set readonly with given callback function `f`,
// if `f` returns true then continue iterating; or false to stop.
func (set *IntSet) Iterator(f func(v int) bool) {
set.mu.RLock()
defer set.mu.RUnlock()
for k := range set.data {
for _, k := range set.Slice() {
if !f(k) {
break
}

View File

@ -47,9 +47,7 @@ func NewStrSetFrom(items []string, safe ...bool) *StrSet {
// Iterator iterates the set readonly with given callback function `f`,
// if `f` returns true then continue iterating; or false to stop.
func (set *StrSet) Iterator(f func(v string) bool) {
set.mu.RLock()
defer set.mu.RUnlock()
for k := range set.data {
for _, k := range set.Slice() {
if !f(k) {
break
}

View File

@ -82,6 +82,23 @@ func TestSet_Basic(t *testing.T) {
})
}
func TestSet_Iterator_Deadlock(t *testing.T) {
gtest.C(t, func(t *gtest.T) {
set := gset.NewFrom([]interface{}{1, 2, 3, 4, 5}, true)
set.Iterator(func(k interface{}) bool {
if gconv.Int(k)%2 == 0 {
set.Remove(k)
}
return true
})
t.Assert(set.Contains(1), true)
t.Assert(set.Contains(2), false)
t.Assert(set.Contains(3), true)
t.Assert(set.Contains(4), false)
t.Assert(set.Contains(5), true)
})
}
func TestSet_Iterator(t *testing.T) {
gtest.C(t, func(t *gtest.T) {
s := gset.NewSet()

View File

@ -62,6 +62,23 @@ func TestIntSet_Basic(t *testing.T) {
})
}
func TestIntSet_Iterator_Deadlock(t *testing.T) {
gtest.C(t, func(t *gtest.T) {
set := gset.NewIntSetFrom([]int{1, 2, 3, 4, 5}, true)
set.Iterator(func(k int) bool {
if k%2 == 0 {
set.Remove(k)
}
return true
})
t.Assert(set.Contains(1), true)
t.Assert(set.Contains(2), false)
t.Assert(set.Contains(3), true)
t.Assert(set.Contains(4), false)
t.Assert(set.Contains(5), true)
})
}
func TestIntSet_Iterator(t *testing.T) {
gtest.C(t, func(t *gtest.T) {
s := gset.NewIntSet()

View File

@ -73,6 +73,23 @@ func TestStrSet_ContainsI(t *testing.T) {
})
}
func TestStrSet_Iterator_Deadlock(t *testing.T) {
gtest.C(t, func(t *gtest.T) {
set := gset.NewStrSetFrom([]string{"1", "2", "3", "4", "5"}, true)
set.Iterator(func(k string) bool {
if gconv.Int(k)%2 == 0 {
set.Remove(k)
}
return true
})
t.Assert(set.Contains("1"), true)
t.Assert(set.Contains("2"), false)
t.Assert(set.Contains("3"), true)
t.Assert(set.Contains("4"), false)
t.Assert(set.Contains("5"), true)
})
}
func TestStrSet_Iterator(t *testing.T) {
gtest.C(t, func(t *gtest.T) {
s := gset.NewStrSet()