Prevent usage of stale cache data.

git-svn-id: https://svn.apache.org/repos/asf/activemq/trunk@1300723 13f79535-47bb-0310-9956-ffa450edef68
This commit is contained in:
Timothy A. Bish 2012-03-14 21:15:20 +00:00
parent f5b69f8783
commit 0bbb73529d
1 changed files with 24 additions and 8 deletions

View File

@ -17,18 +17,19 @@
package org.apache.kahadb.index;
import java.io.IOException;
import java.lang.ref.WeakReference;
import java.util.Iterator;
import java.util.Map;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.concurrent.atomic.AtomicLong;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.apache.kahadb.index.ListNode.ListIterator;
import org.apache.kahadb.page.Page;
import org.apache.kahadb.page.PageFile;
import org.apache.kahadb.page.Transaction;
import org.apache.kahadb.util.Marshaller;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
public class ListIndex<Key,Value> implements Index<Key,Value> {
@ -120,6 +121,7 @@ public class ListIndex<Key,Value> implements Index<Key,Value> {
private ListNode<Key, Value> lastGetNodeCache = null;
private Map.Entry<Key, Value> lastGetEntryCache = null;
private WeakReference<Transaction> lastCacheTxSrc = new WeakReference<Transaction>(null);
@SuppressWarnings({ "rawtypes", "unchecked" })
synchronized public Value get(Transaction tx, Key key) throws IOException {
@ -129,6 +131,7 @@ public class ListIndex<Key,Value> implements Index<Key,Value> {
if (key.equals(candidate.getKey())) {
this.lastGetNodeCache = ((ListIterator) iterator).getCurrent();
this.lastGetEntryCache = candidate;
this.lastCacheTxSrc = new WeakReference<Transaction>(tx);
return candidate.getValue();
}
}
@ -146,7 +149,7 @@ public class ListIndex<Key,Value> implements Index<Key,Value> {
Value oldValue = null;
if (lastGetNodeCache != null) {
if (lastGetNodeCache != null && tx.equals(lastCacheTxSrc.get())) {
if(lastGetEntryCache.getKey().equals(key)) {
oldValue = lastGetEntryCache.setValue(value);
@ -166,6 +169,8 @@ public class ListIndex<Key,Value> implements Index<Key,Value> {
return oldValue;
}
}
} else {
flushCache();
}
// Not found because the cache wasn't set or its not at the end of the list so we
@ -207,7 +212,7 @@ public class ListIndex<Key,Value> implements Index<Key,Value> {
return null;
}
if (lastGetNodeCache != null) {
if (lastGetNodeCache != null && tx.equals(lastCacheTxSrc.get())) {
// This searches from the last location of a call to get for the element to remove
// all the way to the end of the ListIndex.
@ -216,9 +221,12 @@ public class ListIndex<Key,Value> implements Index<Key,Value> {
Map.Entry<Key, Value> entry = iterator.next();
if (entry.getKey().equals(key)) {
iterator.remove();
flushCache();
return entry.getValue();
}
}
} else {
flushCache();
}
// Not found because the cache wasn't set or its not at the end of the list so we
@ -229,6 +237,7 @@ public class ListIndex<Key,Value> implements Index<Key,Value> {
Map.Entry<Key, Value> entry = iterator.next();
if (entry.getKey().equals(key)) {
iterator.remove();
flushCache();
return entry.getValue();
}
}
@ -238,6 +247,7 @@ public class ListIndex<Key,Value> implements Index<Key,Value> {
public void onRemove() {
size.decrementAndGet();
flushCache();
}
public boolean isTransient() {
@ -251,6 +261,7 @@ public class ListIndex<Key,Value> implements Index<Key,Value> {
// break up the transaction
tx.commit();
}
flushCache();
size.set(0);
}
@ -286,10 +297,14 @@ public class ListIndex<Key,Value> implements Index<Key,Value> {
ListNode<Key,Value> loadNode(Transaction tx, long pageId) throws IOException {
Page<ListNode<Key,Value>> page = tx.load(pageId, marshaller);
ListNode<Key, Value> node = page.get();
node.setPage(page);
node.setContainingList(this);
return node;
try {
ListNode<Key, Value> node = page.get();
node.setPage(page);
node.setContainingList(this);
return node;
} catch (ClassCastException e) {
throw e;
}
}
ListNode<Key,Value> createNode(Page<ListNode<Key,Value>> page) throws IOException {
@ -354,5 +369,6 @@ public class ListIndex<Key,Value> implements Index<Key,Value> {
private void flushCache() {
this.lastGetEntryCache = null;
this.lastGetNodeCache = null;
this.lastCacheTxSrc.clear();
}
}