mirror of
				https://github.com/TeaOSLab/EdgeNode.git
				synced 2025-11-04 16:00:25 +08:00 
			
		
		
		
	优化ttlcache回收速度
This commit is contained in:
		@@ -3,6 +3,7 @@ package ttlcache
 | 
				
			|||||||
import (
 | 
					import (
 | 
				
			||||||
	"github.com/TeaOSLab/EdgeNode/internal/utils"
 | 
						"github.com/TeaOSLab/EdgeNode/internal/utils"
 | 
				
			||||||
	"github.com/TeaOSLab/EdgeNode/internal/utils/fasttime"
 | 
						"github.com/TeaOSLab/EdgeNode/internal/utils/fasttime"
 | 
				
			||||||
 | 
						"runtime"
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
var SharedInt64Cache = NewBigCache[int64]()
 | 
					var SharedInt64Cache = NewBigCache[int64]()
 | 
				
			||||||
@@ -19,7 +20,8 @@ type Cache[T any] struct {
 | 
				
			|||||||
	countPieces uint64
 | 
						countPieces uint64
 | 
				
			||||||
	maxItems    int
 | 
						maxItems    int
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	gcPieceIndex int
 | 
						maxPiecesPerGC int
 | 
				
			||||||
 | 
						gcPieceIndex   int
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func NewBigCache[T any]() *Cache[T] {
 | 
					func NewBigCache[T any]() *Cache[T] {
 | 
				
			||||||
@@ -61,9 +63,16 @@ func NewCache[T any](opt ...OptionInterface) *Cache[T] {
 | 
				
			|||||||
		}
 | 
							}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						var maxPiecesPerGC = 4
 | 
				
			||||||
 | 
						var numCPU = runtime.NumCPU() / 2
 | 
				
			||||||
 | 
						if numCPU > maxPiecesPerGC {
 | 
				
			||||||
 | 
							maxPiecesPerGC = numCPU
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var cache = &Cache[T]{
 | 
						var cache = &Cache[T]{
 | 
				
			||||||
		countPieces: uint64(countPieces),
 | 
							countPieces:    uint64(countPieces),
 | 
				
			||||||
		maxItems:    maxItems,
 | 
							maxItems:       maxItems,
 | 
				
			||||||
 | 
							maxPiecesPerGC: maxPiecesPerGC,
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	for i := 0; i < countPieces; i++ {
 | 
						for i := 0; i < countPieces; i++ {
 | 
				
			||||||
@@ -136,15 +145,15 @@ func (this *Cache[T]) Count() (count int) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
func (this *Cache[T]) GC() {
 | 
					func (this *Cache[T]) GC() {
 | 
				
			||||||
	var index = this.gcPieceIndex
 | 
						var index = this.gcPieceIndex
 | 
				
			||||||
	const maxPiecesPerGC = 4
 | 
					
 | 
				
			||||||
	for i := index; i < index+maxPiecesPerGC; i++ {
 | 
						for i := index; i < index+this.maxPiecesPerGC; i++ {
 | 
				
			||||||
		if i >= int(this.countPieces) {
 | 
							if i >= int(this.countPieces) {
 | 
				
			||||||
			break
 | 
								break
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
		this.pieces[i].GC()
 | 
							this.pieces[i].GC()
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	index += maxPiecesPerGC
 | 
						index += this.maxPiecesPerGC
 | 
				
			||||||
	if index >= int(this.countPieces) {
 | 
						if index >= int(this.countPieces) {
 | 
				
			||||||
		index = 0
 | 
							index = 0
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -6,6 +6,7 @@ import (
 | 
				
			|||||||
	"github.com/iwind/TeaGo/assert"
 | 
						"github.com/iwind/TeaGo/assert"
 | 
				
			||||||
	"github.com/iwind/TeaGo/rands"
 | 
						"github.com/iwind/TeaGo/rands"
 | 
				
			||||||
	"github.com/iwind/TeaGo/types"
 | 
						"github.com/iwind/TeaGo/types"
 | 
				
			||||||
 | 
						timeutil "github.com/iwind/TeaGo/utils/time"
 | 
				
			||||||
	"runtime"
 | 
						"runtime"
 | 
				
			||||||
	"strconv"
 | 
						"strconv"
 | 
				
			||||||
	"sync/atomic"
 | 
						"sync/atomic"
 | 
				
			||||||
@@ -46,12 +47,15 @@ func TestCache_Memory(t *testing.T) {
 | 
				
			|||||||
		return
 | 
							return
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	testutils.StartMemoryStats(t)
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	var cache = NewCache[int]()
 | 
						var cache = NewCache[int]()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						testutils.StartMemoryStats(t, func() {
 | 
				
			||||||
 | 
							t.Log(cache.Count(), "items")
 | 
				
			||||||
 | 
						})
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var count = 20_000_000
 | 
						var count = 20_000_000
 | 
				
			||||||
	for i := 0; i < count; i++ {
 | 
						for i := 0; i < count; i++ {
 | 
				
			||||||
		cache.Write("a"+strconv.Itoa(i), 1, time.Now().Unix()+3600)
 | 
							cache.Write("a"+strconv.Itoa(i), 1, time.Now().Unix()+int64(rands.Int(0, 300)))
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	t.Log(cache.Count())
 | 
						t.Log(cache.Count())
 | 
				
			||||||
@@ -67,7 +71,7 @@ func TestCache_Memory(t *testing.T) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
	cache.Count()
 | 
						cache.Count()
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	time.Sleep(10 * time.Second)
 | 
						time.Sleep(3600 * time.Second)
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestCache_IncreaseInt64(t *testing.T) {
 | 
					func TestCache_IncreaseInt64(t *testing.T) {
 | 
				
			||||||
@@ -125,6 +129,10 @@ func TestCache_Read(t *testing.T) {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestCache_GC(t *testing.T) {
 | 
					func TestCache_GC(t *testing.T) {
 | 
				
			||||||
 | 
						if !testutils.IsSingleTesting() {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var cache = NewCache[int](&PiecesOption{Count: 5})
 | 
						var cache = NewCache[int](&PiecesOption{Count: 5})
 | 
				
			||||||
	cache.Write("a", 1, time.Now().Unix()+1)
 | 
						cache.Write("a", 1, time.Now().Unix()+1)
 | 
				
			||||||
	cache.Write("b", 2, time.Now().Unix()+2)
 | 
						cache.Write("b", 2, time.Now().Unix()+2)
 | 
				
			||||||
@@ -159,11 +167,15 @@ func TestCache_GC(t *testing.T) {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestCache_GC2(t *testing.T) {
 | 
					func TestCache_GC2(t *testing.T) {
 | 
				
			||||||
 | 
						if !testutils.IsSingleTesting() {
 | 
				
			||||||
 | 
							return
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	runtime.GOMAXPROCS(1)
 | 
						runtime.GOMAXPROCS(1)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var cache1 = NewCache[int](NewPiecesOption(32))
 | 
						var cache1 = NewCache[int](NewPiecesOption(256))
 | 
				
			||||||
	for i := 0; i < 1_000_000; i++ {
 | 
						for i := 0; i < 10_000_000; i++ {
 | 
				
			||||||
		cache1.Write(strconv.Itoa(i), i, time.Now().Unix()+int64(rands.Int(0, 10)))
 | 
							cache1.Write(strconv.Itoa(i), i, time.Now().Unix()+10)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var cache2 = NewCache[int](NewPiecesOption(5))
 | 
						var cache2 = NewCache[int](NewPiecesOption(5))
 | 
				
			||||||
@@ -171,8 +183,8 @@ func TestCache_GC2(t *testing.T) {
 | 
				
			|||||||
		cache2.Write(strconv.Itoa(i), i, time.Now().Unix()+int64(rands.Int(0, 10)))
 | 
							cache2.Write(strconv.Itoa(i), i, time.Now().Unix()+int64(rands.Int(0, 10)))
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	for i := 0; i < 100; i++ {
 | 
						for i := 0; i < 3600; i++ {
 | 
				
			||||||
		t.Log(cache1.Count(), "items", cache2.Count(), "items")
 | 
							t.Log(timeutil.Format("H:i:s"), cache1.Count(), "items", cache2.Count(), "items")
 | 
				
			||||||
		time.Sleep(1 * time.Second)
 | 
							time.Sleep(1 * time.Second)
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 
 | 
				
			|||||||
@@ -105,14 +105,14 @@ func (this *Piece[T]) GC() {
 | 
				
			|||||||
		this.lastGCTime = currentTime - 3600
 | 
							this.lastGCTime = currentTime - 3600
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	var min = this.lastGCTime
 | 
						var minTime = this.lastGCTime
 | 
				
			||||||
	var max = currentTime
 | 
						var maxTime = currentTime
 | 
				
			||||||
	if min > max {
 | 
						if minTime > maxTime {
 | 
				
			||||||
		// 过去的时间比现在大,则从这一秒重新开始
 | 
							// 过去的时间比现在大,则从这一秒重新开始
 | 
				
			||||||
		min = max
 | 
							minTime = maxTime
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	for i := min; i <= max; i++ {
 | 
						for i := minTime; i <= maxTime; i++ {
 | 
				
			||||||
		var itemMap = this.expiresList.GC(i)
 | 
							var itemMap = this.expiresList.GC(i)
 | 
				
			||||||
		if len(itemMap) > 0 {
 | 
							if len(itemMap) > 0 {
 | 
				
			||||||
			this.gcItemMap(itemMap)
 | 
								this.gcItemMap(itemMap)
 | 
				
			||||||
 
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user