redis redisson 集合的使用案例(RList、Rset、RMap)
redis redisson 集合操作
相關類及接口
Rlist:鏈表
public interface RList<V> extends List<V>, RExpirable, RListAsync<V>, RSortable<List<V>>, RandomAccess {
List<V> get(int... var1); //獲取指定的節(jié)點值
int addAfter(V var1, V var2); //在var1前添加var2
int addBefore(V var1, V var2); //在var1后添加var2
void fastSet(int var1, V var2); //修改var1處的只為var2
List<V> readAll(); //獲取鏈表的所有值
RList<V> subList(int var1, int var2); //獲取var1到var2的子鏈表
List<V> range(int var1); //返回var1往后的鏈表
List<V> range(int var1, int var2); //返回var1到var2的鏈表
void trim(int var1, int var2); //保留var1到var2處的鏈表,其余刪除
void fastRemove(int var1); //刪除var1處的值
boolean remove(Object var1, int var2); //判斷元素是否刪除
<KOut, VOut> RCollectionMapReduce<V, KOut, VOut> mapReduce(); //mapreduce操作
}
RSet:無序集合
public interface RSet<V> extends Set<V>, RExpirable, RSetAsync<V>, RSortable<Set<V>> {
V removeRandom();
Set<V> removeRandom(int var1); //刪除對象
V random();
Set<V> random(int var1); //隨機返回對象
boolean move(String var1, V var2); //判斷集合var1中是否存在var2,類似contains()方法
Set<V> readAll(); //獲取所有對象
int union(String... var1); //集合并集對象個數
Set<V> readUnion(String... var1); //集合并集
int diff(String... var1); //集合差集對象個數
Set<V> readDiff(String... var1); //集合差集
int intersection(String... var1); //集合交集的對象個數
Set<V> readIntersection(String... var1); //集合交集
Iterator<V> iterator(int var1);
Iterator<V> iterator(String var1, int var2);
Iterator<V> iterator(String var1); //遍歷集合
<KOut, VOut> RCollectionMapReduce<V, KOut, VOut> mapReduce();
RSemaphore getSemaphore(V var1);
RCountDownLatch getCountDownLatch(V var1);
RPermitExpirableSemaphore getPermitExpirableSemaphore(V var1); //信號量
RLock getLock(V var1);
RLock getFairLock(V var1);
RReadWriteLock getReadWriteLock(V var1); //鎖操作
Stream<V> stream(int var1);
Stream<V> stream(String var1, int var2);
Stream<V> stream(String var1); //流操作
}
RMap:鍵值對
public interface RMap<K, V> extends ConcurrentMap<K, V>, RExpirable, RMapAsync<K, V> {
void loadAll(boolean var1, int var2);
void loadAll(Set<? extends K> var1, boolean var2, int var3);
V get(Object var1); //獲取var1的值
V put(K var1, V var2); //添加對象
V putIfAbsent(K var1, V var2); //對象不存在則設置
V replace(K var1, V var2); //替換對象
boolean replace(K var1, V var2, V var3); //替換對象
V remove(Object var1); //移除對象
boolean remove(Object var1, Object var2); //移除對象
void putAll(Map<? extends K, ? extends V> var1);
void putAll(Map<? extends K, ? extends V> var1, int var2); //添加對象
Map<K, V> getAll(Set<K> var1); //獲取key在集合var1中的鍵值對
int valueSize(K var1); //key為var1的value大小
V addAndGet(K var1, Number var2); //key為var1的value加var2
long fastRemove(K... var1); //移除對象
boolean fastPut(K var1, V var2); //添加對象
boolean fastReplace(K var1, V var2); //替換key為var1的值為var2
boolean fastPutIfAbsent(K var1, V var2); //如果不存在則設置
Set<K> readAllKeySet(); //獲取所有key,以set形式返回
Collection<V> readAllValues(); //獲取所有value,以collection返回
Set<Entry<K, V>> readAllEntrySet(); //遍歷鍵值對
Map<K, V> readAllMap(); //集合形式轉換為map類型
Set<K> keySet();
Set<K> keySet(int var1);
Set<K> keySet(String var1, int var2);
Set<K> keySet(String var1); //獲取key集合
Collection<V> values();
Collection<V> values(String var1);
Collection<V> values(String var1, int var2);
Collection<V> values(int var1); //獲取所有value
Set<Entry<K, V>> entrySet();
Set<Entry<K, V>> entrySet(String var1);
Set<Entry<K, V>> entrySet(String var1, int var2);
Set<Entry<K, V>> entrySet(int var1); //遍歷鍵值對
<KOut, VOut> RMapReduce<K, V, KOut, VOut> mapReduce();
RSemaphore getSemaphore(K var1);
RCountDownLatch getCountDownLatch(K var1);
RPermitExpirableSemaphore getPermitExpirableSemaphore(K var1); //信號量操作
RLock getLock(K var1);
RLock getFairLock(K var1);
RReadWriteLock getReadWriteLock(K var1); //鎖操作
}
使用示例
public class MyTest {
public static void main(String[] args){
Config config=new Config();
config.useSingleServer().setAddress("redis://******:6379").setPassword("123456");
RedissonClient client= Redisson.create(config);
RList<String> list=client.getList("list");
for (int i=0;i<10;i++){
list.add("瓜田李下 "+i);
}
list.readAll().forEach(System.out::println);
System.out.println("list的數量為:"+list.size()+"\n");
RSet<String> set=client.getSet("set");
for (int i=0;i<10;i++){
set.add("瓜田李下 "+i);
}
for (String s : set) {
System.out.println(s);
}
System.out.println("set的大小為:"+set.size()+"\n");
RMap<Integer,String> map=client.getMap("map");
for (int i=0;i<10;i++){
map.put(i,"瓜田李下 "+i);
}
for (Map.Entry<Integer,String> entry:map.entrySet()){
System.out.println(entry.getKey()+" ==> "+entry.getValue());
}
System.out.println("map的大小為:"+map.size());
}
}
控制臺輸出
瓜田李下 0
瓜田李下 1
瓜田李下 2
瓜田李下 3
瓜田李下 4
瓜田李下 5
瓜田李下 6
瓜田李下 7
瓜田李下 8
瓜田李下 9
list的數量為:10
瓜田李下 0
瓜田李下 1
瓜田李下 7
瓜田李下 3
瓜田李下 5
瓜田李下 4
瓜田李下 9
瓜田李下 8
瓜田李下 6
瓜田李下 2
set的大小為:10
0 ==> 瓜田李下 0
1 ==> 瓜田李下 1
2 ==> 瓜田李下 2
3 ==> 瓜田李下 3
4 ==> 瓜田李下 4
5 ==> 瓜田李下 5
6 ==> 瓜田李下 6
7 ==> 瓜田李下 7
8 ==> 瓜田李下 8
9 ==> 瓜田李下 9
map的大小為:10
Redisson使用注意事項
Redisson 是一個在 Redis 的基礎上實現(xiàn)的 Java 駐內存數據網格,相較于暴露底層操作的Jedis,Redisson提供了一系列的分布式的 Java 常用對象,還提供了許多分布式服務。
特性 & 功能:
- 支持 Redis 單節(jié)點(single)模式、哨兵(sentinel)模式、主從(Master/Slave)模式以及集群(Redis Cluster)模式
- 程序接口調用方式采用異步執(zhí)行和異步流執(zhí)行兩種方式
- 數據序列化,Redisson 的對象編碼類是用于將對象進行序列化和反序列化,以實現(xiàn)對該對象在 Redis 里的讀取和存儲
- 單個集合數據分片,在集群模式下,Redisson 為單個 Redis 集合類型提供了自動分片的功能
- 提供多種分布式對象,如:Object Bucket,Bitset,AtomicLong,Bloom Filter 和 HyperLogLog 等
- 提供豐富的分布式集合,如:Map,Multimap,Set,SortedSet,List,Deque,Queue 等
- 分布式鎖和同步器的實現(xiàn),可重入鎖(Reentrant Lock),公平鎖(Fair Lock),聯(lián)鎖(MultiLock),紅鎖(Red Lock),信號量(Semaphonre),可過期性信號鎖(PermitExpirableSemaphore)等
- 提供先進的分布式服務,如分布式遠程服務(Remote Service),分布式實時對象(Live Object)服務,分布式執(zhí)行服務(Executor Service),分布式調度任務服務(Schedule Service)和分布式映射歸納服務(MapReduce)
- 更多特性和功能,請關注官網:http://redisson.org
實現(xiàn)原理
redis本身是不支持上述的分布式對象和集合,Redisson是通過利用redis的特性在客戶端實現(xiàn)了高級數據結構和特性,例如優(yōu)先隊列的實現(xiàn),是通過客戶端排序整理后再存入redis。
客戶端實現(xiàn),意味著當沒有任何客戶端在線時,這些所有的數據結構和特性都不會保留,也不會自動生效,例如過期事件的觸發(fā)或原來優(yōu)先隊列的元素增加。
注意事項
實時性
RMap中有一個功能是可以設置鍵值對的過期時間的,并可以注冊鍵值對的事件監(jiān)聽器
- 元素淘汰功能(Eviction)
- Redisson的分布式的RMapCache Java對象在基于RMap的前提下實現(xiàn)了針對單個元素的淘汰機制。同時仍然保留了元素的插入順序。由于RMapCache是基于RMap實現(xiàn)的,使它同時繼承了java.util.concurrent.ConcurrentMap接口和java.util.Map接口。Redisson提供的Spring Cache整合以及JCache正是基于這樣的功能來實現(xiàn)的。
- 目前的Redis自身并不支持散列(Hash)當中的元素淘汰,因此所有過期元素都是通過org.redisson.EvictionScheduler實例來實現(xiàn)定期清理的。為了保證資源的有效利用,每次運行最多清理300個過期元素。任務的啟動時間將根據上次實際清理數量自動調整,間隔時間趨于1秒到1小時之間。比如該次清理時刪除了300條元素,那么下次執(zhí)行清理的時間將在1秒以后(最小間隔時間)。一旦該次清理數量少于上次清理數量,時間間隔將增加1.5倍。
正如官方wiki所述,這個功能是通過后臺線程定時去清理的, 所以這個是非實時的(issue-1234:on expired event is not executed in real-time.),延遲在5秒到2小時之間,因此對實時性要求比較高的場景就得自己衡量了。
由于過期時間的非實時性,所以導致過期事件的發(fā)生也是非實時的,相應的監(jiān)聽器可能會延遲了一會兒才收到通知,在我的測試中,ttl設置在秒級誤差是比較大的,分鐘級別的ttl倒還好(左側設置值,右側實際耗時):
1s _ 5s
3s _ 5s
4s _ 5s
5s _ 9s
6s _ 10s
10s _ 15s
1m _ 1m11s
序列化
由Redisson默認的編碼器為JsonJacksonCodec,JsonJackson在序列化有雙向引用的對象時,會出現(xiàn)無限循環(huán)異常。而fastjson在檢查出雙向引用后會自動用引用符$ref替換,終止循環(huán)。
在我的情況中,我序列化了一個service,這個service已被spring托管,而且和另一個service之間也相互注入了,用fastjson能 正常序列化到redis,而JsonJackson則拋出無限循環(huán)異常。
為了序列化后的內容可見,所以不用redission其他自帶的二進制編碼器,自行實現(xiàn)編碼器:
import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.serializer.SerializerFeature;
import io.netty.buffer.ByteBuf;
import io.netty.buffer.ByteBufAllocator;
import io.netty.buffer.ByteBufInputStream;
import io.netty.buffer.ByteBufOutputStream;
import org.redisson.client.codec.BaseCodec;
import org.redisson.client.protocol.Decoder;
import org.redisson.client.protocol.Encoder;
import java.io.IOException;
public class FastjsonCodec extends BaseCodec {
private final Encoder encoder = in -> {
ByteBuf out = ByteBufAllocator.DEFAULT.buffer();
try {
ByteBufOutputStream os = new ByteBufOutputStream(out);
JSON.writeJSONString(os, in,SerializerFeature.WriteClassName);
return os.buffer();
} catch (IOException e) {
out.release();
throw e;
} catch (Exception e) {
out.release();
throw new IOException(e);
}
};
private final Decoder<Object> decoder = (buf, state) ->
JSON.parseObject(new ByteBufInputStream(buf), Object.class);
@Override
public Decoder<Object> getValueDecoder() {
return decoder;
}
@Override
public Encoder getValueEncoder() {
return encoder;
}
}
訂閱發(fā)布
Redisson對訂閱發(fā)布的封裝是RTopic,這也是Redisson中很多事件監(jiān)聽的實現(xiàn)原理(例如鍵值對的事件監(jiān)聽)。
使用單元測試時發(fā)現(xiàn),在事件發(fā)布后,訂閱方需要延時一下才能收到事件。具體原因待查
以上為個人經驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關文章
SpringBoot調用WebService接口的實現(xiàn)示例
本文主要介紹了SpringBoot調用WebService接口的實現(xiàn)示例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2025-03-03
Spring Boot實現(xiàn)動態(tài)更新任務的方法
這篇文章主要介紹了Spring Boot實現(xiàn)動態(tài)更新任務的方法,文中給出了詳細的示例代碼供大家參考學習,對大家學習使用Spring Boot動態(tài)更新任務具有一定的參考價值,需要的朋友們來一起看看吧。2017-04-04
java.lang.NoClassDefFoundError錯誤的原因及解決方法
這篇文章主要給大家介紹了關于java.lang.NoClassDefFoundError錯誤的原因及解決的相關資料,java.lang.NoClassDefFoundError是Java虛擬機在運行時無法找到特定類的錯誤,需要的朋友可以參考下2023-10-10
Spring Boot中使用Spring-data-jpa的配置方法詳解
今天小編就為大家分享一篇關于Spring Boot中使用Spring-data-jpa的配置方法詳解,小編覺得內容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧2019-03-03

