1. Jedis不是线程安全的, 故不应该在多线程环境中共用一个Jedis实例。但是, 也应该避免直接创建多个Jedis实例, 因为这种做法会导致创建过多的socket连接, 性能不高。
2. 要保证线程安全且获得较好的性能, 可以使用JedisPool。JedisPool是一个连接池, 既可以保证线程安全, 又可以保证了较高的效率。
3. 可以声明一个全局的JedisPool变量来保存JedisPool对象的引用, 然后在其他地方使用。要知道, JedisPool是一个线程安全的连接池。
JedisPool jedisPool = new JedisPool("127.0.0.1");
4. 通过jedisPool.getResource()获得一个Jedis实例
Jedis jedis = jedisPool.getResource();
5. 用Jedis类的close方法, 将这个Jedis实例归还给JedisPool
jedis.close();
6. Jedis类的close方法
6.1. Jedis类的close方法代码
public void close() {
if (dataSource != null) {
if (client.isBroken())
this.dataSource.returnBrokenResource(this);
else
this.dataSource.returnResource(this);
} else {
client.close();
}
}
6.2. 可以看到, 如果是从JedisPool取得的Jedis实例(Jedis的dataSource成员不为空, 即指向一个JedisPool), 会进行相应的归还给JedisPool的操作, 如果是单独生成的一个Jedis实例(Jedis的dataSource成员为空), 则会直接断开与Redis服务器的连接。
7. 例子
7.1. 新建一个名为JedisPool的Java项目, 拷入相关jar包, 并添加到build path
7.2. 代码
package com.redis.pool;
import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPool;
/**
* Jedis连接池
*/
public class UseJedisPool {
public static void main(String[] args) {
JedisPool jedisPool = new JedisPool("192.168.25.138");
// 获取一个客户端实例
Jedis jedis = jedisPool.getResource();
jedis.auth("lyw123456");
jedis.set("name", "zhangsan");
System.out.println(jedis.get("name"));
// 把客户端归还给连接池
jedis.close();
// 释放连接池
jedisPool.close();
jedisPool.destroy();
}
}
7.3. 运行结果