源码梳理——Jedis连接池的创建过程

一、Jedis介绍

Jedis 是 Redis 官方首选的 Java 客户端开发包。借助该开发包我们可以通过创建单个redis客户端实例来访问redis数据库,同时它也提供了连接池的实现。Jedis开发包中连接池是利用apachec开发的对象池框架commons-pool实现的,所以要想使用Jedis的连接池功能必须要导入commons-pool包。

二、一个使用连接池的小例子(Jedis-2.1.0+commons-pool-1.6)

package com.fsun.framework.cache.redis;

import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPool;
import redis.clients.jedis.JedisPoolConfig;

public class RedisManager {
	//Redis服务器IP
	private static String ADDR = "";
	    
	//Redis的端口号
	private static int PORT = 6379;
	 
	//访问密码
	private static String AUTH = "admin";
	     
	//可用连接实例的最大数目,默认值为8;
	//如果赋值为-1,则表示不限制;如果pool已经分配了maxActive个jedis实例,则此时pool的状态为exhausted(耗尽)。
	private static int MAX_ACTIVE = 1024;
	  
	//控制一个pool最多有多少个状态为idle(空闲的)的jedis实例,默认值也是8。
	private static int MAX_IDLE = 200;
	    
	//等待可用连接的最大时间,单位毫秒,默认值为-1,表示永不超时。如果超过等待时间,则直接抛出JedisConnectionException;
	private static int MAX_WAIT = 10000;
	   
	private static int TIMEOUT = 10000;
	    
	//在borrow一个jedis实例时,是否提前进行validate操作;如果为true,则得到的jedis实例均是可用的;
	private static boolean TEST_ON_BORROW = true;
	 
	private static JedisPool jedisPool;
	   
	/**
	* 初始化Redis连接池
	*/
	static {
	         try {
	            JedisPoolConfig config = new JedisPoolConfig();
	           config.setMaxActive(MAX_ACTIVE);
	           config.setMaxIdle(MAX_IDLE);
	           config.setMaxWait(MAX_WAIT);
	           config.setMaxWait(MAX_WAIT);
	           config.setTestOnBorrow(TEST_ON_BORROW);
	            jedisPool = new JedisPool(config, ADDR, PORT, TIMEOUT, AUTH);
	         } catch (Exception e) {
	             e.printStackTrace();
	         }
	     }
	    
	    /**
	     * 获取Jedis实例
	     * @return
	      */
	     public static Jedis getJedis() {
	         try {
	             if (jedisPool != null) {
	                 Jedis resource = jedisPool.getResource();
	                 return resource;
	             } else {
	                 return null;
	             }
	        } catch (Exception e) {
	             e.printStackTrace();
	            return null;
	         }
	     }
	     
	     /**
	      * 释放jedis资源
	      * @param jedis
	      */
	     public static void returnResource(final Jedis jedis) {
	         if (jedis != null) {
	             jedisPool.returnResource(jedis);
	         }
	     }
}

三、连接池创建过程

上面例子连接池初始化代码如下;

JedisPoolConfig config = new JedisPoolConfig();
config.setMaxActive(MAX_ACTIVE);
config.setMaxIdle(MAX_IDLE);
config.setMaxWait(MAX_WAIT);
config.setMaxWait(MAX_WAIT);
config.setTestOnBorrow(TEST_ON_BORROW);
jedisPool = new JedisPool(config, ADDR, PORT, TIMEOUT, AUTH);

连接池一些配置信息如池中最大活跃数量、最大空闲数量等等被存储在JedisPoolConfig对象中。JedisPoolConfig继承commons-pool包中的Config类,是一个比较纯碎的javabean,不做过多阐述。

JedisPool继承commons-pool包中的Pool类,其初始化的过程是调用父类Pool类的构造器完成的。

public abstract class Pool<T> {
    private final GenericObjectPool internalPool;

    public Pool(final GenericObjectPool.Config poolConfig,
            PoolableObjectFactory factory) {
        this.internalPool = new GenericObjectPool(factory, poolConfig);
    }

Jedis把JedisFactory传工厂类给了父类的构造器,JedisFactory继承BasePoolableObjectFactory抽象类(BasePoolableObjectFactory实现了PoolableObjectFactory接口),重写了其中的makeObject方法。

 public Object makeObject() throws Exception {
       final Jedis jedis = new Jedis(this.host, this.port, this.timeout);

       jedis.connect();
       if (null != this.password) {
           jedis.auth(this.password);
       }
       if( database != 0 ) {
           jedis.select(database);
       }
            
       return jedis;
 }

makeObject方法提供了创建Jedis实例的功能,commons-pool在初始化池的时候调用我们传入的工厂类makeObject方法来创建要放入缓存池中的对象,这个地方用到了抽象工厂模式这样的设计思想。JedisFactory的makeObject方法做了这几件事:

                                    (1)创建Jedis实例

                                    (2)连接redis服务器

                                    (3)如果redis服务器需要密码则向服务器提供密码

                                    (4)如果指定数据库不是默认的0,则选择指定的数据库

下面是单个Jedis实例的构造很简单,目标就是要创建Client客户端实例保存到成员属性中,Client间接继承了Connection,Jedis的构造仅仅是把从Connection继承而来的三个属性主机地址、端口号和连接超时时间进行赋值:

public Connection(final String host, final int port) {
    super();
    this.host = host;
    this.port = port;
}

Jedis实例创建完之后利用Socket套接字和redis服务器进行连接,这个过程是在Connection类中进行的,建立连接之后将输入流和输出流保存到成员对象中,这样以后就可以向redis服务器发送命令,代码如下:

 public void connect() {
        if (!isConnected()) {
            try {
                socket = new Socket();
                //->@wjw_add
                socket.setReuseAddress(true);
                socket.setKeepAlive(true);  //Will monitor the TCP connection is valid
                socket.setTcpNoDelay(true);  //Socket buffer Whetherclosed, to ensure timely deliver//y of data
                socket.setSoLinger(true,0);  //Control calls close () method, the underlying socket //is closed immediately
                //<-@wjw_add

                socket.connect(new InetSocketAddress(host, port), timeout);
                socket.setSoTimeout(timeout);
                outputStream = new RedisOutputStream(socket.getOutputStream());
                inputStream = new RedisInputStream(socket.getInputStream());
            } catch (IOException ex) {
                throw new JedisConnectionException(ex);
            }
        }
    }

这样redis连接池就完成了初始化的工作。




转载于:https://my.oschina.net/catcloud/blog/524465

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值