RedisTemplate介绍
spring封装了RedisTemplate对象来进行对redis的各种操作,它支持所有的 redis 原生的api。
RedisTemplate在spring代码中的结构如下:
org.springframework.data.redis.coreClass RedisTemplatejava.lang.Object org.springframework.data.redis.core.RedisAccessor org.springframework.data.redis.core.RedisTemplate
Type Parameters:
K
- the Redis key type against which the template works (usually a String) 模板中的Redis key的类型(通常为String)如:RedisTemplate<String, Object> 注意:如果没特殊情况,切勿定义成RedisTemplate<Object, Object>,否则根据里氏替换原则,使用的时候会造成类型错误 。V
- the Redis value type against which the template works 模板中的Redis value的类型
RedisTemplate中定义了对5种数据结构操作
redisTemplate.opsForValue();//操作字符串redisTemplate.opsForHash();//操作hashredisTemplate.opsForList();//操作listredisTemplate.opsForSet();//操作setredisTemplate.opsForZSet();//操作有序set
StringRedisTemplate与RedisTemplate
-
两者的关系是StringRedisTemplate继承RedisTemplate。
-
两者的数据是不共通的;也就是说StringRedisTemplate只能管理StringRedisTemplate里面的数据,RedisTemplate只能管理RedisTemplate中的数据。
-
SDR默认采用的序列化策略有两种,一种是String的序列化策略,一种是JDK的序列化策略。
StringRedisTemplate默认采用的是String的序列化策略,保存的key和value都是采用此策略序列化保存的。
RedisTemplate默认采用的是JDK的序列化策略,保存的key和value都是采用此策略序列化保存的。
RedisTemplate配置如下:
@Bean public RedisTemplateredisTemplate(RedisConnectionFactory redisConnectionFactory) { Jackson2JsonRedisSerializer
Redis的String数据结构 (推荐使用StringRedisTemplate)
注意:如果使用RedisTemplate需要更改序列化方式
RedisSerializerstringSerializer = new StringRedisSerializer(); template.setKeySerializer(stringSerializer ); template.setValueSerializer(stringSerializer ); template.setHashKeySerializer(stringSerializer ); template.setHashValueSerializer(stringSerializer );
public interface ValueOperations<K,V>
Redis operations for simple (or in Redis terminology 'string') values.ValueOperations可以对String数据结构进行操作:- set void set(K key, V value);
使用:redisTemplate.opsForValue().set("name","tom");结果:redisTemplate.opsForValue().get("name") 输出结果为tom
- set void set(K key, V value, long timeout, TimeUnit unit);
使用:redisTemplate.opsForValue().set("name","tom",10, TimeUnit.SECONDS); 结果:redisTemplate.opsForValue().get("name")由于设置的是10秒失效,十秒之内查询有结果,十秒之后返回为null
- set void set(K key, V value, long offset); 该方法是用 value 参数覆写(overwrite)给定 key 所储存的字符串值,从偏移量 offset 开始
使用:template.opsForValue().set("key","hello world"); template.opsForValue().set("key","redis", 6); System.out.println("***************"+template.opsForValue().get("key")); 结果:***************hello redis
- setIfAbsent Boolean setIfAbsent(K key, V value);设置value时先判断该值原来是否存在
使用:System.out.println(template.opsForValue().setIfAbsent("multi1","multi1"));//false multi1之前已经存在 System.out.println(template.opsForValue().setIfAbsent("multi111","multi111"));//true multi111之前不存在 结果:false true
- multiSet void multiSet(Map<? extends K, ? extends V> m);为多个键分别设置它们的值
使用:Mapmaps = new HashMap (); maps.put("multi1","multi1"); maps.put("multi2","multi2"); maps.put("multi3","multi3"); template.opsForValue().multiSet(maps); List keys = new ArrayList (); keys.add("multi1"); keys.add("multi2"); keys.add("multi3"); System.out.println(template.opsForValue().multiGet(keys)); 结果:[multi1, multi2, multi3]
- multiSetIfAbsent Boolean multiSetIfAbsent(Map<? extends K, ? extends V> m);为多个键分别设置它们的值,如果存在则返回false,不存在返回true
使用:Mapmaps = new HashMap (); maps.put("multi11","multi11"); maps.put("multi22","multi22"); maps.put("multi33","multi33"); Map maps2 = new HashMap (); maps2.put("multi1","multi1"); maps2.put("multi2","multi2"); maps2.put("multi3","multi3"); System.out.println(template.opsForValue().multiSetIfAbsent(maps)); System.out.println(template.opsForValue().multiSetIfAbsent(maps2)); 结果:true false
- get V get(Object key);获取key对应的值
使用:template.opsForValue().set("key","hello world"); System.out.println("***************"+template.opsForValue().get("key")); 结果:***************hello world
- getAndSet V getAndSet(K key, V value);设置键的字符串值并返回其旧值
使用:template.opsForValue().set("getSetTest","test"); System.out.println(template.opsForValue().getAndSet("getSetTest","test2")); 结果:test
- multiGet List<V> multiGet(Collection<K> keys);为多个键分别取出它们的值
使用:Mapmaps = new HashMap (); maps.put("multi1","multi1"); maps.put("multi2","multi2"); maps.put("multi3","multi3"); template.opsForValue().multiSet(maps); List keys = new ArrayList (); keys.add("multi1"); keys.add("multi2"); keys.add("multi3"); System.out.println(template.opsForValue().multiGet(keys)); 结果:[multi1, multi2, multi3]
- increment Long increment(K key, long delta);支持整数
使用:template.opsForValue().increment("increlong",1); System.out.println("***************"+template.opsForValue().get("increlong")); 结果:***************2
- increment Double increment(K key, double delta);也支持浮点数
使用:template.opsForValue().increment("increlong",1.2); System.out.println("***************"+template.opsForValue().get("increlong")); 结果:***************2.2
- append Integer append(K key, String value);如果key已经存在并且是一个字符串,则该命令将该值追加到字符串的末尾。如果键不存在,则它被创建并设置为空字符串,因此APPEND在这种特殊情况下将类似于SET。
使用:template.opsForValue().append("appendTest","Hello"); System.out.println(template.opsForValue().get("appendTest")); template.opsForValue().append("appendTest","world"); System.out.println(template.opsForValue().get("appendTest")); 结果:Hello Helloworld
- get String get(K key, long start, long end);截取key所对应的value字符串
使用:appendTest对应的value为HelloworldSystem.out.println("*********"+template.opsForValue().get("appendTest",0,5)); 结果:*********Hellow 使用:System.out.println("*********"+template.opsForValue().get("appendTest",0,-1)); 结果:*********Helloworld 使用:System.out.println("*********"+template.opsForValue().get("appendTest",-3,-1)); 结果:*********rld
- size Long size(K key);返回key所对应的value值得长度
使用:template.opsForValue().set("key","hello world"); System.out.println("***************"+template.opsForValue().size("key")); 结果:***************11
- setBit Boolean setBit(K key, long offset, boolean value);对 key 所储存的字符串值,设置或清除指定偏移量上的位(bit) key键对应的值value对应的ascii码,在offset的位置(从左向右数)变为value
使用:template.opsForValue().set("bitTest","a"); // 'a' 的ASCII码是 97。转换为二进制是:01100001 // 'b' 的ASCII码是 98 转换为二进制是:01100010 // 'c' 的ASCII码是 99 转换为二进制是:01100011 //因为二进制只有0和1,在setbit中true为1,false为0,因此我要变为'b'的话第六位设置为1,第七位设置为0 template.opsForValue().setBit("bitTest",6, true); template.opsForValue().setBit("bitTest",7, false); System.out.println(template.opsForValue().get("bitTest")); 结果:b
- getBit Boolean getBit(K key, long offset);获取键对应值的ascii码的在offset处位值
使用:System.out.println(template.opsForValue().getBit("bitTest",7)); 结果:false
Redis的List数据结构
这边我们把RedisTemplate序列化方式改回之前的
Jackson2JsonRedisSerializerjackson2JsonRedisSerializer = new Jackson2JsonRedisSerializer (Object.class); ObjectMapper om = new ObjectMapper(); om.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY); om.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL); jackson2JsonRedisSerializer.setObjectMapper(om); RedisTemplate template = new RedisTemplate (); template.setKeySerializer(jackson2JsonRedisSerializer); template.setValueSerializer(jackson2JsonRedisSerializer); template.setHashKeySerializer(jackson2JsonRedisSerializer); template.setHashValueSerializer(jackson2JsonRedisSerializer);
public interface ListOperations<K,V>
Redis列表是简单的字符串列表,按照插入顺序排序。你可以添加一个元素到列表的头部(左边)或者尾部(右边) ListOperations专门操作list列表:- List<V> range(K key, long start, long end);返回存储在键中的列表的指定元素。偏移开始和停止是基于零的索引,其中0是列表的第一个元素(列表的头部),1是下一个元素
使用:System.out.println(template.opsForList().range("list",0,-1)); 结果:[c#, c++, python, java, c#, c#]
- void trim(K key, long start, long end);修剪现有列表,使其只包含指定的指定范围的元素,起始和停止都是基于0的索引
使用:System.out.println(template.opsForList().range("list",0,-1)); template.opsForList().trim("list",1,-1);//裁剪第一个元素 System.out.println(template.opsForList().range("list",0,-1)); 结果:[c#, c++, python, java, c#, c#] [c++, python, java, c#, c#]
- Long size(K key);返回存储在键中的列表的长度。如果键不存在,则将其解释为空列表,并返回0。当key存储的值不是列表时返回错误。
使用:System.out.println(template.opsForList().size("list")); 结果:6
- Long leftPush(K key, V value);将所有指定的值插入存储在键的列表的头部。如果键不存在,则在执行推送操作之前将其创建为空列表。(从左边插入)
使用:template.opsForList().leftPush("list","java"); template.opsForList().leftPush("list","python"); template.opsForList().leftPush("list","c++"); 结果:返回的结果为推送操作后的列表的长度
- Long leftPushAll(K key, V... values);批量把一个数组插入到列表中
使用:String[] stringarrays = new String[]{ "1","2","3"}; template.opsForList().leftPushAll("listarray",stringarrays); System.out.println(template.opsForList().range("listarray",0,-1)); 结果:[3, 2, 1]
- Long leftPushAll(K key, Collection<V> values);批量把一个集合插入到列表中
使用:Liststrings = new ArrayList (); strings.add("1"); strings.add("2"); strings.add("3"); template.opsForList().leftPushAll("listcollection4", strings); System.out.println(template.opsForList().range("listcollection4",0,-1)); 结果:[3, 2, 1]
- Long leftPushIfPresent(K key, V value);只有存在key对应的列表才能将这个value值插入到key所对应的列表中
使用:System.out.println(template.opsForList().leftPushIfPresent("leftPushIfPresent","aa")); System.out.println(template.opsForList().leftPushIfPresent("leftPushIfPresent","bb")); ==========分割线=========== System.out.println(template.opsForList().leftPush("leftPushIfPresent","aa")); System.out.println(template.opsForList().leftPushIfPresent("leftPushIfPresent","bb")); 结果: 0 0 ==========分割线=========== 1 2
- Long leftPush(K key, V pivot, V value);把value值放到key对应列表中pivot值的左面,如果pivot值存在的话
使用:template.opsForList().leftPush("list","java","oc"); System.out.print(template.opsForList().range("list",0,-1)); 结果:[c++, python, oc, java, c#, c#]
- Long rightPush(K key, V value);将所有指定的值插入存储在键的列表的头部。如果键不存在,则在执行推送操作之前将其创建为空列表。(从右边插入)
使用:template.opsForList().rightPush("listRight","java"); template.opsForList().rightPush("listRight","python"); template.opsForList().rightPush("listRight","c++"); 结果: 1 2 3
- Long rightPushAll(K key, V... values);
使用:String[] stringarrays = new String[]{ "1","2","3"}; template.opsForList().rightPushAll("listarrayright",stringarrays); System.out.println(template.opsForList().range("listarrayright",0,-1)); 结果:[1, 2, 3]
- Long rightPushAll(K key, Collection<V> values);
使用:Liststrings = new ArrayList (); strings.add("1"); strings.add("2"); strings.add("3"); template.opsForList().rightPushAll("listcollectionright", strings); System.out.println(template.opsForList().range("listcollectionright",0,-1)); 结果:[1, 2, 3]
- Long rightPushIfPresent(K key, V value);只有存在key对应的列表才能将这个value值插入到key所对应的列表中
使用:System.out.println(template.opsForList().rightPushIfPresent("rightPushIfPresent","aa")); System.out.println(template.opsForList().rightPushIfPresent("rightPushIfPresent","bb")); System.out.println("==========分割线==========="); System.out.println(template.opsForList().rightPush("rightPushIfPresent","aa")); System.out.println(template.opsForList().rightPushIfPresent("rightPushIfPresent","bb")); 结果:0 0 ==========分割线=========== 1 2
- Long rightPush(K key, V pivot, V value);把value值放到key对应列表中pivot值的右面,如果pivot值存在的话
使用:System.out.println(template.opsForList().range("listRight",0,-1)); template.opsForList().rightPush("listRight","python","oc"); System.out.println(template.opsForList().range("listRight",0,-1)); 结果:[java, python, c++] [java, python, oc, c++]
- void set(K key, long index, V value);在列表中index的位置设置value值
使用:System.out.println(template.opsForList().range("listRight",0,-1)); template.opsForList().set("listRight",1,"setValue"); System.out.println(template.opsForList().range("listRight",0,-1)); 结果:[java, python, oc, c++] [java, setValue, oc, c++]
- Long remove(K key, long count, Object value);从存储在键中的列表中删除等于值的元素的第一个计数事件。 计数参数以下列方式影响操作: count> 0:删除等于从头到尾移动的值的元素。 count <0:删除等于从尾到头移动的值的元素。 count = 0:删除等于value的所有元素。
使用:System.out.println(template.opsForList().range("listRight",0,-1)); template.opsForList().remove("listRight",1,"setValue");//将删除列表中存储的列表中第一次次出现的“setValue”。 System.out.println(template.opsForList().range("listRight",0,-1)); 结果:[java, setValue, oc, c++] [java, oc, c++]
- V index(K key, long index);根据下表获取列表中的值,下标是从0开始的
使用:System.out.println(template.opsForList().range("listRight",0,-1)); System.out.println(template.opsForList().index("listRight",2)); 结果:[java, oc, c++] c++
- V leftPop(K key);弹出最左边的元素,弹出之后该值在列表中将不复存在
使用:System.out.println(template.opsForList().range("list",0,-1)); System.out.println(template.opsForList().leftPop("list")); System.out.println(template.opsForList().range("list",0,-1)); 结果: [c++, python, oc, java, c#, c#] c++ [python, oc, java, c#, c#]
- V leftPop(K key, long timeout, TimeUnit unit);移出并获取列表的第一个元素, 如果列表没有元素会阻塞列表直到等待超时或发现可弹出元素为止。
使用:用法与 leftPop(K key);一样
- V rightPop(K key);弹出最右边的元素,弹出之后该值在列表中将不复存在
使用: System.out.println(template.opsForList().range("list",0,-1)); System.out.println(template.opsForList().rightPop("list")); System.out.println(template.opsForList().range("list",0,-1)); 结果:[python, oc, java, c#, c#] c# [python, oc, java, c#]
- V rightPop(K key, long timeout, TimeUnit unit);移出并获取列表的最后一个元素, 如果列表没有元素会阻塞列表直到等待超时或发现可弹出元素为止。
使用:用法与 rightPop(K key);一样
- V rightPopAndLeftPush(K sourceKey, K destinationKey);用于移除列表的最后一个元素,并将该元素添加到另一个列表并返回。
使用:System.out.println(template.opsForList().range("list",0,-1)); template.opsForList().rightPopAndLeftPush("list","rightPopAndLeftPush"); System.out.println(template.opsForList().range("list",0,-1)); System.out.println(template.opsForList().range("rightPopAndLeftPush",0,-1)); 结果:[oc, java,c#] [oc, java] [c#]
- V rightPopAndLeftPush(K sourceKey, K destinationKey, long timeout, TimeUnit unit);用于移除列表的最后一个元素,并将该元素添加到另一个列表并返回,如果列表没有元素会阻塞列表直到等待超时或发现可弹出元素为止。
使用:用法与rightPopAndLeftPush(K sourceKey, K destinationKey)一样
Redis的Hash数据机构
Redis的散列可以让用户将多个键值对存储到一个Redis键里面。
public interface HashOperations<H,HK,HV>HashOperations提供一系列方法操作hash:初始数据://template.opsForHash().put("redisHash","name","tom"); //template.opsForHash().put("redisHash","age",26); //template.opsForHash().put("redisHash","class","6");//MaptestMap = new HashMap(); //testMap.put("name","jack"); //testMap.put("age",27); //testMap.put("class","1"); //template.opsForHash().putAll("redisHash1",testMap);
- Long delete(H key, Object... hashKeys);删除给定的哈希hashKeys
使用:System.out.println(template.opsForHash().delete("redisHash","name")); System.out.println(template.opsForHash().entries("redisHash")); 结果:1 { class=6, age=28.1}
- Boolean hasKey(H key, Object hashKey);确定哈希hashKey是否存在
使用:System.out.println(template.opsForHash().hasKey("redisHash","age")); System.out.println(template.opsForHash().hasKey("redisHash","ttt")); 结果:true false
- HV get(H key, Object hashKey);从键中的哈希获取给定hashKey的值
使用:System.out.println(template.opsForHash().get("redisHash","age")); 结果:26
- List<HV> multiGet(H key, Collection<HK> hashKeys);从哈希中获取给定hashKey的值
使用:Listkes = new ArrayList (); kes.add("name"); kes.add("age"); System.out.println(template.opsForHash().multiGet("redisHash",kes)); 结果:[jack, 28.1]
- Long increment(H key, HK hashKey, long delta);通过给定的delta增加散列hashKey的值(整型)
使用:System.out.println(template.opsForHash().get("redisHash","age")); System.out.println(template.opsForHash().increment("redisHash","age",1)); 结果:26 27
- Double increment(H key, HK hashKey, double delta);通过给定的delta增加散列hashKey的值(浮点数)
使用:System.out.println(template.opsForHash().get("redisHash","age")); System.out.println(template.opsForHash().increment("redisHash","age",1.1)); 结果:27 28.1
- Set<HK> keys(H key);获取key所对应的散列表的key
使用:System.out.println(template.opsForHash().keys("redisHash1"));//redisHash1所对应的散列表为{class=1, name=jack, age=27}结果:[name, class, age]
- Long size(H key);获取key所对应的散列表的大小个数
使用:System.out.println(template.opsForHash().size("redisHash1"));//redisHash1所对应的散列表为{class=1, name=jack, age=27}结果:3
- void putAll(H key, Map<? extends HK, ? extends HV> m);使用m中提供的多个散列字段设置到key对应的散列表中
使用:MaptestMap = new HashMap(); testMap.put("name","jack"); testMap.put("age",27); testMap.put("class","1"); template.opsForHash().putAll("redisHash1",testMap); System.out.println(template.opsForHash().entries("redisHash1")); 结果:{ class=1, name=jack, age=27}
- void put(H key, HK hashKey, HV value);设置散列hashKey的值
使用:template.opsForHash().put("redisHash","name","tom"); template.opsForHash().put("redisHash","age",26); template.opsForHash().put("redisHash","class","6"); System.out.println(template.opsForHash().entries("redisHash")); 结果:{age=26, class=6, name=tom}
- Boolean putIfAbsent(H key, HK hashKey, HV value);仅当hashKey不存在时才设置散列hashKey的值。
使用:System.out.println(template.opsForHash().putIfAbsent("redisHash","age",30)); System.out.println(template.opsForHash().putIfAbsent("redisHash","kkk","kkk")); 结果:false true
- List<HV> values(H key);获取整个哈希存储的值根据密钥
使用:System.out.println(template.opsForHash().values("redisHash")); 结果:[tom, 26, 6]
- Map<HK, HV> entries(H key);获取整个哈希存储根据密钥
使用:System.out.println(template.opsForHash().entries("redisHash"));结果:{age=26, class=6, name=tom}
- Cursor<Map.Entry<HK, HV>> scan(H key, ScanOptions options);使用Cursor在key的hash中迭代,相当于迭代器。
使用:Cursor> curosr = template.opsForHash().scan("redisHash", ScanOptions.ScanOptions.NONE); while(curosr.hasNext()){ Map.Entry entry = curosr.next(); System.out.println(entry.getKey()+":"+entry.getValue()); } 结果:age:28.1 class:6 kkk:kkk
Redis的Set数据结构
Redis的Set是string类型的无序集合。集合成员是唯一的,这就意味着集合中不能出现重复的数据。
Redis 中 集合是通过哈希表实现的,所以添加,删除,查找的复杂度都是O(1)。 public interface SetOperations<K,V> SetOperations提供了对无序集合的一系列操作:- Long add(K key, V... values);无序集合中添加元素,返回添加个数 也可以直接在add里面添加多个值 如:template.opsForSet().add("setTest","aaa","bbb")
使用:String[] strarrays = new String[]{ "strarr1","sgtarr2"}; System.out.println(template.opsForSet().add("setTest", strarrays)); 结果:2
- Long remove(K key, Object... values);移除集合中一个或多个成员
使用:String[] strarrays = new String[]{ "strarr1","sgtarr2"}; System.out.println(template.opsForSet().remove("setTest",strarrays)); 结果:2
- V pop(K key);移除并返回集合中的一个随机元素
使用:System.out.println(template.opsForSet().pop("setTest")); System.out.println(template.opsForSet().members("setTest")); 结果:bbb [aaa, ccc]
- Boolean move(K key, V value, K destKey);将 member 元素从 source 集合移动到 destination 集合
使用:template.opsForSet().move("setTest","aaa","setTest2"); System.out.println(template.opsForSet().members("setTest")); System.out.println(template.opsForSet().members("setTest2")); 结果:[ccc] [aaa]
- Long size(K key);无序集合的大小长度
使用:System.out.println(template.opsForSet().size("setTest")); 结果:1
- Boolean isMember(K key, Object o);判断 member 元素是否是集合 key 的成员
使用:System.out.println(template.opsForSet().isMember("setTest","ccc")); System.out.println(template.opsForSet().isMember("setTest","asd")); 结果:true false
- Set<V> intersect(K key, K otherKey);key对应的无序集合与otherKey对应的无序集合求交集
使用:System.out.println(template.opsForSet().members("setTest")); System.out.println(template.opsForSet().members("setTest2")); System.out.println(template.opsForSet().intersect("setTest","setTest2")); 结果:[aaa, ccc] [aaa] [aaa]
- Set<V> intersect(K key, Collection<K> otherKeys);key对应的无序集合与多个otherKey对应的无序集合求交集
使用:System.out.println(template.opsForSet().members("setTest")); System.out.println(template.opsForSet().members("setTest2")); System.out.println(template.opsForSet().members("setTest3")); Liststrlist = new ArrayList (); strlist.add("setTest2"); strlist.add("setTest3"); System.out.println(template.opsForSet().intersect("setTest",strlist)); 结果:[aaa, ccc] [aaa] [ccc, aaa] [aaa]
- Long intersectAndStore(K key, K otherKey, K destKey);key无序集合与otherkey无序集合的交集存储到destKey无序集合中
使用:System.out.println("setTest:" + template.opsForSet().members("setTest")); System.out.println("setTest2:" + template.opsForSet().members("setTest2")); System.out.println(template.opsForSet().intersectAndStore("setTest","setTest2","destKey1")); System.out.println(template.opsForSet().members("destKey1")); 结果:setTest:[ddd, bbb, aaa, ccc] setTest2:[ccc, aaa] 2 [aaa, ccc]
- Long intersectAndStore(K key, Collection<K> otherKeys, K destKey);key对应的无序集合与多个otherKey对应的无序集合求交集存储到destKey无序集合中
使用:System.out.println("setTest:" + template.opsForSet().members("setTest")); System.out.println("setTest2:" + template.opsForSet().members("setTest2")); System.out.println("setTest3:" + template.opsForSet().members("setTest3")); Liststrlist = new ArrayList (); strlist.add("setTest2"); strlist.add("setTest3"); System.out.println(template.opsForSet().intersectAndStore("setTest",strlist,"destKey2")); System.out.println(template.opsForSet().members("destKey2")); 结果:setTest:[ddd, bbb, aaa, ccc] setTest2:[ccc, aaa] setTest3:[ccc, aaa] 2 [aaa, ccc]
- Set<V> union(K key, K otherKey);key无序集合与otherKey无序集合的并集
使用:System.out.println("setTest:" + template.opsForSet().members("setTest")); System.out.println("setTest2:" + template.opsForSet().members("setTest2")); System.out.println(template.opsForSet().union("setTest","setTest2")); 结果:setTest:[ddd, bbb, aaa, ccc] setTest2:[ccc, aaa] [ccc, aaa, ddd, bbb]
- Set<V> union(K key, Collection<K> otherKeys);key无序集合与多个otherKey无序集合的并集
使用:System.out.println("setTest:" + template.opsForSet().members("setTest")); System.out.println("setTest2:" + template.opsForSet().members("setTest2")); System.out.println("setTest3:" + template.opsForSet().members("setTest3")); Liststrlist = new ArrayList (); strlist.add("setTest2"); strlist.add("setTest3"); System.out.println(template.opsForSet().union("setTest",strlist)); 结果:setTest:[ddd, bbb, aaa, ccc] setTest2:[ccc, aaa] setTest3:[xxx, ccc, aaa] [ddd, xxx, bbb, aaa, ccc]
- Long unionAndStore(K key, K otherKey, K destKey);key无序集合与otherkey无序集合的并集存储到destKey无序集合中
使用:System.out.println("setTest:" + template.opsForSet().members("setTest")); System.out.println("setTest2:" + template.opsForSet().members("setTest2")); System.out.println(template.opsForSet().unionAndStore("setTest","setTest2","unionAndStoreTest1")); System.out.println("unionAndStoreTest1:" + template.opsForSet().members("unionAndStoreTest1")); 结果:setTest:[ddd, bbb, aaa, ccc] setTest2:[ccc, aaa] 4 unionAndStoreTest1:[ccc, aaa, ddd, bbb]
- Long unionAndStore(K key, Collection<K> otherKeys, K destKey);key无序集合与多个otherkey无序集合的并集存储到destKey无序集合中
使用:System.out.println("setTest:" + template.opsForSet().members("setTest")); System.out.println("setTest2:" + template.opsForSet().members("setTest2")); System.out.println("setTest3:" + template.opsForSet().members("setTest3")); Liststrlist = new ArrayList (); strlist.add("setTest2"); strlist.add("setTest3"); System.out.println(template.opsForSet().unionAndStore("setTest",strlist,"unionAndStoreTest2")); System.out.println("unionAndStoreTest2:" + template.opsForSet().members("unionAndStoreTest2")); 结果:setTest:[ddd, bbb, aaa, ccc] setTest2:[ccc, aaa] setTest3:[xxx, ccc, aaa] 5 unionAndStoreTest2:[ddd, xxx, bbb, aaa, ccc]
- Set<V> difference(K key, K otherKey);key无序集合与otherKey无序集合的差集
使用:System.out.println("setTest:" + template.opsForSet().members("setTest")); System.out.println("setTest2:" + template.opsForSet().members("setTest2")); System.out.println(template.opsForSet().difference("setTest","setTest2")); 结果:setTest:[ddd, bbb, aaa, ccc] setTest2:[ccc, aaa] [bbb, ddd]
- Set<V> difference(K key, Collection<K> otherKeys);key无序集合与多个otherKey无序集合的差集
使用:System.out.println("setTest:" + template.opsForSet().members("setTest")); System.out.println("setTest2:" + template.opsForSet().members("setTest2")); System.out.println("setTest3:" + template.opsForSet().members("setTest3")); Liststrlist = new ArrayList (); strlist.add("setTest2"); strlist.add("setTest3"); System.out.println(template.opsForSet().difference("setTest",strlist)); 结果:setTest:[ddd, bbb, aaa, ccc] setTest2:[ccc, aaa] setTest3:[xxx, ccc, aaa] [bbb, ddd]
- Long differenceAndStore(K key, K otherKey, K destKey);key无序集合与otherkey无序集合的差集存储到destKey无序集合中
使用:System.out.println("setTest:" + template.opsForSet().members("setTest")); System.out.println("setTest2:" + template.opsForSet().members("setTest2")); System.out.println(template.opsForSet().differenceAndStore("setTest","setTest2","differenceAndStore1")); System.out.println("differenceAndStore1:" + template.opsForSet().members("differenceAndStore1")); 结果:setTest:[ddd, bbb, aaa, ccc] setTest2:[ccc, aaa] 2 differenceAndStore1:[bbb, ddd]
- Long differenceAndStore(K key, Collection<K> otherKeys, K destKey);key无序集合与多个otherkey无序集合的差集存储到destKey无序集合中
使用:System.out.println("setTest:" + template.opsForSet().members("setTest")); System.out.println("setTest2:" + template.opsForSet().members("setTest2")); System.out.println("setTest3:" + template.opsForSet().members("setTest3")); Liststrlist = new ArrayList (); strlist.add("setTest2"); strlist.add("setTest3"); System.out.println(template.opsForSet().differenceAndStore("setTest",strlist,"differenceAndStore2")); System.out.println("differenceAndStore2:" + template.opsForSet().members("differenceAndStore2")); 结果:setTest:[ddd, bbb, aaa, ccc] setTest2:[ccc, aaa] setTest3:[xxx, ccc, aaa] 2 differenceAndStore2:[bbb, ddd]
- Set<V> members(K key);返回集合中的所有成员
使用:System.out.println(template.opsForSet().members("setTest")); 结果:[ddd, bbb, aaa, ccc]
- V randomMember(K key);随机获取key无序集合中的一个元素
使用:System.out.println("setTest:" + template.opsForSet().members("setTest")); System.out.println("setTestrandomMember:" + template.opsForSet().randomMember("setTest")); System.out.println("setTestrandomMember:" + template.opsForSet().randomMember("setTest")); System.out.println("setTestrandomMember:" + template.opsForSet().randomMember("setTest")); System.out.println("setTestrandomMember:" + template.opsForSet().randomMember("setTest")); 结果:setTest:[ddd, bbb, aaa, ccc] setTestrandomMember:aaa setTestrandomMember:bbb setTestrandomMember:aaa setTestrandomMember:ddd
- Set<V> distinctRandomMembers(K key, long count);获取多个key无序集合中的元素(去重),count表示个数
使用:System.out.println("randomMembers:" + template.opsForSet().distinctRandomMembers("setTest",5)); 结果:randomMembers:[aaa, bbb, ddd, ccc]
- List<V> randomMembers(K key, long count);获取多个key无序集合中的元素,count表示个数
使用:System.out.println("randomMembers:" + template.opsForSet().randomMembers("setTest",5)); 结果:randomMembers:[ccc, ddd, ddd, ddd, aaa]
- Cursor<V> scan(K key, ScanOptions options);遍历set
使用: Cursorcurosr = template.opsForSet().scan("setTest", ScanOptions.NONE); while(curosr.hasNext()){ System.out.println(curosr.next()); }结果:dddbbbaaaccc
Redis的ZSet数据结构
Redis 有序集合和无序集合一样也是string类型元素的集合,且不允许重复的成员。
不同的是每个元素都会关联一个double类型的分数。redis正是通过分数来为集合中的成员进行从小到大的排序。 有序集合的成员是唯一的,但分数(score)却可以重复。 public interface ZSetOperations<K,V> ZSetOperations提供了一系列方法对有序集合进行操作:- Boolean add(K key, V value, double score);新增一个有序集合,存在的话为false,不存在的话为true
使用:System.out.println(template.opsForZSet().add("zset1","zset-1",1.0)); 结果:true
- Long add(K key, Set<TypedTuple<V>> tuples);新增一个有序集合
使用:ZSetOperations.TypedTupleobjectTypedTuple1 = new DefaultTypedTuple ("zset-5",9.6); ZSetOperations.TypedTuple objectTypedTuple2 = new DefaultTypedTuple ("zset-6",9.9); Set > tuples = new HashSet >(); tuples.add(objectTypedTuple1); tuples.add(objectTypedTuple2); System.out.println(template.opsForZSet().add("zset1",tuples)); System.out.println(template.opsForZSet().range("zset1",0,-1)); 结果:[zset-1, zset-2, zset-3, zset-4, zset-5, zset-6]
- Long remove(K key, Object... values);从有序集合中移除一个或者多个元素
使用:System.out.println(template.opsForZSet().range("zset1",0,-1)); System.out.println(template.opsForZSet().remove("zset1","zset-6")); System.out.println(template.opsForZSet().range("zset1",0,-1)); 结果:[zset-1, zset-2, zset-3, zset-4, zset-5, zset-6] 1 [zset-1, zset-2, zset-3, zset-4, zset-5]
- Double incrementScore(K key, V value, double delta);增加元素的score值,并返回增加后的值
使用:System.out.println(template.opsForZSet().incrementScore("zset1","zset-1",1.1)); //原为1.1 结果:2.2
- Long rank(K key, Object o);返回有序集中指定成员的排名,其中有序集成员按分数值递增(从小到大)顺序排列
使用:System.out.println(template.opsForZSet().range("zset1",0,-1)); System.out.println(template.opsForZSet().rank("zset1","zset-2")); 结果:[zset-2, zset-1, zset-3, zset-4, zset-5] 0 //表明排名第一
- Long reverseRank(K key, Object o);返回有序集中指定成员的排名,其中有序集成员按分数值递减(从大到小)顺序排列
使用:System.out.println(template.opsForZSet().range("zset1",0,-1)); System.out.println(template.opsForZSet().reverseRank("zset1","zset-2")); 结果:[zset-2, zset-1, zset-3, zset-4, zset-5] 4 //递减之后排到第五位去了
- Set<V> range(K key, long start, long end);通过索引区间返回有序集合成指定区间内的成员,其中有序集成员按分数值递增(从小到大)顺序排列
使用:System.out.println(template.opsForZSet().range("zset1",0,-1)); 结果:[zset-2, zset-1, zset-3, zset-4, zset-5]
- Set<TypedTuple<V>> rangeWithScores(K key, long start, long end);通过索引区间返回有序集合成指定区间内的成员对象,其中有序集成员按分数值递增(从小到大)顺序排列
使用:Set> tuples = template.opsForZSet().rangeWithScores("zset1",0,-1); Iterator > iterator = tuples.iterator(); while (iterator.hasNext()) { ZSetOperations.TypedTuple typedTuple = iterator.next(); System.out.println("value:" + typedTuple.getValue() + "score:" + typedTuple.getScore()); } 结果:value:zset-2score:1.2 value:zset-1score:2.2 value:zset-3score:2.3 value:zset-4score:6.6 value:zset-5score:9.6
- Set<V> rangeByScore(K key, double min, double max);通过分数返回有序集合指定区间内的成员,其中有序集成员按分数值递增(从小到大)顺序排列
使用:System.out.println(template.opsForZSet().rangeByScore("zset1",0,5)); 结果:[zset-2, zset-1, zset-3]
- Set<TypedTuple<V>> rangeByScoreWithScores(K key, double min, double max);通过分数返回有序集合指定区间内的成员对象,其中有序集成员按分数值递增(从小到大)顺序排列
使用:Set> tuples = template.opsForZSet().rangeByScoreWithScores("zset1",0,5); Iterator > iterator = tuples.iterator(); while (iterator.hasNext()) { ZSetOperations.TypedTuple typedTuple = iterator.next(); System.out.println("value:" + typedTuple.getValue() + "score:" + typedTuple.getScore()); } 结果:value:zset-2score:1.2 value:zset-1score:2.2 value:zset-3score:2.3
- Set<V> rangeByScore(K key, double min, double max, long offset, long count);通过分数返回有序集合指定区间内的成员,并在索引范围内,其中有序集成员按分数值递增(从小到大)顺序排列
使用: System.out.println(template.opsForZSet().rangeByScore("zset1",0,5)); System.out.println(template.opsForZSet().rangeByScore("zset1",0,5,1,2)); 结果:[zset-2, zset-1, zset-3] [zset-1, zset-3]
- Set<TypedTuple<V>> rangeByScoreWithScores(K key, double min, double max, long offset, long count);通过分数返回有序集合指定区间内的成员对象,并在索引范围内,其中有序集成员按分数值递增(从小到大)顺序排列
使用:Set> tuples = template.opsForZSet().rangeByScoreWithScores("zset1",0,5,1,2); Iterator > iterator = tuples.iterator(); while (iterator.hasNext()) { ZSetOperations.TypedTuple typedTuple = iterator.next(); System.out.println("value:" + typedTuple.getValue() + "score:" + typedTuple.getScore()); } 结果:value:zset-1score:2.2 value:zset-3score:2.3
- Set<V> reverseRange(K key, long start, long end);通过索引区间返回有序集合成指定区间内的成员,其中有序集成员按分数值递减(从大到小)顺序排列
使用:System.out.println(template.opsForZSet().reverseRange("zset1",0,-1)); 结果:[zset-5, zset-4, zset-3, zset-1, zset-2]
- Set<TypedTuple<V>> reverseRangeWithScores(K key, long start, long end);通过索引区间返回有序集合成指定区间内的成员对象,其中有序集成员按分数值递减(从大到小)顺序排列
使用:Set> tuples = template.opsForZSet().reverseRangeWithScores("zset1",0,-1); Iterator > iterator = tuples.iterator(); while (iterator.hasNext()) { ZSetOperations.TypedTuple typedTuple = iterator.next(); System.out.println("value:" + typedTuple.getValue() + "score:" + typedTuple.getScore()); } 结果:value:zset-5score:9.6 value:zset-4score:6.6 value:zset-3score:2.3 value:zset-1score:2.2 value:zset-2score:1.2
- Set<V> reverseRangeByScore(K key, double min, double max);
使用:与rangeByScore调用方法一样,其中有序集成员按分数值递减(从大到小)顺序排列
- Set<TypedTuple<V>> reverseRangeByScoreWithScores(K key, double min, double max);
使用:与rangeByScoreWithScores调用方法一样,其中有序集成员按分数值递减(从大到小)顺序排列
- Set<V> reverseRangeByScore(K key, double min, double max, long offset, long count);
使用:与rangeByScore调用方法一样,其中有序集成员按分数值递减(从大到小)顺序排列
- Set<TypedTuple<V>> reverseRangeByScoreWithScores(K key, double min, double max, long offset, long count);
使用:与rangeByScoreWithScores调用方法一样,其中有序集成员按分数值递减(从大到小)顺序排列
- Long count(K key, double min, double max);通过分数返回有序集合指定区间内的成员个数
使用:System.out.println(template.opsForZSet().rangeByScore("zset1",0,5)); System.out.println(template.opsForZSet().count("zset1",0,5)); 结果:[zset-2, zset-1, zset-3] 3
- Long size(K key);获取有序集合的成员数,内部调用的就是zCard方法
使用:System.out.println(template.opsForZSet().size("zset1")); 结果:6
- Long zCard(K key);获取有序集合的成员数
使用:System.out.println(template.opsForZSet().zCard("zset1")); 结果:6
- Double score(K key, Object o);获取指定成员的score值
使用:System.out.println(template.opsForZSet().score("zset1","zset-1")); 结果:2.2
- Long removeRange(K key, long start, long end);移除指定索引位置的成员,其中有序集成员按分数值递增(从小到大)顺序排列
使用:System.out.println(template.opsForZSet().range("zset2",0,-1)); System.out.println(template.opsForZSet().removeRange("zset2",1,2)); System.out.println(template.opsForZSet().range("zset2",0,-1)); 结果:[zset-1, zset-2, zset-3, zset-4] 2 [zset-1, zset-4]
- Long removeRangeByScore(K key, double min, double max);根据指定的score值得范围来移除成员
使用://System.out.println(template.opsForZSet().add("zset2","zset-1",1.1)); //System.out.println(template.opsForZSet().add("zset2","zset-2",1.2)); //System.out.println(template.opsForZSet().add("zset2","zset-3",2.3)); //System.out.println(template.opsForZSet().add("zset2","zset-4",6.6)); System.out.println(template.opsForZSet().range("zset2",0,-1)); System.out.println(template.opsForZSet().removeRangeByScore("zset2",2,3)); System.out.println(template.opsForZSet().range("zset2",0,-1)); 结果:[zset-1, zset-2, zset-3,zset-4] 1 [zset-1, zset-2, zset-4]
- Long unionAndStore(K key, K otherKey, K destKey);计算给定的一个有序集的并集,并存储在新的 destKey中,key相同的话会把score值相加
使用:System.out.println(template.opsForZSet().add("zzset1","zset-1",1.0)); System.out.println(template.opsForZSet().add("zzset1","zset-2",2.0)); System.out.println(template.opsForZSet().add("zzset1","zset-3",3.0)); System.out.println(template.opsForZSet().add("zzset1","zset-4",6.0)); System.out.println(template.opsForZSet().add("zzset2","zset-1",1.0)); System.out.println(template.opsForZSet().add("zzset2","zset-2",2.0)); System.out.println(template.opsForZSet().add("zzset2","zset-3",3.0)); System.out.println(template.opsForZSet().add("zzset2","zset-4",6.0)); System.out.println(template.opsForZSet().add("zzset2","zset-5",7.0)); System.out.println(template.opsForZSet().unionAndStore("zzset1","zzset2","destZset11")); Set> tuples = template.opsForZSet().rangeWithScores("destZset11",0,-1); Iterator > iterator = tuples.iterator(); while (iterator.hasNext()) { ZSetOperations.TypedTuple typedTuple = iterator.next(); System.out.println("value:" + typedTuple.getValue() + "score:" + typedTuple.getScore()); } 结果:value:zset-1score:2.0 value:zset-2score:4.0 value:zset-3score:6.0 value:zset-5score:7.0 value:zset-4score:12.0
- Long unionAndStore(K key, Collection<K> otherKeys, K destKey);计算给定的多个有序集的并集,并存储在新的 destKey中
使用://System.out.println(template.opsForZSet().add("zzset1","zset-1",1.0)); //System.out.println(template.opsForZSet().add("zzset1","zset-2",2.0)); //System.out.println(template.opsForZSet().add("zzset1","zset-3",3.0)); //System.out.println(template.opsForZSet().add("zzset1","zset-4",6.0)); // //System.out.println(template.opsForZSet().add("zzset2","zset-1",1.0)); //System.out.println(template.opsForZSet().add("zzset2","zset-2",2.0)); //System.out.println(template.opsForZSet().add("zzset2","zset-3",3.0)); //System.out.println(template.opsForZSet().add("zzset2","zset-4",6.0)); //System.out.println(template.opsForZSet().add("zzset2","zset-5",7.0)); System.out.println(template.opsForZSet().add("zzset3","zset-1",1.0)); System.out.println(template.opsForZSet().add("zzset3","zset-2",2.0)); System.out.println(template.opsForZSet().add("zzset3","zset-3",3.0)); System.out.println(template.opsForZSet().add("zzset3","zset-4",6.0)); System.out.println(template.opsForZSet().add("zzset3","zset-5",7.0)); ListstringList = new ArrayList (); stringList.add("zzset2"); stringList.add("zzset3"); System.out.println(template.opsForZSet().unionAndStore("zzset1",stringList,"destZset22")); Set > tuples = template.opsForZSet().rangeWithScores("destZset22",0,-1); Iterator > iterator = tuples.iterator(); while (iterator.hasNext()) { ZSetOperations.TypedTuple typedTuple = iterator.next(); System.out.println("value:" + typedTuple.getValue() + "score:" + typedTuple.getScore()); } 结果:value:zset-1score:3.0 value:zset-2score:6.0 value:zset-3score:9.0 value:zset-5score:14.0 value:zset-4score:18.0
- Long intersectAndStore(K key, K otherKey, K destKey);计算给定的一个或多个有序集的交集并将结果集存储在新的有序集合 key 中
使用:System.out.println(template.opsForZSet().intersectAndStore("zzset1","zzset2","destZset33")); Set> tuples = template.opsForZSet().rangeWithScores("destZset33",0,-1); Iterator > iterator = tuples.iterator(); while (iterator.hasNext()) { ZSetOperations.TypedTuple typedTuple = iterator.next(); System.out.println("value:" + typedTuple.getValue() + "score:" + typedTuple.getScore()); } 结果:value:zset-1score:2.0 value:zset-2score:4.0 value:zset-3score:6.0 value:zset-4score:12.0
- Long intersectAndStore(K key, Collection<K> otherKeys, K destKey);计算给定的一个或多个有序集的交集并将结果集存储在新的有序集合 key 中
使用:ListstringList = new ArrayList (); stringList.add("zzset2"); stringList.add("zzset3"); System.out.println(template.opsForZSet().intersectAndStore("zzset1",stringList,"destZset44")); Set > tuples = template.opsForZSet().rangeWithScores("destZset44",0,-1); Iterator > iterator = tuples.iterator(); while (iterator.hasNext()) { ZSetOperations.TypedTuple typedTuple = iterator.next(); System.out.println("value:" + typedTuple.getValue() + "score:" + typedTuple.getScore()); } 结果:value:zset-1score:3.0 value:zset-2score:6.0 value:zset-3score:9.0 value:zset-4score:18.0
- Cursor<TypedTuple<V>> scan(K key, ScanOptions options);遍历zset
使用: Cursor> cursor = template.opsForZSet().scan("zzset1", ScanOptions.NONE); while (cursor.hasNext()){ ZSetOperations.TypedTuple item = cursor.next(); System.out.println(item.getValue() + ":" + item.getScore()); } 结果:zset-1:1.0 zset-2:2.0 zset-3:3.0 zset-4:6.0
注:TimeUnit是java.util.concurrent包下面的一个类,表示给定单元粒度的时间段
常用的颗粒度 TimeUnit.DAYS //天 TimeUnit.HOURS //小时 TimeUnit.MINUTES //分钟 TimeUnit.SECONDS //秒 TimeUnit.MILLISECONDS //毫秒RedisTemplate操作key
操作key的失效时间
1.RedisTemplate.expire(String key, long timeout, TimeUnit unit)
设置key的失效时间,timeout是时间参数,timeunit是时间单位,返回值是布尔
2.RedisTemplate.expireAt(String key, Date date)
设置key在一个时间点失效,返回值是布尔
3.RedisTemplate.getExpire(String key)
获取key的存活时间,返回值是long
4.RedisTemplate.getExpire(String key, TimeUnit timeUnit)
获取key的存活时间,加上了时间单位,返回值是long
操作给定列表、集合、有序集合 key来对元素进行排序
1.RedisTemplate.sort(SortQuery<String> query)
对列表、集合、有序集合的key进行排序,SortQuery是封装的对redis查询的类,返回值类型是List<Object>,具体用法
SortQueryquery = SortQueryBuilder.sort("hanlist")// 排序的key //.by("pattern") key的正则过滤 .noSort() //不使用正则过滤key .get("2") //在value里过滤正则,可以连续写多个get .limit(0, 5) //分页,和mysql一样 .order(Order.DESC) //正序or倒序 .alphabetical(true) //ALPHA修饰符用于对字符串进行排序,false的话只针对数字排序 .build();redisTemplate.sort(query);
2.RedisTemplate.sort(SortQuery<String> query, BulkMapper<UserToken, Object> bulkMapper)
对列表、集合、有序集合的key进行排序,SortQuery是封装的对redis查询的类,BulkMapper可以把从redis里取出的元素转化成我们想要的对象,返回值类型是List<T>,具体用法
BulkMapperhm = new BulkMapper () { @Override public UserToken mapBulk(List bulk) { // 实现BulkMapper接口的方法,来把获取到的排序的数据转换为我们需要的返回类型 Iterator iterator = bulk.iterator(); UserToken userToken = new UserToken(); userToken.setUserId((String)iterator.next()); userToken.setLastTime((long)iterator.next()); userToken.setElementids((LinkedList )iterator.next()); return userToken; } }; redisTemplate.sort(query, hm);
UserToken是POJO类
1 public class UserToken { 2 private String userId; 3 private long lastTime; 4 private LinkedListelementids; 5 public String getUserId() { 6 return userId; 7 } 8 public void setUserId(String userId) { 9 this.userId = userId;10 }11 public long getLastTime() {12 return lastTime;13 }14 public void setLastTime(long lastTime) {15 this.lastTime = lastTime;16 }17 public LinkedList getElementids() {18 return elementids;19 }20 public void setElementids(LinkedList elementids) {21 this.elementids = elementids;22 }23 24 }