基于
my previous question
class EquivalenceLock(val size: Int) {
private[this] val locks = IndexedSeq.fill(size)(new Object())
def apply[U](lock: Any)(f: => U) = locks(lock.hashCode().abs % size).synchronized(f)
}
object EquivalenceLock {
implicit val defaultInstance = new EquivalenceLock(1 << 10)
}
我编写了一些测试来验证我的锁是否按预期工作:
import EquivalenceLock.{defaultInstance => lock}
import scala.concurrent.Future
import scala.concurrent.ExecutionContext.Implicits.global
import scala.collection.mutable
val journal = mutable.ArrayBuffer.empty[String]
def log(msg: String) = journal.synchronized {
println(msg)
journal += msg
}
def test(id: String, napTime: Int) = Future {
lock(id) {
log(s"Entering $id=$napTime")
Thread.sleep(napTime * 1000L)
log(s"Exiting $id=$napTime")
}
}
test("foo", 5)
test("foo", 2)
Thread.sleep(20 * 1000L)
val validAnswers = Set(
Seq("Entering foo=5", "Exiting foo=5", "Entering foo=2", "Exiting foo=2"),
Seq("Entering foo=2", "Exiting foo=2", "Entering foo=5", "Exiting foo=5")
)
println(s"Final state = $journal")
assert(validAnswers(journal))
def apply[U](lock: Any)(f: => U) = locks(lock.hashCode().abs % size).synchronized(f)
至:
def apply[U](lock: Any) = locks(lock.hashCode().abs % size).synchronized _
测试失败。
预期:
Entering foo=5
Exiting foo=5
Entering foo=2
Exiting foo=2
或
Entering foo=2
Exiting foo=2
Entering foo=5
Exiting foo=5
实际:
Entering foo=5
Entering foo=2
Exiting foo=2
Exiting foo=5
上述两段代码应该相同,但测试(即
lock(id)
总是同时输入相同的
id
)代码的第二种风格(部分应用)。为什么?