lib/mergeset: skip comparison for every item in the block during merge if the last item in the block is smaller than the first item in the next block

Thanks to @ahfuzhang for the suggestion at https://github.com/VictoriaMetrics/VictoriaMetrics/issues/5651
This commit is contained in:
Aliaksandr Valialkin 2024-01-23 03:15:49 +02:00
parent 7ad16c8f17
commit 389159767d
No known key found for this signature in database
GPG Key ID: 52C003EE2BCDB9EB

View File

@ -125,9 +125,16 @@ again:
} }
items := bsr.Block.items items := bsr.Block.items
data := bsr.Block.data data := bsr.Block.data
for bsr.currItemIdx < len(bsr.Block.items) { compareEveryItem := true
if bsr.currItemIdx < len(items) {
// An optimization, which allows skipping costly comparison for every merged item in the loop below.
// Thanks to @ahfuzhang for the suggestion at https://github.com/VictoriaMetrics/VictoriaMetrics/issues/5651
lastItem := items[len(items)-1].Bytes(data)
compareEveryItem = hasNextItem && string(lastItem) > nextItem
}
for bsr.currItemIdx < len(items) {
item := items[bsr.currItemIdx].Bytes(data) item := items[bsr.currItemIdx].Bytes(data)
if hasNextItem && string(item) > nextItem { if compareEveryItem && string(item) > nextItem {
break break
} }
if !bsm.ib.Add(item) { if !bsm.ib.Add(item) {
@ -137,7 +144,7 @@ again:
} }
bsr.currItemIdx++ bsr.currItemIdx++
} }
if bsr.currItemIdx == len(bsr.Block.items) { if bsr.currItemIdx == len(items) {
// bsr.Block is fully read. Proceed to the next block. // bsr.Block is fully read. Proceed to the next block.
if bsr.Next() { if bsr.Next() {
heap.Fix(&bsm.bsrHeap, 0) heap.Fix(&bsm.bsrHeap, 0)