mirror of
				https://github.com/TeaOSLab/EdgeNode.git
				synced 2025-11-04 07:40:56 +08:00 
			
		
		
		
	优化计数器性能
This commit is contained in:
		@@ -22,7 +22,7 @@ type SupportedUIntType interface {
 | 
				
			|||||||
type Counter[T SupportedUIntType] struct {
 | 
					type Counter[T SupportedUIntType] struct {
 | 
				
			||||||
	countMaps uint64
 | 
						countMaps uint64
 | 
				
			||||||
	locker    *syncutils.RWMutex
 | 
						locker    *syncutils.RWMutex
 | 
				
			||||||
	itemMaps  []map[uint64]*Item[T]
 | 
						itemMaps  []map[uint64]Item[T]
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	gcTicker *time.Ticker
 | 
						gcTicker *time.Ticker
 | 
				
			||||||
	gcIndex  int
 | 
						gcIndex  int
 | 
				
			||||||
@@ -36,9 +36,9 @@ func NewCounter[T SupportedUIntType]() *Counter[T] {
 | 
				
			|||||||
		count = 8
 | 
							count = 8
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var itemMaps = []map[uint64]*Item[T]{}
 | 
						var itemMaps = []map[uint64]Item[T]{}
 | 
				
			||||||
	for i := 0; i < count; i++ {
 | 
						for i := 0; i < count; i++ {
 | 
				
			||||||
		itemMaps = append(itemMaps, map[uint64]*Item[T]{})
 | 
							itemMaps = append(itemMaps, map[uint64]Item[T]{})
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var counter = &Counter[T]{
 | 
						var counter = &Counter[T]{
 | 
				
			||||||
@@ -69,19 +69,22 @@ func (this *Counter[T]) WithGC() *Counter[T] {
 | 
				
			|||||||
func (this *Counter[T]) Increase(key uint64, lifeSeconds int) T {
 | 
					func (this *Counter[T]) Increase(key uint64, lifeSeconds int) T {
 | 
				
			||||||
	var index = int(key % this.countMaps)
 | 
						var index = int(key % this.countMaps)
 | 
				
			||||||
	this.locker.RLock(index)
 | 
						this.locker.RLock(index)
 | 
				
			||||||
	var item = this.itemMaps[index][key]
 | 
						var item = this.itemMaps[index][key] // item MUST NOT be pointer
 | 
				
			||||||
	this.locker.RUnlock(index)
 | 
						this.locker.RUnlock(index)
 | 
				
			||||||
	if item == nil {
 | 
						if !item.IsOk() {
 | 
				
			||||||
		// no need to care about duplication
 | 
							// no need to care about duplication
 | 
				
			||||||
		// always insert new item even when itemMap is full
 | 
							// always insert new item even when itemMap is full
 | 
				
			||||||
		item = NewItem[T](lifeSeconds)
 | 
							item = NewItem[T](lifeSeconds)
 | 
				
			||||||
 | 
							var result = item.Increase()
 | 
				
			||||||
		this.locker.Lock(index)
 | 
							this.locker.Lock(index)
 | 
				
			||||||
		this.itemMaps[index][key] = item
 | 
							this.itemMaps[index][key] = item
 | 
				
			||||||
		this.locker.Unlock(index)
 | 
							this.locker.Unlock(index)
 | 
				
			||||||
 | 
							return result
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	this.locker.Lock(index)
 | 
						this.locker.Lock(index)
 | 
				
			||||||
	var result = item.Increase()
 | 
						var result = item.Increase()
 | 
				
			||||||
 | 
						this.itemMaps[index][key] = item // overwrite
 | 
				
			||||||
	this.locker.Unlock(index)
 | 
						this.locker.Unlock(index)
 | 
				
			||||||
	return result
 | 
						return result
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
@@ -97,7 +100,7 @@ func (this *Counter[T]) Get(key uint64) T {
 | 
				
			|||||||
	this.locker.RLock(index)
 | 
						this.locker.RLock(index)
 | 
				
			||||||
	defer this.locker.RUnlock(index)
 | 
						defer this.locker.RUnlock(index)
 | 
				
			||||||
	var item = this.itemMaps[index][key]
 | 
						var item = this.itemMaps[index][key]
 | 
				
			||||||
	if item != nil {
 | 
						if item.IsOk() {
 | 
				
			||||||
		return item.Sum()
 | 
							return item.Sum()
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
	return 0
 | 
						return 0
 | 
				
			||||||
@@ -115,7 +118,7 @@ func (this *Counter[T]) Reset(key uint64) {
 | 
				
			|||||||
	var item = this.itemMaps[index][key]
 | 
						var item = this.itemMaps[index][key]
 | 
				
			||||||
	this.locker.RUnlock(index)
 | 
						this.locker.RUnlock(index)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if item != nil {
 | 
						if item.IsOk() {
 | 
				
			||||||
		this.locker.Lock(index)
 | 
							this.locker.Lock(index)
 | 
				
			||||||
		delete(this.itemMaps[index], key)
 | 
							delete(this.itemMaps[index], key)
 | 
				
			||||||
		this.locker.Unlock(index)
 | 
							this.locker.Unlock(index)
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -101,7 +101,7 @@ func TestCounterMemory(t *testing.T) {
 | 
				
			|||||||
		var costSeconds = time.Since(before).Seconds()
 | 
							var costSeconds = time.Since(before).Seconds()
 | 
				
			||||||
		var stats = &debug.GCStats{}
 | 
							var stats = &debug.GCStats{}
 | 
				
			||||||
		debug.ReadGCStats(stats)
 | 
							debug.ReadGCStats(stats)
 | 
				
			||||||
		t.Log("GC pause:", stats.PauseTotal.Seconds()*1000, "ms", "cost:", costSeconds*1000, "ms")
 | 
							t.Log("GC pause:", stats.Pause[0].Seconds()*1000, "ms", "cost:", costSeconds*1000, "ms")
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	gcPause()
 | 
						gcPause()
 | 
				
			||||||
@@ -113,12 +113,14 @@ func BenchmarkCounter_Increase(b *testing.B) {
 | 
				
			|||||||
	runtime.GOMAXPROCS(4)
 | 
						runtime.GOMAXPROCS(4)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var counter = counters.NewCounter[uint32]()
 | 
						var counter = counters.NewCounter[uint32]()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						b.ReportAllocs()
 | 
				
			||||||
	b.ResetTimer()
 | 
						b.ResetTimer()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var i uint64
 | 
						var i uint64
 | 
				
			||||||
	b.RunParallel(func(pb *testing.PB) {
 | 
						b.RunParallel(func(pb *testing.PB) {
 | 
				
			||||||
		for pb.Next() {
 | 
							for pb.Next() {
 | 
				
			||||||
			counter.Increase(atomic.AddUint64(&i, 1)%1e6, 20)
 | 
								counter.Increase(atomic.AddUint64(&i, 1)%1_000_000, 20)
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
	})
 | 
						})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@@ -138,11 +140,12 @@ func BenchmarkCounter_IncreaseKey(b *testing.B) {
 | 
				
			|||||||
	}()
 | 
						}()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	b.ResetTimer()
 | 
						b.ResetTimer()
 | 
				
			||||||
 | 
						b.ReportAllocs()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var i uint64
 | 
						var i uint64
 | 
				
			||||||
	b.RunParallel(func(pb *testing.PB) {
 | 
						b.RunParallel(func(pb *testing.PB) {
 | 
				
			||||||
		for pb.Next() {
 | 
							for pb.Next() {
 | 
				
			||||||
			counter.IncreaseKey(types.String(atomic.AddUint64(&i, 1)%1e6), 20)
 | 
								counter.IncreaseKey(types.String(atomic.AddUint64(&i, 1)%1_000_000), 20)
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
	})
 | 
						})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -16,7 +16,7 @@ type Item[T SupportedUIntType] struct {
 | 
				
			|||||||
	spanSeconds    int64
 | 
						spanSeconds    int64
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func NewItem[T SupportedUIntType](lifeSeconds int) *Item[T] {
 | 
					func NewItem[T SupportedUIntType](lifeSeconds int) Item[T] {
 | 
				
			||||||
	if lifeSeconds <= 0 {
 | 
						if lifeSeconds <= 0 {
 | 
				
			||||||
		lifeSeconds = 60
 | 
							lifeSeconds = 60
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
@@ -27,7 +27,7 @@ func NewItem[T SupportedUIntType](lifeSeconds int) *Item[T] {
 | 
				
			|||||||
		spanSeconds++
 | 
							spanSeconds++
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return &Item[T]{
 | 
						return Item[T]{
 | 
				
			||||||
		lifeSeconds:    int64(lifeSeconds),
 | 
							lifeSeconds:    int64(lifeSeconds),
 | 
				
			||||||
		spanSeconds:    int64(spanSeconds),
 | 
							spanSeconds:    int64(spanSeconds),
 | 
				
			||||||
		lastUpdateTime: fasttime.Now().Unix(),
 | 
							lastUpdateTime: fasttime.Now().Unix(),
 | 
				
			||||||
@@ -126,3 +126,7 @@ func (this *Item[T]) calculateSpanIndex(timestamp int64) int {
 | 
				
			|||||||
	}
 | 
						}
 | 
				
			||||||
	return index
 | 
						return index
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (this *Item[T]) IsOk() bool {
 | 
				
			||||||
 | 
						return this.lifeSeconds > 0
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user