【 Redis 】
本文介绍一些Redis基础原理,以及命令操作,最后用Java实现Redis消息队列
1.为什么使用Redis
①性能:对于对于执行耗时特别久,且结果不频繁变动的sql特别适合将结果放入缓存。后面的请求就去缓存中读取,使得请求能够迅速响应
②并发:在大并发情况下,所有请求直接访问数据库,数据库会出现连接异常,这时候需要使用redies做一个缓冲让请求先访问到redis
2、使用redis有什么缺点
①缓存和数据库双写一致性问题
②缓存雪崩问题
③缓存击穿问题:缓存穿透,即黑客故意去请求缓存中不存在的数据,导致所有的请求都怼到数据库上,从而数据库连接异常。
④缓存的并发竞争问题
3、单线程的redis为什么这么快
①纯内存操作
②单线程操作,避免了频繁的上下文切换
③采用了非阻塞I/O多路复用机制(多路复用机制如下图,在同一个线程里面, 通过拨开关的方式,来同时传输多个I/O流, 单个线程通过记录跟踪每一个Sock(I/O流)的状态)
4、redis的数据类型,以及每种数据类型的使用场景
一共五种数据类型
①String
这个其实没啥好说的,最常规的set/get操作,value可以是String也可以是数字。一般做一些复杂的计数功能的缓存。
②hash
这里value存放的是结构化的对象,比较方便的就是操作其中的某个字段。博主在做单点登录的时候,就是用这种数据结构存储用户信息,以cookieId作为key,设置30分钟为缓存过期时间,能很好的模拟出类似session的效果。
③list
使用List的数据结构,可以做简单的消息队列的功能。另外还有一个就是,可以利用lrange命令,做基于redis的分页功能,性能极佳,用户体验好。本人还用一个场景,很合适---取行情信息。就也是个生产者和消费者的场景。LIST可以很好的完成排队,先进先出的原则。
④set
因为set堆放的是一堆不重复值的集合。所以可以做全局去重的功能。为什么不用JVM自带的Set进行去重?因为我们的系统一般都是集群部署,使用JVM自带的Set,比较麻烦,难道为了一个做一个全局去重,再起一个公共服务,太麻烦了。
⑤sorted set
sorted set多了一个权重参数score,集合中的元素能够按score进行排列。可以做排行榜应用,取TOP N操作。
5、redis的过期策略以及内存淘汰机制
redis采用的是定期删除+惰性删除策略
定期删除:redis默认每个100ms检查,是否有过期的key,有过期key则删除。需要说明的是,redis不是每个100ms将所有的key检查一次,而是随机抽取 进行检查(如果每隔100ms,全部key进行检查,redis岂不是卡死)。因此,如果只采用定期删除策略,会导致很多key到时间没有删除。
于是,惰性删除派上用场。也就是说在你获取某个key的时候,redis会检查一下,这个key如果设置了过期时间那么是否过期了?如果过期了此时就会删除。
采用定期删除+惰性删除就没其他问题了么?
不是的,如果定期删除没删除key。然后你也没即时去请求key,也就是说惰性删除也没生效。这样,redis的内存会越来越高。那么就应该采用内存淘汰机制。
在redis.conf中有一行配置
# maxmemory-policy volatile-lru
1)noeviction:当内存不足以容纳新写入数据时,新写入操作会报错。应该没人用吧。
2)allkeys-lru:当内存不足以容纳新写入数据时,在键空间中,移除最近最少使用的key。推荐使用,目前项目在用这种。
3)allkeys-random:当内存不足以容纳新写入数据时,在键空间中,随机移除某个key。应该也没人用吧,你不删最少使用Key,去随机删。
4)volatile-lru:当内存不足以容纳新写入数据时,在设置了过期时间的键空间中,移除最近最少使用的key。这种情况一般是把redis既当缓存,又做持久化存储的时候才用。不推荐
5)volatile-random:当内存不足以容纳新写入数据时,在设置了过期时间的键空间中,随机移除某个key。依然不推荐
6)volatile-ttl:当内存不足以容纳新写入数据时,在设置了过期时间的键空间中,有更早过期时间的key优先移除。不推荐
6、redis和数据库双写一致性问题
分析:一致性问题是分布式常见问题,还可以再分为最终一致性和强一致性。数据库和缓存双写,就必然会存在不一致的问题。答这个问题,先明白一个前提。就是如果对数据有强一致性要求,不能放缓存。我们所做的一切,只能保证最终一致性。另外,我们所做的方案其实从根本上来说,只能说降低不一致发生的概率,无法完全避免。因此,有强一致性要求的数据,不能放缓存。
首先,采取正确更新策略,先更新数据库,再删缓存。其次,因为可能存在删除缓存失败的问题,提供一个补偿措施即可,例如利用消息队列
7、如何应对缓存穿透和缓存雪崩问题
8、如何解决redis的并发竞争问题
不推荐使用redis的事务机制。因为我们的生产环境,基本都是redis集群环境,做了数据分片操作。你一个事务中有涉及到多个key操作的时候,这多个key不一定都存储在同一个redis-server上。因此,redis的事务机制,十分鸡肋。
(1)如果对这个key操作,不要求顺序
这种情况下,准备一个分布式锁,大家去抢锁,抢到锁就做set操作即可,比较简单。
(2)如果对这个key操作,要求顺序
假设有一个key1,系统A需要将key1设置为valueA,系统B需要将key1设置为valueB,系统C需要将key1设置为valueC.
期望按照key1的value值按照 valueA–>valueB–>valueC的顺序变化。这种时候我们在数据写入数据库的时候,需要保存一个时间戳。假设时间戳如下
系统 A key 1 {valueA 3:00}
系统 B key 1 {valueB 3:05}
系统 C key 1 {valueC 3:10}
那么,假设这会系统B先抢到锁,将key1设置为{valueB 3:05}。接下来系统A抢到锁,发现自己的valueA的时间戳早于缓存中的时间戳,那就不做set操作了。以此类推。
【密码设置】
config set requirepass songtao0404
auth password
【redis.conf】
daemonize yes #redis后台运行
appendonly yes #开启aof日志,它会每次写操作都记录一条日志
bind 192.168.1.207
使用 redis-server redis.conf
【Redis】
1.redis-cli shutdown //关闭服务
2.redis-server redis.conf //启动服务
3.如果设置了密码,可以连接但是没有权限。需要命令 auth password
4.redis是key-value的数据,所以每个数据都是一个键值对键的类型是字符串
值的类型分为五种: ①字符串string
②哈希hash:是一个string类型的field和value的映射表,hash特别适合用于存储对象。
③列表list
④集合set:无序集合,元素唯一不重复
⑤有序集合zset
5.Redis 单个命令是原子操作
6.Redis命令
字符串:
1. SET key value //设置指定Key的值
2.GET key //获取指定Key的值
3. //获取key 中字符串值得子字符
4.GETSET key value //将给定key的 值设为value 并返回旧的key 对应的值
5.MGET key1,key,2 //获取多个键对应的值
6.SETEX KEY SECONDS VALUE //给键和值设定失效时间
7.SETNX KEY VALUE //只有在Key不存在时设置值
8.SETRANGE KEY OFFSET VALUE //将key对赢得字符串从偏移量offset处开始用字符串value 覆盖
9.STRLEN KEY //返回KEY所存储的字符串值得长度
10.MSET KEY1 "VALUE1" KEY2 "VALUE2" //同时设置多个KEY-VALUE 对
11.MSETNX KEY1 "VALUE1" KEY2 "VALUE2" //同时设置一个或多个 key-value 对,当且仅当所有给定 key 都不存在。
12. //和SETEX 不同之处在于 这个是以毫秒为单位设置过期时间
13. // 将KEY中存储的数字值加1
14.INCRBY KEY INCREMENT //将key所存储的值加上指定的值
15.INCRBYFLOAT KEY INCREMENT //将key所存储的值加上浮点增量值
16.DECR KEY //将key中存储的数字值减1
17.APPEND KEY VALUE //将value 值追加到 key 对应的值末尾,key不存在则设置key-value
18.EXPIRE KEY seconds //只对键设置过期时间
哈希:
1.HSET KEY FIELD VALUE //将哈希表中key中的字段Field设置为value
2.HMSET KEY FIELD1 VALUE1 FIELD2 VALUE2 //同时将多个 field-value (域-值)对设置到哈希表 key 中。
3.HGET KEY FIELD //获取Key 中的 Field字段对应的Value值
4.HMGET KEY FIELD1 FIELD2 //获取Key中的多个字段对应的value值
5.HDEL KEY FIELD1 FIELD2 //删除哈希表中一个或多个KEY对应的字段
6.HEXISTS KEY FIELD //查看哈希表KEY中指定的Field是否存在
7.HGETALL KEY //获取在哈希表中指定KEY的所有字段的值
8.HINCRBY KEY FIELD INCREMENT //为哈希表KEY中指定的Field字段的整数值加上增量值
9.HINCRBYFLOAT KEY FIELD INCREMENT //为哈希表KEY中指定的FIELD字段的浮点数值上加上增量值
10.HKEYS KEY //获取KEY对应的所有字段
11.HLEN KEY //获取哈希表中KEY对应的字段数量
12.HVALS KEY //获取哈希表中KEY所对应的所有值
13.HSETNX KEY FIELD VALUE //只有在字段Field不存在时,设置哈希表字段的值
14.HSCAN key cursor [MATCH pattern] [COUNT count] //迭代哈希表中的键值对。
列表:
1.BLPOP KEY TIMEOUT //移出并获取列表的第一个元素, 如果列表没有元素会阻塞列表直到等待超时或发现可弹出元素为止。
2.BRPOP KEY TIMEOUT //移出并获取列表的最后一个元素, 如果列表没有元素会阻塞列表直到等待超时或发现可弹出元素为止。
3. //从列表中弹出一个值,将弹出的元素插入到另外一个列表中并返回它;如果列表没有元素会阻塞列表直到等待超时或发现可弹出元素为止。
4.LINDEX KEY index //通过索引获取列表中的元素
5.LINSERT KEY BEFORE|AFTER privot value //在列表元素前或者列表元素后插入元素
6.LLEN KEY //获取列表长度
7.LPOP KEY //移除并获取列表第一个元素
8.LPUSH KEY VALUE1 VALUE2 //将一个或多个值插入到列表中
9.LPUSHX KEY VALUE //将值插入到一个已存在的列表头部
10.LRANGE KEY START END //获取列表指定范围内的元素
11. //移除列表元素
12.LSET KEY INDEX VALUE //通过索引设置列表元素的值,如果索引位置元素存在会被替代
13.LTRIM KEY START STOP //截取列表
14.RPOP KEY //移除列表的最后一个元素,返回值为移除的元素
15.RPOPLPUSH source destination // 移除列表的最后一个元素,并将该元素添加到另一个列表并返回
16.RPUSH KEY VALUE1 VALUE2 //在列表添加一个或多个元素值
17.RPUSHX KEY VALUE //为已存在的列表添加值
无序集合(Set):
1.SADD KEY MEMBER1 MEMBER2 //向集合添加一个或多个元素
2.SCARD KEY //获取集合成员数
3.SDIFF KEY1 [KEY2] //返回给定所有集合的差集
4.SEMEMBERS KEY //返回集合中的所有成员
5.SDIFFSTORE DESTINATION KEY1 KEY2 //返回所有给定集合的差集,并存储在
6.SINTER KEY1 KEY2 //返回给定所有集合的交集
7.SINTERSTORE destination key1 key2 //返回给定所有集合的交集并存储在 destination 中
8.SISMEMBER KEY MEMBER //判断MEMBERS是不是KEY集合中的元素
9.SMOVE source destination member //将member 元素从source 集合移动到destination
10.SPOP key //移除集合并返回集合中一个随机元素
11.SRANDMEMBER KEY COUNT //返回集合中一个或多个随机数
12.SREM KEY member1 member2 ... //移除集合中一个或多个成员
13.SUNION KEY1 KEY2 .. //返回给定所有集合的并集
14.SUNIONSTORE destination KEY1 KEY2 //所有给定集合的并集存储在destination集合中
有序集合:
1.ZADD KEY SCORE1 MEMBER1 SCORE2 MEMBER2 //向有序集合中添加一个或多个成员,以及成员分数
注意: 在 Redis 2.4 版本以前, ZADD 每次只能添加一个元素。
2.ZCARD key //返回集合中元素个数
3.ZCOUNT KEY MIN MAX //计算在有序集合中指定区间分数的成员数
4.ZINCRBY KEY increment member //有序集合中对指定成员的分数加上增量 incrementb,并返回增量后的分数
。。。。
发布订阅:
1.SUBSCRIBE CHANNEL1 CHANNEL2 //订阅一个或多个频道的消息
2.UNSUBSCRIBE CHANNEL1 CHANNEL2 //退订一个或多个指定频道
3.PUBSUB CHANNELS //CHANNELS 就是参数,查看订阅与发布系统状态
4.PUBLISH CHANNELS MESSAGE //给指定频道发送消息
5.PUNSUBSCRIBE PATTERN //退订所有给定模式的频道
6.PSUBSCRIBE PATTERN //订阅一个或多个给定模式的频道
REDIS性能测试:
redis-benchmark -n 10000 -q //以下实例同时执行 10000 个请求来检测性能
Redis密码:
1.客户端查看是否设置了密码:(默认 requirepass参数是空的)
CONFIG get requirepass
2.设置密码
CONFIG SET requirepass "songtao0404"
3.密码验证命令
AUTH songtao0404
Redis备份与恢复:
备份:
> SAVE //此命令创建当前数据库备份,在redis 安装目录中创建dump.rdb文件
或
> BGSAVE //此命令在后台执行
恢复:将生成的dump.rdb考本到安装目录并启动服务即可
获取redis目录命令:CONFIG GET DIR
Java实现Redis消息队列
消息队列介绍以及项目中为什么使用消息队列请参考:https://www.cnblogs.com/linjiqin/p/5720865.html
1.引入Jedis依赖配置
1 23 4 8redis.clients 5jedis 63.0.0 7
2.ObjectUtil.java 消息对象和byte[]互转工具类
1 package redis; 2 3 import java.io.*; 4 /** 5 * 对象和byte[]互转 6 */ 7 public class ObjectUtil { 8 /** 9 * 对象转byte[]10 * @param obj11 * @return12 * @throws IOException13 */14 public static byte[] object2Bytes(Object obj) throws IOException{15 ByteArrayOutputStream bo=new ByteArrayOutputStream();16 ObjectOutputStream oo=new ObjectOutputStream(bo);17 oo.writeObject(obj);18 byte[] bytes=bo.toByteArray();19 bo.close();20 oo.close();21 return bytes;22 }23 /**24 * byte[]转对象25 * @param bytes26 * @return27 * @throws Exception28 */29 public static Object bytes2Object(byte[] bytes) throws Exception{30 ByteArrayInputStream in=new ByteArrayInputStream(bytes);31 ObjectInputStream sIn=new ObjectInputStream(in);32 return sIn.readObject();33 }34 }
3.Message.java 消息类,需要实现序列化接口,因为要进行网络传输
1 package redis; 2 3 import java.io.Serializable; 4 5 public class Message implements Serializable{ 6 7 /** 8 * serialVersionUID适用于Java的序列化机制。简单来说, 9 * Java的序列化机制是通过判断类的serialVersionUID来验证版本一致性的。10 * 在进行反序列化时,JVM会把传来的字节流中的serialVersionUID与本地相应实体类的serialVersionUID进行比较,11 * 如果相同就认为是一致的,可以进行反序列化,否则就会出现序列化版本不一致的异常,即是InvalidCastException。12 */13 private static final long serialVersionUID = 1967413132568134038L;14 15 private int id;16 private String content;17 18 public Message(int id,String content) {19 this.id = id;20 this.content = content;21 }22 23 public int getId() {24 return id;25 }26 27 public void setId(int id) {28 this.id = id;29 }30 31 public String getContent() {32 return content;33 }34 35 public void setContent(String content) {36 this.content = content;37 }38 39 40 41 }
4.RedisUtils.java redis操作工具类()
1 package redis; 2 3 4 import java.io.File; 5 import java.io.IOException; 6 import java.util.HashMap; 7 import java.util.List; 8 import java.util.Map; 9 import java.util.Set; 10 11 import javax.xml.parsers.DocumentBuilder; 12 import javax.xml.parsers.DocumentBuilderFactory; 13 import javax.xml.parsers.ParserConfigurationException; 14 15 import org.w3c.dom.Document; 16 import org.w3c.dom.NodeList; 17 import org.xml.sax.SAXException; 18 19 import redis.clients.jedis.Jedis; 20 import redis.clients.jedis.JedisPool; 21 import redis.clients.jedis.JedisPoolConfig; 22 23 public class RedisUtils { 24 private static String JEDIS_IP; //redis ip地址 25 private static int JEDIS_PORT; //端口 26 private static String JEDIS_PASSWORD; //密码 27 private static JedisPool jedisPool; 28 29 static { 30 getConfig(); 31 JedisPoolConfig config = new JedisPoolConfig(); 32 //设置最大实例总数 33 config.setMaxTotal(5000); 34 //控制一个pool最多有多少个状态为idle(空闲的)的jedis实例 35 config.setMaxIdle(256); 36 //表示当borrow(引入)一个jedis实例时,最大的等待时间,如果超过等待时间,则直接抛出JedisConnectionException; 37 config.setMaxWaitMillis(3*1000); 38 //在borrow一个jedis实例时,是否提前进行alidate操作;如果为true,则得到的jedis实例均是可用的; 39 config.setTestOnBorrow(true); 40 // 在还会给pool时,是否提前进行validate操作 41 config.setTestOnReturn(true); 42 config.setTestWhileIdle(true); 43 config.setMinEvictableIdleTimeMillis(60000L); 44 config.setTimeBetweenEvictionRunsMillis(3000L); 45 config.setNumTestsPerEvictionRun(-1); 46 jedisPool=new JedisPool(config,JEDIS_IP,JEDIS_PORT,60000); 47 } 48 /** 49 * Dom读取jedis xml配置文件 50 * @return 51 */ 52 private static void getConfig(){ 53 HashMap map = new HashMap(); 54 55 //初始化Dom对象 56 DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance(); 57 try { 58 DocumentBuilder builder = dbf.newDocumentBuilder(); 59 Document dom = builder.parse(new File("jedis.xml")); 60 NodeList nodeList = dom.getElementsByTagName("redisconfig"); 61 NodeList nodeList2 = nodeList.item(0).getChildNodes(); 62 for(int i=0; i < nodeList2.getLength(); i++) { 63 if("ip".equals(nodeList2.item(i).getNodeName())) { 64 JEDIS_IP = nodeList2.item(i).getFirstChild().getNodeValue(); 65 } 66 if("port".equals(nodeList2.item(i).getNodeName())) { 67 JEDIS_PORT = Integer.parseInt(nodeList2.item(i).getFirstChild().getNodeValue()); 68 } 69 if("password".equals(nodeList2.item(i).getNodeName())) { 70 JEDIS_PASSWORD = nodeList2.item(i).getFirstChild().getNodeValue(); 71 } 72 } 73 System.out.println(JEDIS_IP+","+JEDIS_PORT+","+JEDIS_PASSWORD); 74 } catch (ParserConfigurationException e) { 75 // TODO Auto-generated catch block 76 e.printStackTrace(); 77 } catch (SAXException e) { 78 // TODO Auto-generated catch block 79 e.printStackTrace(); 80 } catch (IOException e) { 81 // TODO Auto-generated catch block 82 e.printStackTrace(); 83 } 84 } 85 /** 86 * 获取数据 87 * @param key 88 * @return 89 */ 90 public static String getValue(String key) { 91 Jedis jedis = null; 92 String value = null; 93 try { 94 jedis = jedisPool.getResource(); 95 value = jedis.get(key); 96 } catch (Exception e) { 97 // TODO Auto-generated catch block 98 e.printStackTrace(); 99 }finally {100 //返还到连接池101 jedis.close();102 }103 return value;104 }105 106 public static byte[] get(byte[] key) {107 byte[] value = null;108 Jedis jedis = null;109 try {110 jedis = jedisPool.getResource();111 value = jedis.get(key);112 } catch (Exception e) {113 // TODO: handle exception114 e.printStackTrace();115 }finally {116 //返还到连接池117 jedis.close();118 }119 return value;120 }121 122 /**123 * 存储数据124 * @param key125 * @param value126 */127 public static void set(byte[] key,byte[] value) {128 Jedis jedis = null;129 try {130 jedis = jedisPool.getResource();131 jedis.set(key, value);132 } catch (Exception e) {133 // TODO: handle exception134 e.printStackTrace();135 }finally {136 //返还到连接池137 jedis.close();138 }139 }140 141 public static void set(String key,String value) {142 Jedis jedis = null;143 try {144 jedis = jedisPool.getResource();145 jedis.set(key, value);146 } catch (Exception e) {147 // TODO: handle exception148 e.printStackTrace();149 }finally {150 //返还到连接池151 jedis.close();152 }153 }154 155 public static void set(byte[] key,byte[] value,int time) {156 Jedis jedis = null;157 try {158 jedis = jedisPool.getResource();159 jedis.setex(key, time, value);160 } catch (Exception e) {161 // TODO: handle exception162 e.printStackTrace();163 }finally {164 jedis.close();165 }166 }167 168 public static void hset(byte[] key,byte[] field,byte[] value) {169 Jedis jedis = null;170 try {171 jedis = jedisPool.getResource();172 jedis.hset(key,field,value);173 } catch (Exception e) {174 // TODO: handle exception175 e.printStackTrace();176 }finally{177 jedis.close();178 }179 }180 181 public static void hset(String key,String field,String value) {182 Jedis jedis = null;183 try {184 jedis = jedisPool.getResource();185 jedis.hset(key, field,value);186 187 } catch (Exception e) {188 // TODO: handle exception189 e.printStackTrace();190 }finally {191 jedis.close();192 }193 }194 195 /**196 * 获取数据197 * @param args198 */199 public static String hget(String key,String field) {200 Jedis jedis = null;201 String value = null;202 try {203 jedis.hget(key, field);204 } catch (Exception e) {205 // TODO: handle exception206 e.printStackTrace();207 }finally {208 jedis.close();209 }210 return value;211 }212 213 public static byte[] hget(byte[] key,byte[] field) {214 Jedis jedis = null;215 byte[] value = null;216 try {217 jedis = jedisPool.getResource();218 value = jedis.hget(key, field);219 } catch (Exception e) {220 // TODO: handle exception221 e.printStackTrace();222 }finally {223 jedis.close();224 }225 return value;226 }227 228 /**229 * 删除230 * @param args231 */232 233 public static void hdel(byte[] key,byte[] fields) {234 Jedis jedis = null;235 try {236 jedis = jedisPool.getResource();237 jedis.hdel(key, fields);238 } catch (Exception e) {239 // TODO: handle exception240 e.printStackTrace();241 }finally {242 jedis.close();243 }244 }245 246 public static void hdel(String key,String fields) {247 Jedis jedis = null;248 try {249 jedis = jedisPool.getResource();250 jedis.hdel(key, fields);251 } catch (Exception e) {252 // TODO: handle exception253 e.printStackTrace();254 }finally {255 jedis.close();256 }257 }258 259 /**260 * 存储REDIS队列 ,顺序存储261 * @param args262 */263 public static void lpush(byte[] key,byte[] value) {264 Jedis jedis = null;265 try {266 jedis = jedisPool.getResource();267 jedis.lpush(key, value);268 269 } catch (Exception e) {270 // TODO: handle exception271 e.printStackTrace();272 }finally {273 jedis.close();274 }275 }276 /**277 * 存储REDIS队列,反向存储278 * @param args279 */280 public static void rpush(byte[] key,byte[] value) {281 Jedis jedis = null;282 try {283 jedis = jedisPool.getResource();284 jedis.rpush(key, value);285 } catch (Exception e) {286 // TODO: handle exception287 e.printStackTrace();288 }finally {289 jedis.close();290 }291 }292 293 /**294 * Rpoplpush 命令用于移除列表的最后一个元素,并将该元素添加到另一个列表并返回。 295 * @param key296 * @param destination297 */298 public static void rpoplpush(byte[] key,byte[] destination) {299 Jedis jedis = null;300 try {301 jedis = jedisPool.getResource();302 jedis.rpoplpush(key, destination);303 } catch (Exception e) {304 // TODO: handle exception305 e.printStackTrace();306 }finally {307 jedis.close();308 }309 }310 311 /**312 * 获取队列所有数据313 * @param args314 */315 public static List getList(byte[] key) {316 List list = null;317 Jedis jedis = null;318 try {319 jedis = jedisPool.getResource();320 list = jedis.lrange(key, 0, -1);321 322 } catch (Exception e) {323 // TODO: handle exception324 e.printStackTrace();325 }finally {326 jedis.close();327 }328 return list;329 }330 331 /**332 * 获取队列数据,并删除333 * @param args334 */335 public static byte[] rpop(byte[] key) {336 Jedis jedis = null;337 byte[] value = null;338 try {339 jedis = jedisPool.getResource();340 value = jedis.rpop(key);341 } catch (Exception e) {342 // TODO: handle exception343 e.printStackTrace();344 }finally {345 jedis.close();346 }347 return value;348 }349 350 public static byte[] lpop(byte[] key) {351 Jedis jedis = null;352 byte[] value = null;353 try {354 jedis = jedisPool.getResource();355 value = jedis.lpop(key);356 } catch (Exception e) {357 // TODO: handle exception358 e.printStackTrace();359 }finally {360 jedis.close();361 }362 return value;363 }364 365 public static void hmset(Object key,Map hash) {366 Jedis jedis = null;367 try {368 jedis = jedisPool.getResource();369 jedis.hmset(key.toString(), hash);370 } catch (Exception e) {371 // TODO: handle exception372 e.printStackTrace();373 }finally {374 jedis.close();375 }376 }377 378 /**379 * time 秒380 * @param key381 * @param hash382 * @param time383 */384 public static void hmset(Object key, Map hash , int time) {385 Jedis jedis = null;386 try {387 jedis = jedisPool.getResource();388 jedis.hmset(key.toString(), hash);389 jedis.expire(key.toString(), time);390 } catch (Exception e) {391 // TODO: handle exception392 e.printStackTrace();393 }finally {394 jedis.close();395 }396 }397 398 public static List hmget(Object key,String... fields) {399 List result = null;400 Jedis jedis = null;401 try {402 jedis = jedisPool.getResource();403 result = jedis.hmget(key.toString(), fields);404 } catch (Exception e) {405 // TODO: handle exception406 e.printStackTrace();407 }finally {408 jedis.close();409 }410 return result;411 412 }413 414 public static Set hkeys(String key) {415 Jedis jedis = null;416 Set set = null;417 try {418 jedis = jedisPool.getResource();419 set = jedis.hkeys(key);420 } catch (Exception e) {421 // TODO: handle exception422 e.printStackTrace();423 }finally {424 jedis.close();425 }426 return set;427 }428 429 public static List lrange(byte[] key, int from, int to) {430 List result = null;431 Jedis jedis = null;432 try {433 jedis = jedisPool.getResource();434 result = jedis.lrange(key, from, to);435 436 } catch (Exception e) {437 //释放redis对象438 e.printStackTrace();439 } finally {440 //返还到连接池441 jedis.close();442 443 }444 return result;445 }446 public static Map hgetAll(byte[] key) {447 Map result = null;448 Jedis jedis = null;449 try {450 jedis = jedisPool.getResource();451 result = jedis.hgetAll(key);452 } catch (Exception e) {453 //释放redis对象454 e.printStackTrace();455 } finally {456 //返还到连接池457 jedis.close();458 }459 return result;460 }461 462 public static void del(byte[] key) {463 464 Jedis jedis = null;465 try {466 jedis = jedisPool.getResource();467 jedis.del(key);468 } catch (Exception e) {469 //释放redis对象470 e.printStackTrace();471 } finally {472 //返还到连接池473 jedis.close();474 }475 }476 477 public static long llen(byte[] key) {478 479 long len = 0;480 Jedis jedis = null;481 try {482 jedis = jedisPool.getResource();483 jedis.llen(key);484 } catch (Exception e) {485 //释放redis对象486 e.printStackTrace();487 } finally {488 //返还到连接池489 jedis.close();490 }491 return len;492 }493 494 }
5.Test 测试类
1 package redis; 2 3 import java.io.IOException; 4 5 public class Test { 6 public static byte[] redisKey = "key".getBytes(); 7 8 /** 9 * 初始化队列10 * @throws IOException11 */12 13 static {14 try {15 initStart();16 } catch (IOException e) {17 // TODO Auto-generated catch block18 e.printStackTrace();19 }20 }21 22 private static void initStart() throws IOException {23 for(int i=0 ; i < 10 ; i++) {24 Message message = new Message(i,"这是第"+i+"个内容");25 RedisUtils.lpush(redisKey, ObjectUtil.object2Bytes(message));26 }27 }28 29 private static void pop() throws Exception {30 byte[] bytes = RedisUtils.rpop(redisKey);31 Message msg = (Message)ObjectUtil.bytes2Object(bytes);32 if(msg != null ) {33 System.out.println(msg.getId()+"----"+msg.getContent());34 }35 }36 public static void main(String[] args) {37 try {38 Test.pop();39 System.out.println("退出");40 } catch (Exception e) {41 // TODO Auto-generated catch block42 e.printStackTrace();43 }44 }45 }
---恢复内容结束---