欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

在redisCluster中模糊获取key方式

程序员文章站 2022-06-26 13:53:03
在一个集群中,显然不能通过keys方法通过pattern直接获取key的集合;鉴于这种问题,产生了两种思路,如下:方案1:已知相同的tag的kv会在一个节点上,所以只要key带有相同的hashtag,...

在一个集群中,显然不能通过keys方法通过pattern直接获取key的集合;

鉴于这种问题,产生了两种思路,如下:

方案1:

已知相同的tag的kv会在一个节点上,所以只要key带有相同的hashtag,则会在一个节点上,所以只要扫描该节点即可,这样就将集群转化为了单点。

@requestmapping(value = "/ceshi", method = requestmethod.get)
    @responsebody
    public void rediskeys() {
        /**
         * 模糊匹配
         * @param pattern key的正则表达式
         * @param count 每次扫描多少条记录,值越大消耗的时间越短,但会影响redis性能。建议设为一千到一万
         * @return 匹配的key集合
         */
        try{
            jediscluster.getclusternodes();
            scanparams scanparams = new scanparams();
            scanparams.match("{operatingsystem}*");
            scanparams.count(1000);
            scanresult<string> result = jediscluster.scan("0", scanparams);
            list<string> keylist = result.getresult();
            system.out.println("keylist======="+keylist);
        }finally{
        }
    }
//scanparams.match("*{zmc}*");//success
//scanparams.match("zmc_text:{zmc}*");//success

上述match方法中:括号中的参数也可以按照如上方式编写;

其关键在于 key 上传到redis中,命名方式里面,必须含有 {};

并且{}前面、后面有无参数必须指定;若key为 zmc_text: {zmc}:1

scanparams.match("{zmc}*");查不出结果

方案2:

获取所有的节点,分别扫描每个节点,根据pattern获取节点中的key,整合起来即可;

注意:cluster模式执行多key操作的时候,这些key必须在同一个slot上,不然会报jedisdataexception异常;

@requestmapping(value = "/ceshi3", method = requestmethod.get)
@responsebody
public void rediskeys() {
    string rediskeystartwith="ad:ads:id:";
    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()) {
                        system.out.println("integerlistentry="+integerlistentry);
                        //jedis.del(integerlistentry.getvalue().toarray(new string[integerlistentry.getvalue().size()]));
                    }
                }
            }
        }
        logger.info("success rediskeys:{}", rediskeystartwith);
    } finally {
    }
}

redis集群获取所有的key

redis单机查询所有key命令

keys *

查询结果示例:

在redisCluster中模糊获取key方式

redis集群查所有key命令:

如果使用keys *,那么查询的仍旧是本服务器的所有key,不是集群的(结合本图结果以及参考上图,都是插入后查询,无心插入或者删除key)

正确的命令是

./redis-cli -c --cluster call 192.168.168.161:7001 keys \*

注意:

1.不能去掉\;

2.换成你redis集群的一个节点的ip和端口

3.如果集群有密码加上参数 -a password(你的redis集群密码) 如本地测试环境查询结果:

在redisCluster中模糊获取key方式

以上为个人经验,希望能给大家一个参考,也希望大家多多支持。