Skip to content

Commit

Permalink
Skiplist iterator
Browse files Browse the repository at this point in the history
  • Loading branch information
tomfran committed Oct 1, 2023
1 parent 186acf3 commit 6590192
Show file tree
Hide file tree
Showing 4 changed files with 74 additions and 30 deletions.
4 changes: 2 additions & 2 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -91,7 +91,7 @@ To locate an element, we start from the top level and move forward until we find
we are looking for. Then we move down to the next level and repeat the process until we find the element.

Insertions, deletions, and updates are done by first locating the element, then performing
the operation on the node. All of them have a time complexity of `O(log(n))`.
the operation on the node. All of them have an average time complexity of `O(log(n))`.

---

Expand Down Expand Up @@ -147,7 +147,7 @@ c.t.l.memtable.SkipListBenchmark.get thrpt 10 823423.128 ± 8302
- [x] File initialization
- [ ] Skip-List
- [x] Operations
- [ ] Iterator
- [x] Iterator
- [ ] Tree
- [ ] Operations
- [ ] Background compaction
Expand Down
28 changes: 11 additions & 17 deletions src/main/java/com/tomfran/lsm/memtable/Memtable.java
Original file line number Diff line number Diff line change
Expand Up @@ -3,42 +3,36 @@
import com.tomfran.lsm.sstable.SSTable;
import com.tomfran.lsm.types.Item;

import java.util.LinkedList;

public class Memtable {

SkipList mutableData;
LinkedList<SkipList> immutableData;
LinkedList<LinkedList<SSTable>> sstables;
static final int DEFAULT_SAMPLE_SIZE = 1000;

SkipList list;


public Memtable() {
mutableData = new SkipList();
immutableData = new LinkedList<>();
list = new SkipList();
}

public void add(Item item) {
mutableData.add(item);
list.add(item);
}

public void get(byte[] key) {
mutableData.get(key);
list.get(key);
}

public void remove(byte[] key) {
mutableData.remove(key);
list.add(new Item(key, null));
}

public int size() {
return mutableData.size();
}

private void replaceMutableData() {
immutableData.addFirst(mutableData);
mutableData = new SkipList();
return list.size();
}

public SSTable flush() {
return null;
String filename = "sstable-" + System.currentTimeMillis();
return new SSTable(filename, list, DEFAULT_SAMPLE_SIZE, list.size());
}

}
57 changes: 55 additions & 2 deletions src/main/java/com/tomfran/lsm/memtable/SkipList.java
Original file line number Diff line number Diff line change
Expand Up @@ -9,19 +9,33 @@
import static java.lang.Math.ceil;
import static java.lang.Math.log;

/**
* A skip list implementation of items.
*/
public class SkipList implements Iterable<Item> {

static final int DEFAULT_ELEMENTS = 1 << 16;

final Node sentinel;

private final Node[] buffer;
private final XoRoShiRo128PlusRandom rn;

int levels;
int size;

/**
* Create a skip list with a default number of elements, 2 ^ 16.
*/
public SkipList() {
this(DEFAULT_ELEMENTS);
}

/**
* Create a skip list with a specified number of elements.
*
* @param numElements The number of elements to size the skip list for.
*/
public SkipList(int numElements) {
levels = (int) ceil(log(numElements) / log(2));
size = 0;
Expand All @@ -30,6 +44,11 @@ public SkipList(int numElements) {
buffer = new Node[levels];
}

/**
* Add an item to the skip list.
*
* @param item The item to add.
*/
public void add(Item item) {
Node current = sentinel;
for (int i = levels - 1; i >= 0; i--) {
Expand Down Expand Up @@ -58,6 +77,12 @@ private int randomLevel() {
return level;
}

/**
* Retrieve an item from the skip list.
*
* @param key The key of the item to retrieve.
* @return The item if found, null otherwise.
*/
public Item get(byte[] key) {
Node current = sentinel;
for (int i = levels - 1; i >= 0; i--) {
Expand All @@ -71,6 +96,11 @@ public Item get(byte[] key) {
return null;
}

/**
* Remove an item from the skip list.
*
* @param key The key of the item to remove.
*/
public void remove(byte[] key) {
Node current = sentinel;
for (int i = levels - 1; i >= 0; i--) {
Expand All @@ -90,13 +120,23 @@ public void remove(byte[] key) {
}
}

/**
* Get the number of items in the skip list.
*
* @return Skip list size.
*/
public int size() {
return size;
}

/**
* Get an iterator over the items in the skip list at the lowest level.
*
* @return An iterator over the items in the skip list.
*/
@Override
public Iterator<Item> iterator() {
return null;
return new SkipListIterator(sentinel);
}

@Override
Expand All @@ -114,7 +154,7 @@ public String toString() {
return sb.toString();
}

static final class Node {
private static final class Node {
Item value;
Node[] next;

Expand All @@ -124,4 +164,17 @@ static final class Node {
}
}

private record SkipListIterator(Node node) implements Iterator<Item> {

@Override
public boolean hasNext() {
return node.next[0] != null;
}

@Override
public Item next() {
return node.next[0].value;
}
}

}
15 changes: 6 additions & 9 deletions src/main/java/com/tomfran/lsm/sstable/SSTable.java
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ public class SSTable {
public static final String BLOOM_FILE_EXTENSION = ".bloom";
public static final String INDEX_FILE_EXTENSION = ".index";

String filename;
ItemsInputStream is;
int size;

Expand All @@ -36,6 +37,7 @@ public class SSTable {
* @param numItems The number of items in the SSTable.
*/
public SSTable(String filename, Iterable<Item> items, int sampleSize, int numItems) {
this.filename = filename;
writeItems(filename, items, sampleSize, numItems);
is = new ItemsInputStream(filename + DATA_FILE_EXTENSION);
}
Expand All @@ -46,6 +48,7 @@ public SSTable(String filename, Iterable<Item> items, int sampleSize, int numIte
* @param filename The base filename of the SSTable.
*/
public SSTable(String filename) {
this.filename = filename;
initializeFromDisk(filename);
}

Expand Down Expand Up @@ -205,13 +208,7 @@ private void writeItems(String filename, Iterable<Item> items, int sampleSize, i
indexOs.close();
}

private static class SSTableIterator implements Iterator<Item> {

private final SSTable table;

public SSTableIterator(SSTable table) {
this.table = table;
}
private record SSTableIterator(SSTable table) implements Iterator<Item> {

@Override
public boolean hasNext() {
Expand All @@ -233,7 +230,7 @@ public Item next() {
*/
private static class SSTableMergerIterator extends IteratorMerger<Item> implements Iterable<Item> {

private Item last, next;
private Item last;

@SafeVarargs
public SSTableMergerIterator(Iterator<Item>... iterators) {
Expand All @@ -248,7 +245,7 @@ public boolean hasNext() {

@Override
public Item next() {
next = super.next();
Item next = super.next();
while (next != null && last.compareTo(next) == 0)
next = super.next();

Expand Down

0 comments on commit 6590192

Please sign in to comment.