Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
B
b
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
b
Commits
0496ce60
Commit
0496ce60
authored
Apr 19, 2014
by
Jan Mercl
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Add Put().
parent
a9488ca6
Changes
2
Show whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
210 additions
and
7 deletions
+210
-7
all_test.go
all_test.go
+120
-1
btree.go
btree.go
+90
-6
No files found.
all_test.go
View file @
0496ce60
...
@@ -11,6 +11,7 @@ import (
...
@@ -11,6 +11,7 @@ import (
"path"
"path"
"runtime"
"runtime"
"runtime/debug"
"runtime/debug"
"strings"
"testing"
"testing"
"github.com/cznic/fileutil"
"github.com/cznic/fileutil"
...
@@ -19,7 +20,10 @@ import (
...
@@ -19,7 +20,10 @@ import (
func
use
(
...
interface
{})
{}
func
use
(
...
interface
{})
{}
var
dbg
=
func
(
s
string
,
va
...
interface
{})
{
func
dbg
(
s
string
,
va
...
interface
{})
{
if
s
==
""
{
s
=
strings
.
Repeat
(
"%v "
,
len
(
va
))
}
_
,
fn
,
fl
,
_
:=
runtime
.
Caller
(
1
)
_
,
fn
,
fl
,
_
:=
runtime
.
Caller
(
1
)
fmt
.
Printf
(
"%s:%d: "
,
path
.
Base
(
fn
),
fl
)
fmt
.
Printf
(
"%s:%d: "
,
path
.
Base
(
fn
),
fl
)
fmt
.
Printf
(
s
,
va
...
)
fmt
.
Printf
(
s
,
va
...
)
...
@@ -931,3 +935,118 @@ func TestSeekLast3(t *testing.T) {
...
@@ -931,3 +935,118 @@ func TestSeekLast3(t *testing.T) {
t
.
Fatal
(
k
,
v
,
err
)
t
.
Fatal
(
k
,
v
,
err
)
}
}
}
}
func
TestPut
(
t
*
testing
.
T
)
{
tab
:=
[]
struct
{
pre
[]
int
// even index: K, odd index: V
newK
int
// Put(newK, ...
oldV
int
// Put()->oldV
exists
bool
// upd(exists)
write
bool
// upd()->write
post
[]
int
// even index: K, odd index: V
}{
// 0
{
[]
int
{},
1
,
0
,
false
,
false
,
[]
int
{},
},
{
[]
int
{},
1
,
0
,
false
,
true
,
[]
int
{
1
,
-
1
},
},
{
[]
int
{
1
,
10
},
0
,
0
,
false
,
false
,
[]
int
{
1
,
10
},
},
{
[]
int
{
1
,
10
},
0
,
0
,
false
,
true
,
[]
int
{
0
,
-
1
,
1
,
10
},
},
{
[]
int
{
1
,
10
},
1
,
10
,
true
,
false
,
[]
int
{
1
,
10
},
},
// 5
{
[]
int
{
1
,
10
},
1
,
10
,
true
,
true
,
[]
int
{
1
,
-
1
},
},
{
[]
int
{
1
,
10
},
2
,
0
,
false
,
false
,
[]
int
{
1
,
10
},
},
{
[]
int
{
1
,
10
},
2
,
0
,
false
,
true
,
[]
int
{
1
,
10
,
2
,
-
1
},
},
}
for
iTest
,
test
:=
range
tab
{
tr
:=
TreeNew
(
cmp
)
for
i
:=
0
;
i
<
len
(
test
.
pre
);
i
+=
2
{
k
,
v
:=
test
.
pre
[
i
],
test
.
pre
[
i
+
1
]
tr
.
Set
(
k
,
v
)
}
oldV
,
written
:=
tr
.
Put
(
test
.
newK
,
func
(
old
interface
{},
exists
bool
)
(
newV
interface
{},
write
bool
)
{
if
g
,
e
:=
exists
,
test
.
exists
;
g
!=
e
{
t
.
Fatal
(
iTest
,
g
,
e
)
}
if
exists
{
if
g
,
e
:=
old
.
(
int
),
test
.
oldV
;
g
!=
e
{
t
.
Fatal
(
iTest
,
g
,
e
)
}
}
return
-
1
,
test
.
write
})
if
test
.
exists
{
if
g
,
e
:=
oldV
.
(
int
),
test
.
oldV
;
g
!=
e
{
t
.
Fatal
(
iTest
,
g
,
e
)
}
}
if
g
,
e
:=
written
,
test
.
write
;
g
!=
e
{
t
.
Fatal
(
iTest
,
g
,
e
)
}
n
:=
len
(
test
.
post
)
en
,
err
:=
tr
.
SeekFirst
()
if
err
!=
nil
{
if
n
==
0
&&
err
==
io
.
EOF
{
continue
}
t
.
Fatal
(
iTest
,
err
)
}
for
i
:=
0
;
i
<
len
(
test
.
post
);
i
+=
2
{
k
,
v
,
err
:=
en
.
Next
()
if
err
!=
nil
{
t
.
Fatal
(
iTest
,
err
)
}
if
g
,
e
:=
k
.
(
int
),
test
.
post
[
i
];
g
!=
e
{
t
.
Fatal
(
iTest
,
g
,
e
)
}
if
g
,
e
:=
v
.
(
int
),
test
.
post
[
i
+
1
];
g
!=
e
{
t
.
Fatal
(
iTest
,
g
,
e
)
}
}
_
,
_
,
err
=
en
.
Next
()
if
g
,
e
:=
err
,
io
.
EOF
;
g
!=
e
{
t
.
Fatal
(
iTest
,
g
,
e
)
}
}
}
btree.go
View file @
0496ce60
...
@@ -4,6 +4,12 @@
...
@@ -4,6 +4,12 @@
// Package b implements a B+tree.
// Package b implements a B+tree.
//
//
// Changelog
//
// 2014-04-18: Added new method Put.
//
// Generic types
//
// Keys and their associated values are interface{} typed, similar to all of
// Keys and their associated values are interface{} typed, similar to all of
// the containers in the standard library.
// the containers in the standard library.
//
//
...
@@ -289,8 +295,8 @@ func (t *Tree) catX(p, q, r *x, pi int) {
...
@@ -289,8 +295,8 @@ func (t *Tree) catX(p, q, r *x, pi int) {
t
.
r
=
q
//TODO recycle r
t
.
r
=
q
//TODO recycle r
}
}
//Delete removes the k's KV pair, if it exists, in which case Delete returns
//
Delete removes the k's KV pair, if it exists, in which case Delete returns
//true.
//
true.
func
(
t
*
Tree
)
Delete
(
k
interface
{}
/*K*/
)
(
ok
bool
)
{
func
(
t
*
Tree
)
Delete
(
k
interface
{}
/*K*/
)
(
ok
bool
)
{
pi
:=
-
1
pi
:=
-
1
var
p
*
x
var
p
*
x
...
@@ -397,7 +403,7 @@ func (t *Tree) find(q interface{}, k interface{} /*K*/) (i int, ok bool) {
...
@@ -397,7 +403,7 @@ func (t *Tree) find(q interface{}, k interface{} /*K*/) (i int, ok bool) {
}
}
// First returns the first item of the tree in the key collating order, or
// First returns the first item of the tree in the key collating order, or
// (
nil, nil
) if the tree is empty.
// (
zero-value, zero-value
) if the tree is empty.
func
(
t
*
Tree
)
First
()
(
k
interface
{}
/*K*/
,
v
interface
{}
/*V*/
)
{
func
(
t
*
Tree
)
First
()
(
k
interface
{}
/*K*/
,
v
interface
{}
/*V*/
)
{
if
q
:=
t
.
first
;
q
!=
nil
{
if
q
:=
t
.
first
;
q
!=
nil
{
q
:=
&
q
.
d
[
0
]
q
:=
&
q
.
d
[
0
]
...
@@ -407,7 +413,7 @@ func (t *Tree) First() (k interface{} /*K*/, v interface{} /*V*/) {
...
@@ -407,7 +413,7 @@ func (t *Tree) First() (k interface{} /*K*/, v interface{} /*V*/) {
}
}
// 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 (
nil
, false).
// returns (
zero-value
, false).
func
(
t
*
Tree
)
Get
(
k
interface
{}
/*K*/
)
(
v
interface
{}
/*V*/
,
ok
bool
)
{
func
(
t
*
Tree
)
Get
(
k
interface
{}
/*K*/
)
(
v
interface
{}
/*V*/
,
ok
bool
)
{
q
:=
t
.
r
q
:=
t
.
r
if
q
==
nil
{
if
q
==
nil
{
...
@@ -446,8 +452,8 @@ func (t *Tree) insert(q *d, i int, k interface{} /*K*/, v interface{} /*V*/) *d
...
@@ -446,8 +452,8 @@ 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
(nil,
// Last returns the last item of the tree in the key collating order, or
//
nil
) if the tree is empty.
//
(zero-value, zero-value
) if the tree is empty.
func
(
t
*
Tree
)
Last
()
(
k
interface
{}
/*K*/
,
v
interface
{}
/*V*/
)
{
func
(
t
*
Tree
)
Last
()
(
k
interface
{}
/*K*/
,
v
interface
{}
/*V*/
)
{
if
q
:=
t
.
last
;
q
!=
nil
{
if
q
:=
t
.
last
;
q
!=
nil
{
q
:=
&
q
.
d
[
q
.
c
-
1
]
q
:=
&
q
.
d
[
q
.
c
-
1
]
...
@@ -581,6 +587,84 @@ func (t *Tree) Set(k interface{} /*K*/, v interface{} /*V*/) {
...
@@ -581,6 +587,84 @@ func (t *Tree) Set(k interface{} /*K*/, v interface{} /*V*/) {
return
return
}
}
// Put combines Get and Set in a more efficient way where the tree is walked
// only once. The upd(ater) receives (old-value, true) if a KV pair for k
// exists or (zero-value, false) otherwise. It can then return a (new-value,
// true) to create or overwrite the existing value in the KV pair, or
// (whatever, false) if it decides not to create or not to update the value of
// the KV pair.
//
// tree.Set(k, v) conceptually equals
//
// tree.Put(k, func(k, v []byte){ return v, true }([]byte, bool))
//
// modulo the differing return values.
func
(
t
*
Tree
)
Put
(
k
interface
{}
/*K*/
,
upd
func
(
oldV
interface
{}
/*V*/
,
exists
bool
)
(
newV
interface
{}
/*V*/
,
write
bool
))
(
oldV
interface
{}
/*V*/
,
written
bool
)
{
pi
:=
-
1
var
p
*
x
q
:=
t
.
r
var
newV
interface
{}
/*V*/
if
q
!=
nil
{
for
{
i
,
ok
:=
t
.
find
(
q
,
k
)
if
ok
{
switch
x
:=
q
.
(
type
)
{
case
*
x
:
oldV
=
x
.
x
[
i
]
.
sep
.
d
[
0
]
.
v
newV
,
written
=
upd
(
oldV
,
true
)
if
!
written
{
return
}
x
.
x
[
i
]
.
sep
.
d
[
0
]
.
v
=
newV
case
*
d
:
oldV
=
x
.
d
[
i
]
.
v
newV
,
written
=
upd
(
oldV
,
true
)
if
!
written
{
return
}
x
.
d
[
i
]
.
v
=
newV
}
return
}
switch
x
:=
q
.
(
type
)
{
case
*
x
:
if
x
.
c
>
2
*
kx
{
t
.
splitX
(
p
,
&
x
,
pi
,
&
i
)
}
pi
=
i
p
=
x
q
=
x
.
x
[
i
]
.
ch
case
*
d
:
// new KV pair
newV
,
written
=
upd
(
newV
,
false
)
if
!
written
{
return
}
switch
{
case
x
.
c
<
2
*
kd
:
t
.
insert
(
x
,
i
,
k
,
newV
)
default
:
t
.
overflow
(
p
,
x
,
pi
,
i
,
k
,
newV
)
}
return
}
}
}
// new KV pair in empty tree
newV
,
written
=
upd
(
newV
,
false
)
if
!
written
{
return
}
z
:=
t
.
insert
(
&
d
{},
0
,
k
,
newV
)
t
.
r
,
t
.
first
,
t
.
last
=
z
,
z
,
z
return
}
func
(
t
*
Tree
)
split
(
p
*
x
,
q
*
d
,
pi
,
i
int
,
k
interface
{}
/*K*/
,
v
interface
{}
/*V*/
)
{
func
(
t
*
Tree
)
split
(
p
*
x
,
q
*
d
,
pi
,
i
int
,
k
interface
{}
/*K*/
,
v
interface
{}
/*V*/
)
{
t
.
ver
++
t
.
ver
++
r
:=
&
d
{}
r
:=
&
d
{}
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment