Redis和Redis集羣模糊匹配刪除key

Questions

在數據庫內我們可以通過like關鍵字、*或者REGEX關鍵字進行模糊匹配。而在Redis內我們如何進行模糊匹配呢?集羣情況Redis Cluster的情況是否和單機一致呢?前段時間我對於這個議題進行了調查和研究。

單節點的情況

  • Jedis 
    參考stackoverflow上的解答,在Java內使用Jedis主要有如下2中寫法:
    ### 方法1
    Set<String> keys = jedis.keys(pattern);
    for (String key : keys) {
        jedis.del(key);
    } 
    
    ### 方法2
    Jedis jedis = new Jedis("127.0.0.1");
    ScanParams scanParams = new ScanParams();
    scanParams.match("prifix*");
    scanParams.count(1000);
    ScanResult<String> result = jedis.scan(0,scanParams);
    result.getResult().forEach(key -> {
        jedis.del(key);
    });
    
    ### 注意scan方法由於某些bug在2.9版本內scan(int,ScanParams)改爲了scan(String,ScanParams)。由於cursor的位數,方法有些調整。
    

     

  • 方法1,通過keys命令先尋找到所有符合的key,然後把它們刪除;
  • 方法2,通過scan命令掃描所有符合的key,然後把它們刪除。
  • 注意: Redis飾單線程模式,全局掃描的話有可能會導致Redis在一段時間內的卡頓情況發生。

  • Redis-cli

  • redis-cli keys 1.cn*|xargs redis-cli del
     

    Redis Cluster情況
    在Redis Cluster情況與單節點多情況完全不太一樣。

    首先,Redis Cluster是將整個Redis 的hash槽分佈在三臺機器上,要想一下全部掃描出來,顯然是不太現實的。
    Redis內提供Hash-Tag,將相類似的鍵放在一臺機器上。可以通過Hash-Tag進行掃描,可以剪短時間消耗。
    最後需要考慮,主從集羣節點的情況。
    Hash-Tag
    Hash-Tag 是用一個花括號將主要的Hash判斷部分擴起來,例如{hello1}key1、{hello1}key2。一般Hash-tag一致的情況,鍵會存儲在集羣的同一臺機器上。在Jedis 2.9版本提供了這樣的掃描方法。 
    (PS . rediscluster是沒有keys方法的)
     

    public static void deleteRedisKeyStartWith(String redisKeyStartWith) {
            try{
                jedisCluster.getClusterNodes();
    
                ScanParams scanParams = new ScanParams();
    //          scanParams.match("{123}keys*");
    
    //          scanParams.count(1000);
                ScanResult<String> result = jedisCluster.scan("0", scanParams);
                result.getResult().forEach(key -> {
                    jedisCluster.del(key);
                });
    //          jedisCluster.del(wrapperKey(redisKeyStartWith)+".*");
                log.info("success deleted redisKeyStartWith:{}", redisKeyStartWith);
            }finally{
            }
        }
    

    土辦法 分別掃描各個hash槽
     

    public static void deleteRedisKeyStartWith(String redisKeyStartWith) {
            try {
                Map<String, JedisPool> clusterNodes = jedisCluster.getClusterNodes();
    
                for (Map.Entry<String, JedisPool> entry : clusterNodes.entrySet()) {
                    Jedis jedis = entry.getValue().getResource();
                    // 判斷非從節點(因爲若主從複製,從節點會跟隨主節點的變化而變化)
                    if (!jedis.info("replication").contains("role:slave")) {
                        Set<String> keys = jedis.keys(redisKeyStartWith + "*");
                        if (keys.size() > 0) {
                            Map<Integer, List<String>> map = new HashMap<>();
                            for (String key : keys) {
                                // cluster模式執行多key操作的時候,這些key必須在同一個slot上,不然會報:JedisDataException:
                                // CROSSSLOT Keys in request don't hash to the same slot
                                int slot = JedisClusterCRC16.getSlot(key);
                                // 按slot將key分組,相同slot的key一起提交
                                if (map.containsKey(slot)) {
                                    map.get(slot).add(key);
                                } else {
                                    map.put(slot, Lists.newArrayList(key));
                                }
                            }
                            for (Map.Entry<Integer, List<String>> integerListEntry : map.entrySet()) {
                                jedis.del(integerListEntry.getValue().toArray(new String[integerListEntry.getValue().size()]));
                            }
                        }
                    }
                }
                log.info("success deleted redisKeyStartWith:{}", redisKeyStartWith);
            } finally {
            }
        }
    
    //### 未使用slot批次提交(有可能效率略差於前者)
    //獲取jedis連接
    
             private JedisCluster jedisCluster=JedisClusterUtil.getJedisCluster();
    
             //@param pattern  獲取key的前綴  全是是 * 
    
     public static TreeSet<String> keys(String pattern){  
    
    
           TreeSet<String> keys = new TreeSet<>();  
            //獲取所有的節點
    
                   Map<String, JedisPool> clusterNodes = jedisCluster.getClusterNodes();  
           //遍歷節點 獲取所有符合條件的KEY 
    
                   for(String k : clusterNodes.keySet()){  
               logger.debug("Getting keys from: {}", k);  
               JedisPool jp = clusterNodes.get(k);  
               Jedis connection = jp.getResource();  
               try {  
                   keys.addAll(connection.keys(pattern));  
               } catch(Exception e){  
                   logger.error("Getting keys error: {}", e);  
               } finally{  
                   logger.debug("Connection closed.");  
                   connection.close();//用完一定要close這個鏈接!!!  
               }  
           }  
           logger.debug("Keys gotten!");  
           return keys;  
      }  
    
              //main方法
    
     public static void main(String[] args ){
     TreeSet<String> keys=keys("*");
    
     //遍歷key  進行刪除  可以用多線程
    
     for(String key:keys){
    
                              jedisCluster.del(key);
     System.out.println(key);
     }
     }
    

     

  • 原文地址:https://blog.csdn.net/u010416101/article/details/80754171

點贊或者評論是我最大的動力,有問題歡迎留言或者聯繫q:1559810637  
 

 

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章