一、Redis(Jedis)的RedisPool的使用

RedisConfig.java配置文件

package main.java.com.smart.dao.redis;

public class RedisConfig {

    public static int MAX_IDLE = 50;//最大空闲数,数据库连接的最大空闲时间。超过空闲时间,数据库连接将被标记为不可用,然后被释放。设为0表示无限制。

    public static int MIN_IDLE = 5;

    public static int MAX_ACTIVE = 5;//连接池的最大数据库连接数。设为0表示无限制。

    public static int TIME_OUT = 1000;//最大建立连接等待时间。如果超过此时间将接到异常。设为-1表示无限制。

    public static int RETRY_NUM = 6;//重新连接次数
}

JedisApi.java

package main.java.com.smart.dao.redis;

import java.util.HashMap;
import java.util.Map;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.mysql.jdbc.log.Log;

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

/** 
 * 在不同的线程中使用相同的Jedis实例会发生奇怪的错误。但是创建太多的实现也不好因为这意味着会建立很多sokcet连接, 
 * 也会导致奇怪的错误发生。单一Jedis实例不是线程安全的。为了避免这些问题,可以使用JedisPool, 
 * JedisPool是一个线程安全的网络连接池。可以用JedisPool创建一些可靠Jedis实例,可以从池中拿到Jedis的实例。 
 * 这种方式可以解决那些问题并且会实现高效的性能 
 * 思路:1)创建JedisPool对象pool  2)通过pool.getResource获取一个Jedis实例 3)将所有的jedisPool保存在一张Map中 
 */

public class JedisApi {
    protected static final Logger LOG = LoggerFactory.getLogger(JedisApi.class);

    private volatile static JedisApi jedisApi;//可见性修饰,保证对线程之间具有可见性

    /**
     * 保存多个连接源
     */
    private static Map<String, JedisPool> poolMap = new HashMap<String, JedisPool>();

    /**
     * jedisPool
     * 
     * @param ip
     * @param port
     * @return
     */
    private static JedisPool getPool(String ip, int port){

        try {
            String key = ip + ":" + port;
            JedisPool pool = null;
            if(!poolMap.containsKey(key)) {
                System.out.println("触发poolMap模块");
                JedisPoolConfig config = new JedisPoolConfig();//需要依赖common-pool-ver.jar
                config.setMaxIdle(RedisConfig.MAX_IDLE);
                config.setMinIdle(RedisConfig.MIN_IDLE);
                config.setMaxTotal(RedisConfig.MAX_ACTIVE);
                //在获取连接的时候检查有效性,默认为false
                config.setTestOnBorrow(true);
                //在空闲时检查有效性,默认为false
                config.setTestOnReturn(true);
                //连接池耗尽的时候,是否阻塞,false会抛出异常,true阻塞直到超时,会抛出超时异常,默认为true
                config.setBlockWhenExhausted(true);
                pool = new JedisPool(config, ip, port, RedisConfig.TIME_OUT);
                System.out.println("key:"+key+" pool:"+pool.toString());
                poolMap.put(key, pool);
            } else {
                pool = poolMap.get(key);
            }
            return pool;
        } catch (Exception e) {
            LOG.error("Init jedis pool failed!"+e.getMessage(),e);
        }
        return null;
    }

    /**
     * 线程安全单例模式
     * @return
     */
    public static JedisApi getRedisApi() {

        if(jedisApi == null){
            synchronized (JedisApi.class) {
                if(jedisApi == null){
                    jedisApi = new JedisApi();
                }
            }
        }
        return jedisApi;
    }

    /**
     * 获取一个jedis实例
     * 
     * @param ip
     * @param port
     * @return
     */
    public Jedis getRedis(String ip, int port) {
        Jedis jedis = null;
        int count = 0;
        while(jedis == null && count <= RedisConfig.RETRY_NUM) {
            try {
                jedis = getPool(ip, port).getResource();
                LOG.info(ip+":"+port+"JedisPool is status:"+jedis.ping(),jedis);
            } catch (Exception e) {
                LOG.error("Get redis failed!" + e.getMessage(),e);
                count++;
            }
        }
        return jedis;
    }

    /**
     * 释放jedis到jedisPool中
     * @param jedis
     */
    public void closeRedis(Jedis jedis) {

        if(jedis != null) {
            try {
                jedis.close();
            } catch (Exception e) {
                LOG.error("Close jedis failed!" + e.getMessage(), e);
            }
        }
    }

    public static Map<String, JedisPool> getPoolMap() {
        return poolMap;
    }       
}

JedisDao.java

package main.java.com.smart.dao.redis;

import java.util.Map;

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

public class JedisDao {

    //单例模式,每台服务器只能有一个JedisPool对象,可多个Jedis对象
    static JedisApi jedisApi = JedisApi.getRedisApi();
    //同一JedisPool池的不同对象(此处应该通过配置引入对象)
    static Jedis jpool = jedisApi.getRedis("localhost", 6379);

    /**
     * 返回JedisPool列表,查看有多少服务器
     * @return
     */
    public Map<String, JedisPool> printJedisPool(){
        Map<String, JedisPool> poolMap = JedisApi.getPoolMap();
        return poolMap;
    }

    public static void main(String[] args) {
        JedisDao jd = new JedisDao();
        jpool.set("zoo", "cats");
        System.out.println(jpool.get("zoo"));
        for(String key : jd.printJedisPool().keySet()) {
            System.out.println("key="+key);
        }
        jedisApi.closeRedis(jpool);//
    }
}

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
使用JedisPool来管理Redis连接池,可以按照以下步骤操作: 1. 首先,确保你已经将Jedis库添加到你的项目中。你可以在Maven或Gradle的配置文件中添加以下依赖: ```xml <dependency> <groupId>redis.clients</groupId> <artifactId>jedis</artifactId> <version>版本号</version> </dependency> ``` 2. 在代码中导入JedisJedisPool相关的类: ```java import redis.clients.jedis.Jedis; import redis.clients.jedis.JedisPool; import redis.clients.jedis.JedisPoolConfig; ``` 3. 创建JedisPoolConfig对象,并设置连接池的相关属性(如最大连接数、最大空闲连接数等): ```java JedisPoolConfig jedisPoolConfig = new JedisPoolConfig(); jedisPoolConfig.setMaxTotal(100); // 设置最大连接数 jedisPoolConfig.setMaxIdle(10); // 设置最大空闲连接数 // 可以根据需要设置其他属性,比如最小空闲连接数、连接超时时间等 ``` 4. 创建JedisPool对象,传入Redis服务器的主机名、端口号、连接超时时间和密码(如果有的话): ```java JedisPool jedisPool = new JedisPool(jedisPoolConfig, "localhost", 6379, 10000, "password"); ``` 注意:如果Redis服务器没有设置密码,可以省略密码参数。 5. 从连接池中获取Jedis对象,并使用Jedis对象进行Redis操作: ```java try (Jedis jedis = jedisPool.getResource()) { // 执行Redis操作,如存储键值对 jedis.set("key", "value"); String value = jedis.get("key"); System.out.println(value); } catch (Exception e) { // 异常处理 } ``` 6. 当不再需要使用Redis连接时,记得将Jedis对象还回连接池: ```java jedis.close(); ``` 这就是使用JedisPool来管理Redis连接池的基本步骤。通过合理配置连接池的属性,可以提高Redis操作的性能和效率。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值