1. 程式人生 > >jedispool 連 redis 高併發卡死

jedispool 連 redis 高併發卡死

java端在使用jedispool 連線redis的時候,在高併發的時候經常卡死,或報連線異常,JedisConnectionException,或者getResource 異常等各種問題

在使用jedispool 的時候一定要注意兩點

1。 在獲取 jedisPool和jedis的時候加上執行緒同步,保證不要建立過多的jedispool 和 jedis

2。 用完Jedis例項後需要返還給JedisPool

整理了一下redis工具類,通過大量測試和高併發測試的

import org.apache.log4j.Logger;import redis.clients.jedis.Jedis;import
redis.clients.jedis.JedisPool;import redis.clients.jedis.JedisPoolConfig;/** * Redis 工具類 * @author caspar * https://blog.csdn.net/tuposky */public class RedisUtil { protected static Logger logger = Logger.getLogger(RedisUtil.class); //Redis伺服器IP private static String ADDR_ARRAY = FileUtil.getPropertyValue
("/properties/redis.properties", "server"); //Redis的埠號 private static int PORT = FileUtil.getPropertyValueInt("/properties/redis.properties", "port"); //訪問密碼// private static String AUTH = FileUtil.getPropertyValue("/properties/redis.properties", "auth"); //可用連線例項的最大數目,預設值為8; //如果賦值為-1,則表示不限制;如果pool已經分配了maxActive個jedis例項,則此時pool的狀態為exhausted(耗盡)。
private static int MAX_ACTIVE = FileUtil.getPropertyValueInt("/properties/redis.properties", "max_active");; //控制一個pool最多有多少個狀態為idle(空閒的)的jedis例項,預設值也是8。 private static int MAX_IDLE = FileUtil.getPropertyValueInt("/properties/redis.properties", "max_idle");; //等待可用連線的最大時間,單位毫秒,預設值為-1,表示永不超時。如果超過等待時間,則直接丟擲JedisConnectionException; private static int MAX_WAIT = FileUtil.getPropertyValueInt("/properties/redis.properties", "max_wait");; //超時時間 private static int TIMEOUT = FileUtil.getPropertyValueInt("/properties/redis.properties", "timeout");; //在borrow一個jedis例項時,是否提前進行validate操作;如果為true,則得到的jedis例項均是可用的; private static boolean TEST_ON_BORROW = FileUtil.getPropertyValueBoolean("/properties/redis.properties", "test_on_borrow");; private static JedisPool jedisPool = null; /** * redis過期時間,以秒為單位 */ public final static int EXRP_HOUR = 60*60; //一小時 public final static int EXRP_DAY = 60*60*24; //一天 public final static int EXRP_MONTH = 60*60*24*30; //一個月 /** * 初始化Redis連線池 */ private static void initialPool(){ try { JedisPoolConfig config = new JedisPoolConfig(); config.setMaxTotal(MAX_ACTIVE); config.setMaxIdle(MAX_IDLE); config.setMaxWaitMillis(MAX_WAIT); config.setTestOnBorrow(TEST_ON_BORROW); jedisPool = new JedisPool(config, ADDR_ARRAY.split(",")[0], PORT, TIMEOUT); } catch (Exception e) { logger.error("First create JedisPool error : "+e); try{ //如果第一個IP異常,則訪問第二個IP JedisPoolConfig config = new JedisPoolConfig(); config.setMaxTotal(MAX_ACTIVE); config.setMaxIdle(MAX_IDLE); config.setMaxWaitMillis(MAX_WAIT); config.setTestOnBorrow(TEST_ON_BORROW); jedisPool = new JedisPool(config, ADDR_ARRAY.split(",")[1], PORT, TIMEOUT); }catch(Exception e2){ logger.error("Second create JedisPool error : "+e2); } } } /** * 在多執行緒環境同步初始化 */ private static synchronized void poolInit() { if (jedisPool == null) { initialPool(); } } /** * 同步獲取Jedis例項 * @return Jedis */ public synchronized static Jedis getJedis() { if (jedisPool == null) { poolInit(); } Jedis jedis = null; try { if (jedisPool != null) { jedis = jedisPool.getResource(); } } catch (Exception e) { logger.error("Get jedis error : "+e); }finally{ returnResource(jedis); } return jedis; } /** * 釋放jedis資源 * @paramjedis */ public static void returnResource(final Jedis jedis) { if (jedis != null && jedisPool !=null) { jedisPool.returnResource(jedis); } } /** * 設定 String * @paramkey * @paramvalue */ public static void setString(String key ,String value){ try { value = StringUtil.isEmpty(value) ? "" : value; getJedis().set(key,value); } catch (Exception e) { logger.error("Set key error : "+e); } } /** * 設定 過期時間 * @paramkey * @paramseconds 以秒為單位 * @paramvalue */ public static void setString(String key ,int seconds,String value){ try { value = StringUtil.isEmpty(value) ? "" : value; getJedis().setex(key, seconds, value); } catch (Exception e) { logger.error("Set keyex error : "+e); } } /** * 獲取String值 * @paramkey * @return value */ public static String getString(String key){ if(getJedis() == null || !getJedis().exists(key)){ return null; } return getJedis().get(key); }}

其中 jedis官方現在最高版本為2.9.0 而jedis.close()取代pool.returnResource(Jedis)方法將3.0版本開始,所以暫時只能使用後者。

//官方重寫了close。 try { jedis = pool.getResource(); } finally { if (jedis != null) { jedis.close(); } }