Skip to content

Commit

Permalink
Rename to FastLruStringMap.
Browse files Browse the repository at this point in the history
  • Loading branch information
kennethshackleton committed Jun 21, 2024
1 parent b02fa10 commit fbe82ea
Show file tree
Hide file tree
Showing 3 changed files with 54 additions and 42 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@

package com.bloomberg.selekt.collections.map.benchmarks

import com.bloomberg.selekt.collections.map.FastLruStringMap
import com.bloomberg.selekt.collections.map.FastAccessOrderedStringMap
import org.openjdk.jmh.annotations.Benchmark
import org.openjdk.jmh.annotations.BenchmarkMode
import org.openjdk.jmh.annotations.Level
Expand All @@ -27,11 +27,11 @@ import org.openjdk.jmh.annotations.State

@State(Scope.Thread)
open class LruMapInput {
internal lateinit var map: FastLruStringMap<Any>
internal lateinit var map: FastAccessOrderedStringMap<Any>

@Setup(Level.Iteration)
fun setUp() {
map = FastLruStringMap(1)
map = FastAccessOrderedStringMap(1)
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -19,11 +19,11 @@ package com.bloomberg.selekt.collections.map
import javax.annotation.concurrent.NotThreadSafe

@NotThreadSafe
class FastLruStringMap<T>(
class FastAccessOrderedStringMap<T>(
capacity: Int
) : FastStringMap<T>(capacity) {
private var accessCount = Int.MIN_VALUE
private var spare: LruEntry<T>? = null
private var accessNumber = Int.MIN_VALUE
private var spare: AOEntry<T>? = null

inline fun getElsePut(
key: String,
Expand All @@ -32,7 +32,7 @@ class FastLruStringMap<T>(
val hashCode = hash(key)
val index = hashIndex(hashCode)
entryMatching(index, hashCode, key)?.let {
(it as LruEntry<T>).accessCount = nextAccessCount()
(it as AOEntry<T>).accessNumber = nextAccessCount()
return it.value!!
}
return addAssociation(index, hashCode, key, supplier()).value!!
Expand All @@ -48,46 +48,58 @@ class FastLruStringMap<T>(
spare = null
return it.update(index, hashCode, key, value, nextAccessCount(), store[index])
}
return LruEntry(index, hashCode, key, value, nextAccessCount(), store[index])
return AOEntry(index, hashCode, key, value, nextAccessCount(), store[index])
}

override fun clear() {
super.clear()
spare = null
}

internal fun asLinkedMap(
capacity: Int = size
) = FastLinkedStringMap<T>(capacity = capacity).apply {
entries().sortedBy {
(it as AOEntry<T>).accessNumber
}.forEach {
addAssociation(it.index, it.hashCode, it.key, it.value!!)
}
}

@PublishedApi
internal fun nextAccessCount(): Int {
accessCount += 1
if (accessCount == Int.MIN_VALUE) {
accessNumber += 1
if (accessNumber == Int.MIN_VALUE) {
resetAllAccessCounts()
}
return accessCount
return accessNumber
}

private fun entries(): Iterable<Entry<T>> = store.flatMap {
sequence {
var current = it
while (current != null) {
yield(current)
current = current.after
}
}
}.asIterable()

private fun resetAllAccessCounts() {
store.flatMap<Entry<T>?, Entry<T>> {
sequence {
var current = it
while (current != null) {
yield(current)
current = current.after
}
}.toList()
}.sortedBy {
(it as LruEntry<T>).accessCount
entries().sortedBy {
(it as AOEntry<T>).accessNumber
}.forEachIndexed { index, it ->
(it as LruEntry<T>).accessCount = Int.MIN_VALUE + index
(it as AOEntry<T>).accessNumber = Int.MIN_VALUE + index
}
}

@PublishedApi
internal class LruEntry<T>(
internal class AOEntry<T>(
index: Int,
hashCode: Int,
key: String,
value: T,
var accessCount: Int,
var accessNumber: Int,
after: Entry<T>?
) : Entry<T>(index, hashCode, key, value, after) {
@Suppress("NOTHING_TO_INLINE")
Expand All @@ -103,7 +115,7 @@ class FastLruStringMap<T>(
this.hashCode = hashCode
this.key = key
this.value = value
this.accessCount = accessCount
this.accessNumber = accessCount
this.after = after
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -30,26 +30,26 @@ import kotlin.test.assertSame
import kotlin.test.assertTrue
import kotlin.test.fail

internal class FastLruStringMapTest {
internal class FastAccessOrderedStringMapTest {
@Test
fun get() {
val first = Any()
val map = FastLruStringMap<Any>(1)
val map = FastAccessOrderedStringMap<Any>(1)
assertSame(first, map.getElsePut("1") { first })
}

@Test
fun sizeOne() {
val first = Any()
val map = FastLruStringMap<Any>(1)
val map = FastAccessOrderedStringMap<Any>(1)
map.getElsePut("1") { first }
assertEquals(1, map.size)
}

@Test
fun getTwice() {
val first = Any()
val map = FastLruStringMap<Any>(1)
val map = FastAccessOrderedStringMap<Any>(1)
map.getElsePut("1") { first }
assertSame(first, map.getElsePut("1") { fail() })
}
Expand All @@ -58,7 +58,7 @@ internal class FastLruStringMapTest {
fun getWhenAbsent() {
val supplier = mock<() -> Any>()
whenever(supplier.invoke()) doReturn Any()
val map = FastLruStringMap<Any>(1)
val map = FastAccessOrderedStringMap<Any>(1)
val item = map.getElsePut("1", supplier)
verify(supplier, times(1)).invoke()
assertSame(item, map.getElsePut("1", supplier))
Expand All @@ -69,7 +69,7 @@ internal class FastLruStringMapTest {
fun getTwo() {
val first = Any()
val second = Any()
val map = FastLruStringMap<Any>(64)
val map = FastAccessOrderedStringMap<Any>(64)
map.getElsePut("1") { first }
map.getElsePut("2") { second }
assertEquals(2, map.size)
Expand All @@ -79,7 +79,7 @@ internal class FastLruStringMapTest {
fun getTwoWithCollisions() {
val first = Any()
val second = Any()
val map = FastLruStringMap<Any>(1)
val map = FastAccessOrderedStringMap<Any>(1)
map.getElsePut("1") { first }
map.getElsePut("2") { second }
assertSame(first, map.getElsePut("1") { fail() })
Expand All @@ -90,7 +90,7 @@ internal class FastLruStringMapTest {
fun sizeTwo() {
val first = Any()
val second = Any()
val map = FastLruStringMap<Any>(1)
val map = FastAccessOrderedStringMap<Any>(1)
map.getElsePut("1") { first }
map.getElsePut("2") { second }
assertSame(first, map.getElsePut("1") { fail() })
Expand All @@ -100,7 +100,7 @@ internal class FastLruStringMapTest {
@Test
fun removeOne() {
val first = Any()
val map = FastLruStringMap<Any>(1)
val map = FastAccessOrderedStringMap<Any>(1)
map.getElsePut("1") { first }
assertSame(first, map.removeEntry("1").value)
}
Expand All @@ -109,7 +109,7 @@ internal class FastLruStringMapTest {
fun removeTwo() {
val first = Any()
val second = Any()
val map = FastLruStringMap<Any>(2)
val map = FastAccessOrderedStringMap<Any>(2)
map.getElsePut("1") { first }
map.getElsePut("2") { second }
assertSame(first, map.removeEntry("1").value)
Expand All @@ -120,7 +120,7 @@ internal class FastLruStringMapTest {
fun removeTwoWithCollisions() {
val first = Any()
val second = Any()
val map = FastLruStringMap<Any>(1)
val map = FastAccessOrderedStringMap<Any>(1)
map.getElsePut("1") { first }
map.getElsePut("2") { second }
assertSame(first, map.removeEntry("1").value)
Expand All @@ -130,15 +130,15 @@ internal class FastLruStringMapTest {
@Test
fun removeThenSize() {
val first = Any()
val map = FastLruStringMap<Any>(1)
val map = FastAccessOrderedStringMap<Any>(1)
map.getElsePut("1") { first }
map.removeEntry("1")
assertEquals(0, map.size)
}

@Test
fun removeWhenEmpty() {
val map = FastLruStringMap<Any>(1)
val map = FastAccessOrderedStringMap<Any>(1)
assertThrows<NoSuchElementException> {
map.removeEntry("1")
}
Expand All @@ -147,7 +147,7 @@ internal class FastLruStringMapTest {

@Test
fun clear() {
val map = FastLruStringMap<Any>(1)
val map = FastAccessOrderedStringMap<Any>(1)
map.getElsePut("1") { Any() }
assertEquals(1, map.size)
map.clear()
Expand All @@ -156,7 +156,7 @@ internal class FastLruStringMapTest {

@Test
fun clearWhenEmpty() {
val map = FastLruStringMap<Any>(1)
val map = FastAccessOrderedStringMap<Any>(1)
map.clear()
assertTrue(map.isEmpty())
}
Expand All @@ -165,7 +165,7 @@ internal class FastLruStringMapTest {
fun containsFalse() {
val supplier = mock<() -> Any>()
whenever(supplier.invoke()) doReturn Any()
val map = FastLruStringMap<Any>(1)
val map = FastAccessOrderedStringMap<Any>(1)
map.getElsePut("1", supplier)
assertFalse(map.containsKey("2"))
}
Expand All @@ -174,7 +174,7 @@ internal class FastLruStringMapTest {
fun containsTrue() {
val supplier = mock<() -> Any>()
whenever(supplier.invoke()) doReturn Any()
val map = FastLruStringMap<Any>(1)
val map = FastAccessOrderedStringMap<Any>(1)
map.getElsePut("1", supplier)
assertTrue(map.containsKey("1"))
}
Expand Down

0 comments on commit fbe82ea

Please sign in to comment.