Commit 2d6fbbb4 authored by Jan Mercl's avatar Jan Mercl

Make index page embed separator keys directly, not data page refs.

parent 5a1179fd
...@@ -5,6 +5,7 @@ ...@@ -5,6 +5,7 @@
package b package b
import ( import (
"bytes"
"fmt" "fmt"
"io" "io"
"math" "math"
...@@ -14,11 +15,16 @@ import ( ...@@ -14,11 +15,16 @@ import (
"strings" "strings"
"testing" "testing"
"github.com/cznic/fileutil"
"github.com/cznic/mathutil" "github.com/cznic/mathutil"
"github.com/cznic/strutil"
) )
func use(...interface{}) {} var caller = func(s string, va ...interface{}) {
_, fn, fl, _ := runtime.Caller(2)
fmt.Printf("%s:%d: ", path.Base(fn), fl)
fmt.Printf(s, va...)
fmt.Println()
}
func dbg(s string, va ...interface{}) { func dbg(s string, va ...interface{}) {
if s == "" { if s == "" {
...@@ -30,13 +36,15 @@ func dbg(s string, va ...interface{}) { ...@@ -30,13 +36,15 @@ func dbg(s string, va ...interface{}) {
fmt.Println() fmt.Println()
} }
var caller = func(s string, va ...interface{}) { func TODO(...interface{}) string {
_, fn, fl, _ := runtime.Caller(2) _, fn, fl, _ := runtime.Caller(1)
fmt.Printf("%s:%d: ", path.Base(fn), fl) return fmt.Sprintf("TODO: %s:%d:\n", path.Base(fn), fl)
fmt.Printf(s, va...)
fmt.Println()
} }
func use(...interface{}) {}
// ============================================================================
func isNil(p interface{}) bool { func isNil(p interface{}) bool {
switch x := p.(type) { switch x := p.(type) {
case *x: case *x:
...@@ -52,6 +60,9 @@ func isNil(p interface{}) bool { ...@@ -52,6 +60,9 @@ func isNil(p interface{}) bool {
} }
func (t *Tree) dump() string { func (t *Tree) dump() string {
var buf bytes.Buffer
f := strutil.IndentFormatter(&buf, "\t")
num := map[interface{}]int{} num := map[interface{}]int{}
visited := map[interface{}]bool{} visited := map[interface{}]bool{}
...@@ -81,20 +92,20 @@ func (t *Tree) dump() string { ...@@ -81,20 +92,20 @@ func (t *Tree) dump() string {
h := handle(p) h := handle(p)
n := 0 n := 0
for i, v := range x.x { for i, v := range x.x {
if v.ch != nil || v.sep != nil { if v.ch != nil || v.k != nil {
n = i + 1 n = i + 1
} }
} }
fmt.Printf("%sX#%d n %d:%d {", pref, h, x.c, n) f.Format("%sX#%d n %d:%d {", pref, h, x.c, n)
a := []interface{}{} a := []interface{}{}
for i, v := range x.x[:n] { for i, v := range x.x[:n] {
a = append(a, v.ch, v.sep) a = append(a, v.ch)
if i != 0 { if i != 0 {
fmt.Printf(" ") f.Format(" ")
} }
fmt.Printf("(C#%d D#%d)", handle(v.ch), handle(v.sep)) f.Format("(C#%d K %v)", handle(v.ch), v.k)
} }
fmt.Printf("}\n") f.Format("}\n")
for _, p := range a { for _, p := range a {
pagedump(p, pref+". ") pagedump(p, pref+". ")
} }
...@@ -106,19 +117,23 @@ func (t *Tree) dump() string { ...@@ -106,19 +117,23 @@ func (t *Tree) dump() string {
n = i + 1 n = i + 1
} }
} }
fmt.Printf("%sD#%d P#%d N#%d n %d:%d {", pref, h, handle(x.p), handle(x.n), x.c, n) f.Format("%sD#%d P#%d N#%d n %d:%d {", pref, h, handle(x.p), handle(x.n), x.c, n)
for i, v := range x.d[:n] { for i, v := range x.d[:n] {
if i != 0 { if i != 0 {
fmt.Printf(" ") f.Format(" ")
} }
fmt.Printf("%v:%v", v.k, v.v) f.Format("%v:%v", v.k, v.v)
} }
fmt.Printf("}\n") f.Format("}\n")
} }
} }
pagedump(t.r, "") pagedump(t.r, "")
return "" s := buf.String()
if s != "" {
s = s[:len(s)-1]
}
return s
} }
func rng() *mathutil.FC32 { func rng() *mathutil.FC32 {
...@@ -203,7 +218,7 @@ func TestSetGet0(t *testing.T) { ...@@ -203,7 +218,7 @@ func TestSetGet0(t *testing.T) {
} }
func TestSetGet1(t *testing.T) { func TestSetGet1(t *testing.T) {
const N = 90000 const N = 50000
for _, x := range []int{0, -1, 0x555555, 0xaaaaaa, 0x314259} { for _, x := range []int{0, -1, 0x555555, 0xaaaaaa, 0x314259} {
r := TreeNew(cmp) r := TreeNew(cmp)
set := r.Set set := r.Set
...@@ -387,7 +402,7 @@ func benchmarkGetRnd(b *testing.B, n int) { ...@@ -387,7 +402,7 @@ func benchmarkGetRnd(b *testing.B, n int) {
} }
func TestSetGet2(t *testing.T) { func TestSetGet2(t *testing.T) {
const N = 70000 const N = 50000
for _, x := range []int{0, -1, 0x555555, 0xaaaaaa, 0x314259} { for _, x := range []int{0, -1, 0x555555, 0xaaaaaa, 0x314259} {
r := TreeNew(cmp) r := TreeNew(cmp)
set := r.Set set := r.Set
...@@ -533,7 +548,7 @@ func TestDelete0(t *testing.T) { ...@@ -533,7 +548,7 @@ func TestDelete0(t *testing.T) {
} }
func TestDelete1(t *testing.T) { func TestDelete1(t *testing.T) {
const N = 100000 const N = 60000
for _, x := range []int{0, -1, 0x555555, 0xaaaaaa, 0x314259} { for _, x := range []int{0, -1, 0x555555, 0xaaaaaa, 0x314259} {
r := TreeNew(cmp) r := TreeNew(cmp)
set := r.Set set := r.Set
...@@ -638,7 +653,7 @@ func benchmarkDelRnd(b *testing.B, n int) { ...@@ -638,7 +653,7 @@ func benchmarkDelRnd(b *testing.B, n int) {
} }
func TestDelete2(t *testing.T) { func TestDelete2(t *testing.T) {
const N = 80000 const N = 50000
for _, x := range []int{0, -1, 0x555555, 0xaaaaaa, 0x314259} { for _, x := range []int{0, -1, 0x555555, 0xaaaaaa, 0x314259} {
r := TreeNew(cmp) r := TreeNew(cmp)
set := r.Set set := r.Set
...@@ -703,7 +718,7 @@ func TestEnumeratorNext(t *testing.T) { ...@@ -703,7 +718,7 @@ func TestEnumeratorNext(t *testing.T) {
k, v, err := en.Next() k, v, err := en.Next()
if err != nil { if err != nil {
if !fileutil.IsEOF(err) { if err != io.EOF {
t.Fatal(i, err) t.Fatal(i, err)
} }
...@@ -773,7 +788,7 @@ func TestEnumeratorPrev(t *testing.T) { ...@@ -773,7 +788,7 @@ func TestEnumeratorPrev(t *testing.T) {
k, v, err := en.Prev() k, v, err := en.Prev()
if err != nil { if err != nil {
if !fileutil.IsEOF(err) { if err != io.EOF {
t.Fatal(i, err) t.Fatal(i, err)
} }
......
...@@ -60,8 +60,8 @@ import ( ...@@ -60,8 +60,8 @@ import (
//TODO check vs orig initialize/finalize //TODO check vs orig initialize/finalize
const ( const (
kx = 128 //TODO benchmark tune this number if using custom key/value type(s). kx = 128 // min 2 //TODO benchmark tune this number if using custom key/value type(s).
kd = 64 //TODO benchmark tune this number if using custom key/value type(s). kd = 64 // min 1 //TODO benchmark tune this number if using custom key/value type(s).
) )
type ( type (
...@@ -114,8 +114,8 @@ type ( ...@@ -114,8 +114,8 @@ type (
} }
xe struct { // x element xe struct { // x element
ch interface{} ch interface{}
sep *d k interface{} /*K*/
} }
x struct { // index page x struct { // index page
...@@ -129,6 +129,7 @@ var ( // R/O zero values ...@@ -129,6 +129,7 @@ var ( // R/O zero values
zde de zde de
zx x zx x
zxe xe zxe xe
zk interface{} /*K*/
) )
func clr(q interface{}) { func clr(q interface{}) {
...@@ -156,21 +157,21 @@ func (q *x) extract(i int) { ...@@ -156,21 +157,21 @@ func (q *x) extract(i int) {
if i < q.c { if i < q.c {
copy(q.x[i:], q.x[i+1:q.c+1]) copy(q.x[i:], q.x[i+1:q.c+1])
q.x[q.c].ch = q.x[q.c+1].ch q.x[q.c].ch = q.x[q.c+1].ch
q.x[q.c].sep = nil // GC q.x[q.c].k = zk // GC
q.x[q.c+1] = zxe // GC q.x[q.c+1] = zxe // GC
} }
} }
func (q *x) insert(i int, d *d, ch interface{}) *x { func (q *x) insert(i int, k interface{} /*K*/, ch interface{}) *x {
c := q.c c := q.c
if i < c { if i < c {
q.x[c+1].ch = q.x[c].ch q.x[c+1].ch = q.x[c].ch
copy(q.x[i+2:], q.x[i+1:c]) copy(q.x[i+2:], q.x[i+1:c])
q.x[i+1].sep = q.x[i].sep q.x[i+1].k = q.x[i].k
} }
c++ c++
q.c = c q.c = c
q.x[i].sep = d q.x[i].k = k
q.x[i+1].ch = ch q.x[i+1].ch = ch
return q return q
} }
...@@ -241,7 +242,7 @@ func (t *Tree) cat(p *x, q, r *d, pi int) { ...@@ -241,7 +242,7 @@ func (t *Tree) cat(p *x, q, r *d, pi int) {
func (t *Tree) catX(p, q, r *x, pi int) { func (t *Tree) catX(p, q, r *x, pi int) {
t.ver++ t.ver++
q.x[q.c].sep = p.x[pi].sep q.x[q.c].k = p.x[pi].k
copy(q.x[q.c+1:], r.x[:r.c]) copy(q.x[q.c+1:], r.x[:r.c])
q.c += r.c + 1 q.c += r.c + 1
q.x[q.c].ch = r.x[r.c].ch //TODO recycle r q.x[q.c].ch = r.x[r.c].ch //TODO recycle r
...@@ -249,10 +250,10 @@ func (t *Tree) catX(p, q, r *x, pi int) { ...@@ -249,10 +250,10 @@ func (t *Tree) catX(p, q, r *x, pi int) {
p.c-- p.c--
pc := p.c pc := p.c
if pi < pc { if pi < pc {
p.x[pi].sep = p.x[pi+1].sep p.x[pi].k = p.x[pi+1].k
copy(p.x[pi+1:], p.x[pi+2:pc+1]) copy(p.x[pi+1:], p.x[pi+2:pc+1])
p.x[pc].ch = p.x[pc+1].ch p.x[pc].ch = p.x[pc+1].ch
p.x[pc].sep = nil // GC p.x[pc].k = zk // GC
p.x[pc+1].ch = nil // GC p.x[pc+1].ch = nil // GC
} }
return return
...@@ -268,7 +269,7 @@ func (t *Tree) Delete(k interface{} /*K*/) (ok bool) { ...@@ -268,7 +269,7 @@ func (t *Tree) Delete(k interface{} /*K*/) (ok bool) {
var p *x var p *x
q := t.r q := t.r
if q == nil { if q == nil {
return return false
} }
for { for {
...@@ -277,24 +278,18 @@ func (t *Tree) Delete(k interface{} /*K*/) (ok bool) { ...@@ -277,24 +278,18 @@ func (t *Tree) Delete(k interface{} /*K*/) (ok bool) {
if ok { if ok {
switch x := q.(type) { switch x := q.(type) {
case *x: case *x:
dp := x.x[i].sep if x.c < kx && q != t.r {
switch { t.underflowX(p, &x, pi, &i)
case dp.c > kd:
t.extract(dp, 0)
default:
if x.c < kx && q != t.r {
t.underflowX(p, &x, pi, &i)
}
pi = i + 1
p = x
q = x.x[pi].ch
ok = false
continue
} }
pi = i + 1
p = x
q = x.x[pi].ch
ok = false
continue
case *d: case *d:
t.extract(x, i) t.extract(x, i)
if x.c >= kd { if x.c >= kd {
return return true
} }
if q != t.r { if q != t.r {
...@@ -302,8 +297,8 @@ func (t *Tree) Delete(k interface{} /*K*/) (ok bool) { ...@@ -302,8 +297,8 @@ func (t *Tree) Delete(k interface{} /*K*/) (ok bool) {
} else if t.c == 0 { } else if t.c == 0 {
t.Clear() t.Clear()
} }
return true
} }
return
} }
switch x := q.(type) { switch x := q.(type) {
...@@ -315,7 +310,7 @@ func (t *Tree) Delete(k interface{} /*K*/) (ok bool) { ...@@ -315,7 +310,7 @@ func (t *Tree) Delete(k interface{} /*K*/) (ok bool) {
p = x p = x
q = x.x[i].ch q = x.x[i].ch
case *d: case *d:
return return false
} }
} }
} }
...@@ -340,7 +335,7 @@ func (t *Tree) find(q interface{}, k interface{} /*K*/) (i int, ok bool) { ...@@ -340,7 +335,7 @@ func (t *Tree) find(q interface{}, k interface{} /*K*/) (i int, ok bool) {
h := x.c - 1 h := x.c - 1
for l <= h { for l <= h {
m := (l + h) >> 1 m := (l + h) >> 1
mk = x.x[m].sep.d[0].k mk = x.x[m].k
switch cmp := t.cmp(k, mk); { switch cmp := t.cmp(k, mk); {
case cmp > 0: case cmp > 0:
l = m + 1 l = m + 1
...@@ -368,15 +363,15 @@ func (t *Tree) find(q interface{}, k interface{} /*K*/) (i int, ok bool) { ...@@ -368,15 +363,15 @@ func (t *Tree) find(q interface{}, k interface{} /*K*/) (i int, ok bool) {
return l, false return l, false
} }
// First returns the first item of the tree in the key collating order, or //A // First returns the first item of the tree in the key collating order, or
// (zero-value, zero-value) if the tree is empty. //A // (zero-value, zero-value) if the tree is empty.
func (t *Tree) First() (k interface{} /*K*/, v interface{} /*V*/) { //A func (t *Tree) First() (k interface{} /*K*/, v interface{} /*V*/) {
if q := t.first; q != nil { //A if q := t.first; q != nil {
q := &q.d[0] //A q := &q.d[0]
k, v = q.k, q.v //A k, v = q.k, q.v
} //A }
return //A return
} //A }
// Get returns the value associated with k and true if it exists. Otherwise Get // Get returns the value associated with k and true if it exists. Otherwise Get
// returns (zero-value, false). // returns (zero-value, false).
...@@ -391,7 +386,8 @@ func (t *Tree) Get(k interface{} /*K*/) (v interface{} /*V*/, ok bool) { ...@@ -391,7 +386,8 @@ func (t *Tree) Get(k interface{} /*K*/) (v interface{} /*V*/, ok bool) {
if i, ok = t.find(q, k); ok { if i, ok = t.find(q, k); ok {
switch x := q.(type) { switch x := q.(type) {
case *x: case *x:
return x.x[i].sep.d[0].v, true q = x.x[i+1].ch
continue
case *d: case *d:
return x.d[i].v, true return x.d[i].v, true
} }
...@@ -418,15 +414,15 @@ func (t *Tree) insert(q *d, i int, k interface{} /*K*/, v interface{} /*V*/) *d ...@@ -418,15 +414,15 @@ func (t *Tree) insert(q *d, i int, k interface{} /*K*/, v interface{} /*V*/) *d
return q return q
} }
// Last returns the last item of the tree in the key collating order, or //A // Last returns the last item of the tree in the key collating order, or
// (zero-value, zero-value) if the tree is empty. //A // (zero-value, zero-value) if the tree is empty.
func (t *Tree) Last() (k interface{} /*K*/, v interface{} /*V*/) { //A func (t *Tree) Last() (k interface{} /*K*/, v interface{} /*V*/) {
if q := t.last; q != nil { //A if q := t.last; q != nil {
q := &q.d[q.c-1] //A q := &q.d[q.c-1]
k, v = q.k, q.v //A k, v = q.k, q.v
} //A }
return //A return
} //A }
// Len returns the number of items in the tree. // Len returns the number of items in the tree.
func (t *Tree) Len() int { func (t *Tree) Len() int {
...@@ -440,6 +436,7 @@ func (t *Tree) overflow(p *x, q *d, pi, i int, k interface{} /*K*/, v interface{ ...@@ -440,6 +436,7 @@ func (t *Tree) overflow(p *x, q *d, pi, i int, k interface{} /*K*/, v interface{
if l != nil && l.c < 2*kd { if l != nil && l.c < 2*kd {
l.mvL(q, 1) l.mvL(q, 1)
t.insert(q, i-1, k, v) t.insert(q, i-1, k, v)
p.x[pi-1].k = q.d[0].k
return return
} }
...@@ -447,8 +444,10 @@ func (t *Tree) overflow(p *x, q *d, pi, i int, k interface{} /*K*/, v interface{ ...@@ -447,8 +444,10 @@ func (t *Tree) overflow(p *x, q *d, pi, i int, k interface{} /*K*/, v interface{
if i < 2*kd { if i < 2*kd {
q.mvR(r, 1) q.mvR(r, 1)
t.insert(q, i, k, v) t.insert(q, i, k, v)
p.x[pi].k = r.d[0].k
} else { } else {
t.insert(r, 0, k, v) t.insert(r, 0, k, v)
p.x[pi].k = k
} }
return return
} }
...@@ -471,8 +470,8 @@ func (t *Tree) Seek(k interface{} /*K*/) (e *Enumerator, ok bool) { ...@@ -471,8 +470,8 @@ func (t *Tree) Seek(k interface{} /*K*/) (e *Enumerator, ok bool) {
if i, ok = t.find(q, k); ok { if i, ok = t.find(q, k); ok {
switch x := q.(type) { switch x := q.(type) {
case *x: case *x:
e = &Enumerator{nil, ok, 0, k, x.x[i].sep, t, t.ver} q = x.x[i+1].ch
return continue
case *d: case *d:
e = &Enumerator{nil, ok, i, k, x, t, t.ver} e = &Enumerator{nil, ok, i, k, x, t, t.ver}
return return
...@@ -512,45 +511,50 @@ func (t *Tree) SeekLast() (e *Enumerator, err error) { ...@@ -512,45 +511,50 @@ func (t *Tree) SeekLast() (e *Enumerator, err error) {
// Set sets the value associated with k. // Set sets the value associated with k.
func (t *Tree) Set(k interface{} /*K*/, v interface{} /*V*/) { func (t *Tree) Set(k interface{} /*K*/, v interface{} /*V*/) {
//dbg("--- PRE Set(%v, %v)\n%s", k, v, t.dump())
//defer func() {
// dbg("--- POST\n%s\n====\n", t.dump())
//}()
pi := -1 pi := -1
var p *x var p *x
q := t.r q := t.r
if q != nil { if q == nil {
for { z := t.insert(&d{}, 0, k, v)
i, ok := t.find(q, k) t.r, t.first, t.last = z, z, z
if ok { return
switch x := q.(type) { }
case *x:
x.x[i].sep.d[0].v = v
case *d:
x.d[i].v = v
}
return
}
for {
i, ok := t.find(q, k)
if ok {
switch x := q.(type) { switch x := q.(type) {
case *x: case *x:
if x.c > 2*kx { q = x.x[i+1].ch
t.splitX(p, &x, pi, &i) continue
}
pi = i
p = x
q = x.x[i].ch
case *d: case *d:
switch { x.d[i].v = v
case x.c < 2*kd:
t.insert(x, i, k, v)
default:
t.overflow(p, x, pi, i, k, v)
}
return
} }
return
} }
}
z := t.insert(&d{}, 0, k, v) switch x := q.(type) {
t.r, t.first, t.last = z, z, z case *x:
return if x.c > 2*kx {
t.splitX(p, &x, pi, &i)
}
pi = i
p = x
q = x.x[i].ch
case *d:
switch {
case x.c < 2*kd:
t.insert(x, i, k, v)
default:
t.overflow(p, x, pi, i, k, v)
}
return
}
}
} }
// Put combines Get and Set in a more efficient way where the tree is walked // Put combines Get and Set in a more efficient way where the tree is walked
...@@ -576,13 +580,8 @@ func (t *Tree) Put(k interface{} /*K*/, upd func(oldV interface{} /*V*/, exists ...@@ -576,13 +580,8 @@ func (t *Tree) Put(k interface{} /*K*/, upd func(oldV interface{} /*V*/, exists
if ok { if ok {
switch x := q.(type) { switch x := q.(type) {
case *x: case *x:
oldV = x.x[i].sep.d[0].v q = x.x[i+1].ch
newV, written = upd(oldV, true) continue
if !written {
return
}
x.x[i].sep.d[0].v = newV
case *d: case *d:
oldV = x.d[i].v oldV = x.d[i].v
newV, written = upd(oldV, true) newV, written = upd(oldV, true)
...@@ -649,13 +648,17 @@ func (t *Tree) split(p *x, q *d, pi, i int, k interface{} /*K*/, v interface{} / ...@@ -649,13 +648,17 @@ func (t *Tree) split(p *x, q *d, pi, i int, k interface{} /*K*/, v interface{} /
} }
q.c = kd q.c = kd
r.c = kd r.c = kd
var done bool
if i > kd {
done = true
t.insert(r, i-kd, k, v)
}
if pi >= 0 { if pi >= 0 {
p.insert(pi, r, r) p.insert(pi, r.d[0].k, r)
} else { } else {
t.r = newX(q).insert(0, r, r) t.r = newX(q).insert(0, r.d[0].k, r)
} }
if i > kd { if done {
t.insert(r, i-kd, k, v)
return return
} }
...@@ -670,11 +673,11 @@ func (t *Tree) splitX(p *x, pp **x, pi int, i *int) { ...@@ -670,11 +673,11 @@ func (t *Tree) splitX(p *x, pp **x, pi int, i *int) {
q.c = kx q.c = kx
r.c = kx r.c = kx
if pi >= 0 { if pi >= 0 {
p.insert(pi, q.x[kx].sep, r) p.insert(pi, q.x[kx].k, r)
} else { } else {
t.r = newX(q).insert(0, q.x[kx].sep, r) t.r = newX(q).insert(0, q.x[kx].k, r)
} }
q.x[kx].sep = nil q.x[kx].k = zk
for i := range q.x[kx+1:] { for i := range q.x[kx+1:] {
q.x[kx+i+1] = zxe q.x[kx+i+1] = zxe
} }
...@@ -690,8 +693,10 @@ func (t *Tree) underflow(p *x, q *d, pi int) { ...@@ -690,8 +693,10 @@ func (t *Tree) underflow(p *x, q *d, pi int) {
if l != nil && l.c+q.c >= 2*kd { if l != nil && l.c+q.c >= 2*kd {
l.mvR(q, 1) l.mvR(q, 1)
p.x[pi-1].k = q.d[0].k
} else if r != nil && q.c+r.c >= 2*kd { } else if r != nil && q.c+r.c >= 2*kd {
q.mvL(r, 1) q.mvL(r, 1)
p.x[pi].k = r.d[0].k
r.d[r.c] = zde // GC r.d[r.c] = zde // GC
} else if l != nil { } else if l != nil {
t.cat(p, l, q, pi-1) t.cat(p, l, q, pi-1)
...@@ -718,24 +723,24 @@ func (t *Tree) underflowX(p *x, pp **x, pi int, i *int) { ...@@ -718,24 +723,24 @@ func (t *Tree) underflowX(p *x, pp **x, pi int, i *int) {
q.x[q.c+1].ch = q.x[q.c].ch q.x[q.c+1].ch = q.x[q.c].ch
copy(q.x[1:], q.x[:q.c]) copy(q.x[1:], q.x[:q.c])
q.x[0].ch = l.x[l.c].ch q.x[0].ch = l.x[l.c].ch
q.x[0].sep = p.x[pi-1].sep q.x[0].k = p.x[pi-1].k
q.c++ q.c++
*i++ *i++
l.c-- l.c--
p.x[pi-1].sep = l.x[l.c].sep p.x[pi-1].k = l.x[l.c].k
return return
} }
if r != nil && r.c > kx { if r != nil && r.c > kx {
q.x[q.c].sep = p.x[pi].sep q.x[q.c].k = p.x[pi].k
q.c++ q.c++
q.x[q.c].ch = r.x[0].ch q.x[q.c].ch = r.x[0].ch
p.x[pi].sep = r.x[0].sep p.x[pi].k = r.x[0].k
copy(r.x[:], r.x[1:r.c]) copy(r.x[:], r.x[1:r.c])
r.c-- r.c--
rc := r.c rc := r.c
r.x[rc].ch = r.x[rc+1].ch r.x[rc].ch = r.x[rc+1].ch
r.x[rc].sep = nil r.x[rc].k = zk
r.x[rc+1].ch = nil r.x[rc+1].ch = nil
return return
} }
......
This diff is collapsed.
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment