美文网首页极乐科技
Redis学习记录(二)--使用Jedis连接

Redis学习记录(二)--使用Jedis连接

作者: 此博废弃_更新在个人博客 | 来源:发表于2016-12-08 21:49 被阅读842次

    Redis学习记录(二)--使用Jedis连接

    标签(空格分隔): javaWEB


    在Java环境中连接Redis首选Jedis,因为Jedis封装的特别好,所以连接对于开发者也就变得很简单了,本文主要讲述如何写出优雅的Jedis连接代码.


    1.设计模式分析

    对于Jedis,有一个连接池JedisPool,这个连接池管理着程序中的jedis示例,对于单机部署,每一次使用jedis都是去池中取出来连接池,然后再使用他获取我们想要的结果.是不是有点感觉像数据库连接池?那么一般形式代码如下:

        public String get(final String key) {
            Jedis jedis = jedisPool.getResource();
            String result = jedis.get(key);
            jedis.close();
            return result;
        }
    

    然而这样写方法多的话,每一个都要如此重复获取,然后执行,然后关闭,返回结果,一套流程,就像模板一样,那么设计模式就可以考虑模板模式.

    模板设计模式主要是将通用的逻辑都抽离出来,不通用的逻辑根据实现类的具体策略而执行不同的策略.因为JedisPool比较类似数据库连接池,因此我们可以参考Spring里的JdbcTemplate,里面有如下代码:

    public <T> T execute(ConnectionCallback<T> action) throws DataAccessException {
            Assert.notNull(action, "Callback object must not be null");
            //获取连接
            Connection con = DataSourceUtils.getConnection(getDataSource());
            try {
                Connection conToUse = con;
                if (this.nativeJdbcExtractor != null) {
                    conToUse = this.nativeJdbcExtractor.getNativeConnection(con);
                }
                else {
                    // Create close-suppressing Connection proxy, also preparing returned Statements.
                    conToUse = createConnectionProxy(con);
                }
                //执行策略
                return action.doInConnection(conToUse);
            }
            catch (SQLException ex) {
                DataSourceUtils.releaseConnection(con, getDataSource());
                con = null;
                throw getExceptionTranslator().translate("ConnectionCallback", getSql(action), ex);
            }
            finally {
                //关闭连接
                DataSourceUtils.releaseConnection(con, getDataSource());
            }
        }
    

    分析:
    代码分为三部分,获取连接,执行策略,关闭释放连接.除了策略,其他都是模板,而策略是一个函数式接口,如下:

    public interface ConnectionCallback<T> {
        T doInConnection(Connection con) throws SQLException, DataAccessException;
    }
    
    

    那么具体策略就是实现该接口的实现类里面的策略了.这样就很好地做到模板和策略分离,还有一个好处就是函数式接口可以很好地配合java8语法,写出相当优雅的代码.


    2.实现

    2.1 函数式接口实现

    首先参照Spring,实现自己的函数式接口:
    比上面多了一个泛型E代表传入参数,这样做的话,就会使得该接口更加具有通用性.

    /**
     * 模仿Spring ConnectionCallback写的模板接口
     * T为操作返回值  E为参数类型
     * 配合java8使用,效果更佳
     * @author Niu Li
     * @date 2016/12/8
     */
    public interface WorkCallback<T,E> {
        /**
         * 具体执行策略
         * @param e 传入参数
         * @return 结果
         */
        T doWorkCallback(E e);
    }
    

    2.2 实现单机版Jedis

    实现思路,首先在JedisClientSingle写一个私有的excute方法,用于实现函数式接口,定义不同策略,代码具体如下:

    /**
     * 单机版jedis(配置建议都放在依赖注入配置中)
     * @author Niu Li
     * @date 2016/12/8
     */
    public class JedisClientSingle{
        /**
         * 连接池,建议使用其他工具注入进来
         */
        private JedisPool jedisPool;
    
    
        public String get(final String key) {
            return excute(new WorkCallback<String, Jedis>() {
                public String doWorkCallback(Jedis jedis) {
                    return jedis.get(key);
                }
            });
        }
    
        public String set(final String key, final String value) {
            return excute(new WorkCallback<String, Jedis>() {
                public String doWorkCallback(Jedis jedis) {
                    return jedis.set(key,value);
                }
            });
        }
        /**
         * 模板方法,很适合提取公共操作
         * @param workCallback 处理函数
         * @param <T> 返回类型
         * @return 结果
         */
        private <T> T excute(WorkCallback<T, Jedis> workCallback) {
            Jedis jedis=null;
            try {
                jedis = jedisPool.getResource();
                return workCallback.doWorkCallback(jedis);
            } catch (Exception e) {
                e.printStackTrace();
            }finally {
                if (null !=jedis){
                    jedis.close();
                }
            }
            return null;
        }
    
    
    }
    

    对于jedisPool的注入的话,Spring中可以如下配置

        <!-- 配置redis客户端单机版 -->
        <bean id="jedispoolConfig" class="redis.clients.jedis.JedisPoolConfig">
            <property name="maxTotal" value="${redis.maxactive}"/>
            <property name="maxIdle" value="${redis.maxidle}"/>
            <property name="minIdle" value="${redis.minidle}"/>
            <property name="maxWaitMillis" value="${redis.timeout}"/>
        </bean>
        <bean id="jedisPool" class="redis.clients.jedis.JedisPool" destroy-method="destroy"  depends-on="jedispoolConfig">
            <constructor-arg name="poolConfig" ref="jedispoolConfig"/>
            <constructor-arg name="host" value="${redis.ip}"/>
            <constructor-arg name="port" value="${redis.port}"/>
        </bean>
    

    2.3实现集群版

    集群版就简单很多了,只能说Jedis封装的太好了

    /**
     * jedis集群
     * @author Niu Li
     * @date 2016/12/8
     */
    public class JedisClientCluster {
        /**
         * 集群控制,建议使用其他工具注入进来
         */
        private JedisCluster jedisCluster;
    
        /**
         * 其他的直接使用jedisCluster的方法即可
         */
        public String get(final String key) {
            return jedisCluster.get(key);
        }
    }
    

    对于jedisCluster的注入,Spring配置如下:

     <!-- 配置redis客户端集群版 -->
        <!--<bean id="jedisCluster" class="redis.clients.jedis.JedisCluster">
           <constructor-arg>
               <set>
                   <bean class="redis.clients.jedis.HostAndPort">
                       <constructor-arg name="host" value="192.168.1.101"/>
                       <constructor-arg name="port" value="1001"/>
                   </bean>
                   <bean class="redis.clients.jedis.HostAndPort">
                       <constructor-arg name="host" value="192.168.1.101"/>
                       <constructor-arg name="port" value="1002"/>
                   </bean>
                   <bean class="redis.clients.jedis.HostAndPort">
                       <constructor-arg name="host" value="192.168.1.101"/>
                       <constructor-arg name="port" value="1003"/>
                   </bean>
               </set>
           </constructor-arg>
       </bean>
    

    这种实现是不是很优雅?

    参考代码:

    https://github.com/nl101531/JavaWEB 下Util-Demo

    参考链接:

    本文从该博客学习总结而得

    https://muyinchen.github.io

    相关文章

      网友评论

        本文标题:Redis学习记录(二)--使用Jedis连接

        本文链接:https://www.haomeiwen.com/subject/naosmttx.html