mirror of
https://github.com/seaweedfs/seaweedfs.git
synced 2024-01-19 02:48:24 +00:00
return previous element if visited
This commit is contained in:
parent
22d8684e88
commit
a481c4a45e
|
@ -67,17 +67,17 @@ func (t *SkipList) generateLevel(maxLevel int) int {
|
||||||
return level
|
return level
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *SkipList) findEntryIndex(key []byte, level int) int {
|
func (t *SkipList) findEntryIndex(key []byte, minLevel int) int {
|
||||||
// Find good entry point so we don't accidentally skip half the list.
|
// Find good entry point so we don't accidentally skip half the list.
|
||||||
for i := t.maxLevel; i >= 0; i-- {
|
for i := t.maxLevel; i >= 0; i-- {
|
||||||
if t.startLevels[i] != nil && bytes.Compare(t.startLevels[i].Key, key) < 0 || i <= level {
|
if t.startLevels[i] != nil && bytes.Compare(t.startLevels[i].Key, key) < 0 || i <= minLevel {
|
||||||
return i
|
return i
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return 0
|
return 0
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t *SkipList) findExtended(key []byte, findGreaterOrEqual bool) (foundElem *SkipListElement, ok bool, err error) {
|
func (t *SkipList) findExtended(key []byte, findGreaterOrEqual bool) (prevElementIfVisited *SkipListElement, foundElem *SkipListElement, ok bool, err error) {
|
||||||
|
|
||||||
foundElem = nil
|
foundElem = nil
|
||||||
ok = false
|
ok = false
|
||||||
|
@ -120,6 +120,7 @@ func (t *SkipList) findExtended(key []byte, findGreaterOrEqual bool) (foundElem
|
||||||
|
|
||||||
// Early exit
|
// Early exit
|
||||||
if currentNode.Next[0] != nil && bytes.Compare(currentNode.Next[0].Key, key) == 0 {
|
if currentNode.Next[0] != nil && bytes.Compare(currentNode.Next[0].Key, key) == 0 {
|
||||||
|
prevElementIfVisited = currentNode
|
||||||
var currentNodeNext *SkipListElement
|
var currentNodeNext *SkipListElement
|
||||||
currentNodeNext, err = t.loadElement(currentNode.Next[0])
|
currentNodeNext, err = t.loadElement(currentNode.Next[0])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -150,26 +151,26 @@ func (t *SkipList) findExtended(key []byte, findGreaterOrEqual bool) (foundElem
|
||||||
// Find tries to find an element in the skiplist based on the key from the given ListElement.
|
// Find tries to find an element in the skiplist based on the key from the given ListElement.
|
||||||
// elem can be used, if ok is true.
|
// elem can be used, if ok is true.
|
||||||
// Find runs in approx. O(log(n))
|
// Find runs in approx. O(log(n))
|
||||||
func (t *SkipList) Find(key []byte) (elem *SkipListElement, ok bool, err error) {
|
func (t *SkipList) Find(key []byte) (prevIfVisited *SkipListElement, elem *SkipListElement, ok bool, err error) {
|
||||||
|
|
||||||
if t == nil || key == nil {
|
if t == nil || key == nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
elem, ok, err = t.findExtended(key, false)
|
prevIfVisited, elem, ok, err = t.findExtended(key, false)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// FindGreaterOrEqual finds the first element, that is greater or equal to the given ListElement e.
|
// FindGreaterOrEqual finds the first element, that is greater or equal to the given ListElement e.
|
||||||
// The comparison is done on the keys (So on ExtractKey()).
|
// The comparison is done on the keys (So on ExtractKey()).
|
||||||
// FindGreaterOrEqual runs in approx. O(log(n))
|
// FindGreaterOrEqual runs in approx. O(log(n))
|
||||||
func (t *SkipList) FindGreaterOrEqual(key []byte) (elem *SkipListElement, ok bool, err error) {
|
func (t *SkipList) FindGreaterOrEqual(key []byte) (prevIfVisited *SkipListElement, elem *SkipListElement, ok bool, err error) {
|
||||||
|
|
||||||
if t == nil || key == nil {
|
if t == nil || key == nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
elem, ok, err = t.findExtended(key, true)
|
prevIfVisited, elem, ok, err = t.findExtended(key, true)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -23,12 +23,12 @@ func TestInsertAndFind(t *testing.T) {
|
||||||
|
|
||||||
var listPointer *SkipList
|
var listPointer *SkipList
|
||||||
listPointer.Insert(k0, k0)
|
listPointer.Insert(k0, k0)
|
||||||
if _, ok, _ := listPointer.Find(k0); ok {
|
if _, _, ok, _ := listPointer.Find(k0); ok {
|
||||||
t.Fail()
|
t.Fail()
|
||||||
}
|
}
|
||||||
|
|
||||||
list = New(memStore)
|
list = New(memStore)
|
||||||
if _, ok, _ := list.Find(k0); ok {
|
if _, _, ok, _ := list.Find(k0); ok {
|
||||||
t.Fail()
|
t.Fail()
|
||||||
}
|
}
|
||||||
if !list.IsEmpty() {
|
if !list.IsEmpty() {
|
||||||
|
@ -42,7 +42,7 @@ func TestInsertAndFind(t *testing.T) {
|
||||||
}
|
}
|
||||||
for i := 0; i < maxN; i++ {
|
for i := 0; i < maxN; i++ {
|
||||||
key := []byte(strconv.Itoa(maxN - i))
|
key := []byte(strconv.Itoa(maxN - i))
|
||||||
if _, ok, _ := list.Find(key); !ok {
|
if _, _, ok, _ := list.Find(key); !ok {
|
||||||
t.Fail()
|
t.Fail()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -55,7 +55,7 @@ func TestInsertAndFind(t *testing.T) {
|
||||||
}
|
}
|
||||||
for i := 0; i < maxN; i++ {
|
for i := 0; i < maxN; i++ {
|
||||||
key := []byte(strconv.Itoa(i))
|
key := []byte(strconv.Itoa(i))
|
||||||
if _, ok, _ := list.Find(key); !ok {
|
if _, _, ok, _ := list.Find(key); !ok {
|
||||||
t.Fail()
|
t.Fail()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -71,7 +71,7 @@ func TestInsertAndFind(t *testing.T) {
|
||||||
for _, e := range rList {
|
for _, e := range rList {
|
||||||
key := []byte(strconv.Itoa(e))
|
key := []byte(strconv.Itoa(e))
|
||||||
// println("find", e)
|
// println("find", e)
|
||||||
if _, ok, _ := list.Find(key); !ok {
|
if _, _, ok, _ := list.Find(key); !ok {
|
||||||
t.Fail()
|
t.Fail()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -215,7 +215,7 @@ func TestFindGreaterOrEqual(t *testing.T) {
|
||||||
var listPointer *SkipList
|
var listPointer *SkipList
|
||||||
|
|
||||||
// Test on empty list.
|
// Test on empty list.
|
||||||
if _, ok, _ := listPointer.FindGreaterOrEqual(Element(0)); ok {
|
if _, _, ok, _ := listPointer.FindGreaterOrEqual(Element(0)); ok {
|
||||||
t.Fail()
|
t.Fail()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -227,7 +227,7 @@ func TestFindGreaterOrEqual(t *testing.T) {
|
||||||
|
|
||||||
for i := 0; i < maxN; i++ {
|
for i := 0; i < maxN; i++ {
|
||||||
key := Element(rand.Intn(maxNumber))
|
key := Element(rand.Intn(maxNumber))
|
||||||
if v, ok, _ := list.FindGreaterOrEqual(key); ok {
|
if _, v, ok, _ := list.FindGreaterOrEqual(key); ok {
|
||||||
// if f is v should be bigger than the element before
|
// if f is v should be bigger than the element before
|
||||||
if v.Prev != nil && bytes.Compare(v.Prev.Key, key) >= 0 {
|
if v.Prev != nil && bytes.Compare(v.Prev.Key, key) >= 0 {
|
||||||
fmt.Printf("PrevV: %s\n key: %s\n\n", string(v.Prev.Key), string(key))
|
fmt.Printf("PrevV: %s\n key: %s\n\n", string(v.Prev.Key), string(key))
|
||||||
|
@ -261,7 +261,7 @@ func TestChangeValue(t *testing.T) {
|
||||||
|
|
||||||
for i := 0; i < maxN; i++ {
|
for i := 0; i < maxN; i++ {
|
||||||
// The key only looks at the int so the string doesn't matter here!
|
// The key only looks at the int so the string doesn't matter here!
|
||||||
f1, ok, _ := list.Find(Element(i))
|
_, f1, ok, _ := list.Find(Element(i))
|
||||||
if !ok {
|
if !ok {
|
||||||
t.Fail()
|
t.Fail()
|
||||||
}
|
}
|
||||||
|
@ -269,7 +269,7 @@ func TestChangeValue(t *testing.T) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fail()
|
t.Fail()
|
||||||
}
|
}
|
||||||
f2, ok, _ := list.Find(Element(i))
|
_, f2, ok, _ := list.Find(Element(i))
|
||||||
if !ok {
|
if !ok {
|
||||||
t.Fail()
|
t.Fail()
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue