concurrency - Redis 出队时处于 'parking to wait' 状态的线程

标签 concurrency locking java.util.concurrent jedis apache-commons-pool

我有一个运行多个线程的 tomcat - spring4.2 应用程序。每个线程仅从一个队列中出队,但是分配给一个队列的线程不止一个。

事情开始很好,但是在几个小时/大约 50 万个出队操作之后,我发现线程出队的速度非常慢。

在 jvisualvm 中,我看到橙色的线程,即公园
线程转储如下:

"EMLT_2" - Thread t@64
   java.lang.Thread.State: WAITING
    at sun.misc.Unsafe.park(Native Method)
    - parking to wait for <2cf42d7> (a java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject)
    at java.util.concurrent.locks.LockSupport.park(LockSupport.java:175)
    at java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject.await(AbstractQueuedSynchronizer.java:2039)
    at org.apache.commons.pool2.impl.LinkedBlockingDeque.takeFirst(LinkedBlockingDeque.java:583)
    at org.apache.commons.pool2.impl.GenericObjectPool.borrowObject(GenericObjectPool.java:442)
    at org.apache.commons.pool2.impl.GenericObjectPool.borrowObject(GenericObjectPool.java:363)
    at redis.clients.util.Pool.getResource(Pool.java:48)
    at redis.clients.jedis.JedisPool.getResource(JedisPool.java:86)
    at com.mycomp.sam.processors.SimpleDequeuer.dequeue(SimpleDequeuer.java:25)
    at com.mycomp.sam.processors.EMLT.run(EMLT.java:29)
    at java.lang.Thread.run(Thread.java:745)

   Locked ownable synchronizers:
    - None

"EMLT_1" - Thread t@63
   java.lang.Thread.State: WAITING
    at sun.misc.Unsafe.park(Native Method)
    - parking to wait for <2cf42d7> (a java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject)
    at java.util.concurrent.locks.LockSupport.park(LockSupport.java:175)
    at java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject.await(AbstractQueuedSynchronizer.java:2039)
    at org.apache.commons.pool2.impl.LinkedBlockingDeque.takeFirst(LinkedBlockingDeque.java:583)
    at org.apache.commons.pool2.impl.GenericObjectPool.borrowObject(GenericObjectPool.java:442)
    at org.apache.commons.pool2.impl.GenericObjectPool.borrowObject(GenericObjectPool.java:363)
    at redis.clients.util.Pool.getResource(Pool.java:48)
    at redis.clients.jedis.JedisPool.getResource(JedisPool.java:86)
    at com.mycomp.sam.processors.SimpleDequeuer.dequeue(SimpleDequeuer.java:25)
    at com.mycomp.sam.processors.EMLT.run(EMLT.java:29)
    at java.lang.Thread.run(Thread.java:745)

   Locked ownable synchronizers:
    - None

出队方法是:
public String dequeue(String queue) {
        try (Jedis jedis = jedispool.getResource()) {
            List<String> str = jedis.blpop(10, queue);
            if(str!=null){
                return str.get(1);
            }
            else 
                return null; 
        }
    }

将欣赏输入。重新启动后,该应用程序再次运行良好。
池配置:
<bean id="poolConfig" class="redis.clients.jedis.JedisPoolConfig">
    <property name="maxIdle" value="10" />
    <property name="maxTotal" value="70" />
    <property name="minIdle" value="10" />
</bean>
<bean id="jedispool" class="redis.clients.jedis.JedisPool">
    <constructor-arg name="poolConfig" ref="poolConfig" />
    <constructor-arg name="host" value="${REDIS_HOST}" />
    <constructor-arg name="port" value="6379" />
</bean>

最佳答案

看来您的绝地武士池连接用完了。
您是使用 returnResourceObject 还是 returnResource 返回资源? (我知道两者都已弃用,但它们仍然适用于最新版本的绝地武士)。

请记住,jedis 池与 DBCP/Apache 池并不完全相同。

我有一个类似的问题,当我调用上面的方法时它就结束了。

另一方面,由于 Redis 非常快,也许您应该重新考虑您的架构,并且只有很少的线程调用 blpop(每个队列一个)并将值转发到其他线程,所以如果他们不知道 redis不必。

关于concurrency - Redis 出队时处于 'parking to wait' 状态的线程,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37299408/

相关文章:

java - Java 中的 MCS 锁缓存未更新

java - 使用 JavaFX 将视频循环到 JPanel 时的线程

java - 从 ScheduledExecutorService 中提取异常

c++ - 是否有无锁 vector 实现?

具有列表映射的 Java 并发性

c# - 在 C# 中使用 ThreadPool 的 lock 语句

c++ - 无分支内存管理器?

locking - 计算密码锁距离的有效方法

java - 使用并发类并行处理目录中的文件

java - 为什么在 ReentrantReadWriteLock 中,readLock() 应该在 writeLock().lock() 之前解锁?