2011-11-07 9 views
6

Korzystanie z ehCache 2.4.4, wydaje mi się, że wpadłem w impas w obiekcie segmentu ehCache. Z innych rejestrów wiem, że "oczekujący wątek", 1694, ostatni uruchomił się na 9 godzin przed wygenerowaniem tego śladu stosu. W międzyczasie, 1696 poszło i zrobiło dużo innych prac, więc ten zamek jest zdecydowanie trzymany błędnie.Jak mogę obejść ten pozorny impas EhCache?

Jestem przekonany, że bezpośrednio nie blokuję bezpośrednio żadnych instancji Segmentów, więc zakładam, że jest to jakiś problem wewnętrzny biblioteki. Jakieś pomysły?

"Model Executor - 1696" Id=1696 in TIMED_WAITING on lock=java.u[email protected]92eb1ed 
at sun.misc.Unsafe.park(Native Method) 
at java.util.concurrent.locks.LockSupport.parkNanos(Unknown Source) 
at java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject.awaitNanos(Unknown Source) 
at java.util.concurrent.PriorityBlockingQueue.poll(Unknown Source) 
at com.rtrms.application.modeling.local.BlockingTaskList.takeTask(BlockingTaskList.java:20) 
at com.rtrms.application.modeling.local.ModelExecutor.executeNextTask(ModelExecutor.java:71) 
at com.rtrms.application.modeling.local.ModelExecutor.run(ModelExecutor.java:46) 

Locked synchronizers: count = 1 
    - [email protected]3d767f 

"Model Executor - 1694" Id=1694 in WAITING on loc[email protected]4a3d767f 
owned by Model Executor - 1696 Id=1696 
at sun.misc.Unsafe.park(Native Method) 
at java.util.concurrent.locks.LockSupport.park(Unknown Source) 
at java.util.concurrent.locks.AbstractQueuedSynchronizer.parkAndCheckInterrupt(Unknown Source) 
at java.util.concurrent.locks.AbstractQueuedSynchronizer.doAcquireShared(Unknown Source) 
at java.util.concurrent.locks.AbstractQueuedSynchronizer.acquireShared(Unknown Source) 
at java.util.concurrent.locks.ReentrantReadWriteLock$ReadLock.lock(Unknown Source) 
at net.sf.ehcache.store.compound.Segment.unretrievedGet(Segment.java:248) 
at net.sf.ehcache.store.compound.CompoundStore.unretrievedGet(CompoundStore.java:191) 
at net.sf.ehcache.store.compound.impl.DiskPersistentStore.containsKeyInMemory(DiskPersistentStore.java:72) 
at net.sf.ehcache.Cache.searchInStoreWithStats(Cache.java:1884) 
at net.sf.ehcache.Cache.get(Cache.java:1549) 
at com.rtrms.amoeba.cache.DistributedModeledSecurities.get(DistributedModeledSecurities.java:57) 
at com.rtrms.amoeba.modeling.AssertPersistedModeledSecurities.get(AssertPersistedModeledSecurities.java:44) 
at com.rtrms.application.modeling.tasks.ExpandableModelingTask.getNextUnexecutedTask(ExpandableModelingTask.java:35) 
at com.rtrms.application.modeling.local.BlockingTaskList.takeTask(BlockingTaskList.java:36) 
at com.rtrms.application.modeling.local.ModelExecutor.executeNextTask(ModelExecutor.java:71) 
at com.rtrms.application.modeling.local.ModelExecutor.run(ModelExecutor.java:46) 

Locked synchronizers: count = 0 
+0

Okazuje się, że to pytanie jest nieprawidłowe. Zinterpretowałem tę dokumentację (http://ehcache.org/documentation/user-guide/jta#performance) jako wskazującą, że jawne blokady nie używają blokowania opartego na segmentach, ale okazuje się, że to nieprawda. Ten impas był spowodowany przez mój kod, było wydanie blokady, które nie było w bloku finally(). – sharakan

+3

dlaczego nie odpowiesz na własne pytanie, jeśli już to wymyśliłeś, więc nie będzie ono obecne w pytaniach bez odpowiedzi? – xmoex

Odpowiedz

1

Okazuje się, że połączenia takie jak Cache.acquireWriteLockOnKey końcu uzyskania blokady na odcinku wewnętrznym, tak więc oczywiste zakleszczenie spowodowane przez wywołanie .unlock, że nie była w końcu zablokować.

Komentarz redakcyjny: Oznacza to również, że można uzyskać rywalizację, próbując zablokować dwa różne klucze, które znajdowały się w tym samym segmencie, co jest dość niefortunne.

+1

Jeśli tak jest naprawdę, możesz oznaczyć ten post jako odpowiedź. Po prostu kliknij na duży symbol zaznaczenia poniżej licznika głosów po lewej stronie odpowiedzi - w ten sposób wskażesz odpowiedź na pytanie :) –