Skip to content

Commit

Permalink
Remove method synchronization in ConnectionPoolSupport #531
Browse files Browse the repository at this point in the history
borrowObject and returnObject methods are no longer synchronized to use the pool synchronization and to solve a deadlock in which the pool is exhausted, a new connection is requested, and a connection cannot be returned to the pool because of instance-level locking.

Previously, if a client got stuck in the borrowObject method (pool exhausted, long connect duration) and a concurrent thread tried to return the connection via returnObject, the second thread was blocked by the first thread. Synchronized methods use the object instance to create a synchronization lock. The instance was locked by the first thread and so the second thread had to wait (an indefinite time) for a release, that would likely never happen.
  • Loading branch information
mp911de committed Apr 30, 2017
1 parent e8eaff1 commit ba6c6a1
Showing 1 changed file with 10 additions and 10 deletions.
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
/*
* Copyright 2011-2016 the original author or authors.
* Copyright 2011-2017 the original author or authors.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
Expand Down Expand Up @@ -58,18 +58,18 @@
* the blocking command is resolved.
*
* <h2>Example usage</h2>
*
*
* <pre>
* // application initialization
* RedisClusterClient clusterClient = RedisClusterClient.create(RedisURI.create(host, port));
* GenericObjectPool&lt;StatefulRedisClusterConnection&lt;String, String&gt;&gt; pool = ConnectionPoolSupport
* .createGenericObjectPool(() -&gt; clusterClient.connect(), new GenericObjectPoolConfig());
*
* GenericObjectPool&lt;StatefulRedisClusterConnection&lt;String, String&gt;&gt; pool = ConnectionPoolSupport.createGenericObjectPool(
* () -&gt; clusterClient.connect(), new GenericObjectPoolConfig());
*
* // executing work
* try (StatefulRedisClusterConnection&lt;String, String&gt; connection = pool.borrowObject()) {
* // perform some work
* }
*
*
* // terminating
* pool.close();
* clusterClient.shutdown();
Expand Down Expand Up @@ -120,12 +120,12 @@ private ConnectionPoolSupport() {
GenericObjectPool<T> pool = new GenericObjectPool<T>(new RedisPooledObjectFactory<>(connectionSupplier), config) {

@Override
public synchronized T borrowObject() throws Exception {
public T borrowObject() throws Exception {
return wrapConnections ? wrapConnection(super.borrowObject(), this) : super.borrowObject();
}

@Override
public synchronized void returnObject(T obj) {
public void returnObject(T obj) {

if (wrapConnections && obj instanceof HasTargetConnection) {
super.returnObject((T) ((HasTargetConnection) obj).getTargetConnection());
Expand Down Expand Up @@ -173,12 +173,12 @@ public synchronized void returnObject(T obj) {

SoftReferenceObjectPool<T> pool = new SoftReferenceObjectPool<T>(new RedisPooledObjectFactory<>(connectionSupplier)) {
@Override
public synchronized T borrowObject() throws Exception {
public T borrowObject() throws Exception {
return wrapConnections ? wrapConnection(super.borrowObject(), this) : super.borrowObject();
}

@Override
public synchronized void returnObject(T obj) throws Exception {
public void returnObject(T obj) throws Exception {

if (wrapConnections && obj instanceof HasTargetConnection) {
super.returnObject((T) ((HasTargetConnection) obj).getTargetConnection());
Expand Down

0 comments on commit ba6c6a1

Please sign in to comment.