Keep data as a linked list on disk,A alternative way to reduce redundant operation for DiskLruCache
DiskLinkedList
Keep data as a linked list on disk. An alternative way to reduce redundant operation for DiskLruCache
Use-case
Android have build-in DiskLruCache.java, but it has too many redundant operations on runtime,
DiskLinkedList
using memory-mapped file mechanism to keep data on disk as a linked list.
Advantage
- Compatible with all systems running Java (Android, JVM, Kotlin/JVM).
- Get existed / remove (move / free): O(1).
- New put / get (allocation): O(n) on worst fragment or O(1) on defragmented.
- High performance.
- Inconsiderable I/O execution.
- Easy to use.
File data structure
1 byte (B) to make use or not | 4B prev | 4B next | 4B length | extra byte array
Each entry, DiskLinkedList
storage as a node and keep as structure above.
Usage
Create new disk linked list data
val diskLinkedList = DiskLinkedList.open(
file = File("data.dm"), // file on disk.
initialByteSize = 20 * 1024L, // initial size of file.
scaleFactor = 0.75f // file size scale factor on data larger than current size.
)
// Create new record.
diskLinkedList.addFirst("Hello, World!".toData())
diskLinkedList.log() // 0=Hello, World!
Data on disk
More and more
// Add "Second Node" to first.
diskLinkedList.addFirst("Second Node".toData())
diskLinkedList.log()
// Output: 26=Second Node > 0=Hello, World!
// Keep "Second Node" to move later.
val second = diskLinkedList.first!!
// Add 3 nodes to first.
repeat(3) { index ->
diskLinkedList.addFirst("Hello $index".toData())
}
diskLinkedList.log()
// Output: 90=Hello 2 > 70=Hello 1 > 50=Hello 0 > 26=Second Node > 0=Hello, World!
// Move "Second Node" to first.
diskLinkedList.moveToFirst(second)
diskLinkedList.log()
// Output: 26=Second Node > 90=Hello 2 > 70=Hello 1 > 50=Hello 0 > 0=Hello, World!
Sync and de-fragment
DiskLinkedList
NOT auto-sync to file, it only sync to file
aftersynchronize
executed.- Beside of that, to avoid fragment on file,
DiskLinkedList
supportdefragment
function to do.
// Remove last.
diskLinkedList.removeLast()
diskLinkedList.log()
// Output: 26=Second Node > 90=Hello 2 > 70=Hello 1 > 50=Hello 0
// Then continue to add 2 nodes at first.
repeat(2) { index ->
diskLinkedList.addFirst("Bye $index".toData())
}
diskLinkedList.log()
// Output: 110=Bye 1 > 0=Bye 0 > 26=Second Node > 90=Hello 2 > 70=Hello 1 > 50=Hello 0
// New node "Bye 1" will add at pointer 110 instead of 0 -> FRAGMENTED!!!
// Defragment data.
diskLinkedList.defragment()
diskLinkedList.log()
// Output: 0=Bye 1 > 18=Bye 0 > 36=Second Node > 60=Hello 2 > 80=Hello 1 > 100=Hello 0
// Sync every change to disk.
diskLinkedList.synchronize()
Dependency
Maven
<dependency>
<groupId>org.cuongnv.disklinkedlist</groupId>
<artifactId>disklinkedlist</artifactId>
<version>0.5.0</version>
</dependency>
Gradle Kotlin DSL
implementation("org.cuongnv.disklinkedlist:disklinkedlist:0.5.0")
Gradle Groovy
implementation 'org.cuongnv.disklinkedlist:disklinkedlist:0.5.0'
License
Copyright 2021 Cuong V. Nguyen (github.com/cuongnv126).
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.