redis之set应用 实现对账(集合比较)功能
《》
现状:每日在进行系统之间的订单对账时,往往是这样的操作流程;
1.从外部系统拉取数据存入本地数据库;
2.查询本地订单数据集合localSet;
3.查询外部系统订单数据集合outerSet;
4.以本地localSet为基准,对照outerSet,进行遍历,将数据不一致(金额、状态等),或者localSet存在而outerSet不存在的数据,放入新集合localDiffSet;
5.以外部outerSet为基准,对照localSet,进行遍历,将数据(金额、状态等)不一致,或者outerSet存在而localSet不存在的数据,放入新集合outerDiffSet;
6.将localDiffSet与outerDiffSet的数据,存入差异账表
一、JVM的list求交集、并集和差集(去重)示例
package com.dxz.base; import java.util.ArrayList; import java.util.Arrays; import java.util.List; import java.util.stream.Collectors; public class ListTest { public static void main(String[] args) { List<String> stringList = new ArrayList<>(); stringList.add("a"); stringList.add("b"); stringList.add("c"); stringList.add("i"); stringList.add("j"); stringList.add("a"); // 一、求交集 // 方法1:直接通过retainAll直接过滤 List<String> stringList1 = new ArrayList<>(Arrays.asList("a,b,c,d,e,f,g,h".split(","))); stringList1.retainAll(stringList); System.out.println("交集1: " + stringList1); // 方法2:通过过滤掉存在于stringList的数据 List<String> stringList1_2 = new ArrayList<>(Arrays.asList("a,b,c,d,e,f,g,h".split(","))); List<String> strings = stringList1_2.stream().filter(item -> stringList.contains(item)).collect(Collectors.toList()); System.out.println("交集2:" + strings); // 二、并集 // 有重并集 List<String> stringList2 = new ArrayList<>(Arrays.asList("a,b,c,d,e,f,g,h".split(","))); stringList2.addAll(stringList); System.out.println("并集: " + stringList2); // 无重并集 List<String> stringList2_2 = new ArrayList<>(Arrays.asList("a,b,c,d,e,f,g,h".split(","))); List<String> stringList_1 = new ArrayList<>(Arrays.asList("a,b,c,i,j,a".split(","))); stringList2_2.removeAll(stringList_1); stringList_1.addAll(stringList2_2); System.out.println("无重并集: " + stringList_1); // 三、求差集 // 方法1:直接使用removeAll()方法 List<String> stringList3 = new ArrayList<>(Arrays.asList("a,b,c,d,e,f,g,h".split(","))); stringList3.removeAll(stringList); System.out.println("差集1: " + stringList3); // 方法2:通过过滤掉不存在于stringList的数据,然后和本数组进行交集处理 List<String> stringList3_2 = new ArrayList<>(Arrays.asList("a,b,c,d,e,f,g,h".split(","))); stringList3_2.retainAll(stringList3_2.stream().filter(item -> !stringList.contains(item)).collect(Collectors.toList())); System.out.println("差集2:" + stringList3_2); } }
问题:
当比对数据无限多,数据全部在JVM中比对,对服务器的影响就比较大,执行效率也低下;
二、redis解决方案
步骤一:外部系统数据拉取入库
步骤二:从数据库查询需要比对的数据,本地数据(localSet),外部系统数据(outerSet)
//---查询订单信息,组成字符串 SELECT CONCAT(order_no,','outer_order_no,',',trans_amount,',',status) FROM `order_info` where create_time BETWEEN '2015-12-01 00:00:00' and '2015-12-31 23:59:59';
为什么组合成字符串,而不是object对象;因为我们届时要比对的信息就是上述字段,如果字符串一致,那么就说明订单信息一致,而不用再去一一比对对象的属性;
步骤三:将localSet与outerSet分别存入redis
//相关函数:redis.clients.jedis.JedisCluster.sadd(String key, String... member) redisClusterUtils.sadd("{account}:localSet", "GM002215120800002,0.01,3","GM002215120800003,0.01,3"); redisClusterUtils.sadd("{account}:outerSet", "CZ001215120800010,0.01,3","CZ001215120800013,0.01,1");
注意点:这里的key,必须要用{}形式,来指定,使我们要比对的集合都处于同一slot,不然在稍后比对时会出现异常:
No way to dispatch this command to Redis Cluster because keys have different slots.
步骤四:进行2个集合的比对,得出交集union,将交集放入key”{account}:union”中
redisClusterUtils.sinterstore("{account}:union", "{account}:localSet", "{account}:outerSet");
步骤五:localSet和outerSet分别与交集进行比较,得出差集{account}:localDiff、{account}:outerDiff
redisClusterUtils.sdiffstore("{account}:localDiff", "{account}:localSet", "{account}:union");
redisClusterUtils.sdiffstore("{account}:outerDiff", "{account}:localSet", "{account}:union");
步骤六:将差集的数据存入数据库差异账表
//--获取差集的每个成员 redisClusterUtils.smembers("{account}:localDiff"); redisClusterUtils.smembers("{account}:outerDiff");
------------------------------------------------------------
补充一下api:
private SetOperations<String, String> setOpe; @SuppressWarnings("rawtypes") public RedisManage(StringRedisTemplate template) { this.setOpe = template.opsForSet(); } //---------------------------------------------- /** * 往set中添加元素 * @param setKey * @param sets * @return */ public Long sadd(int ttl, String setKey, String... sets) { Long count = setOpe.add(setKey, sets); stringRedisTemplate.expire(setKey, ttl, TimeUnit.SECONDS); return count; } /** * 求两个set集合的交集 * @param setKey * @param otherSetKey * @return */ public Set<String> sinter(String setKey, String otherSetKey) { return setOpe.intersect(setKey, otherSetKey); } /** * 求两个set集合的交集,并将交集保存到指定的set集合 * @param setKey * @param otherSetKey * @param destSetKey * @return */ public Long sinterstore(int ttl, String setKey, String otherSetKey, String destSetKey) { Long count = setOpe.intersectAndStore(setKey, otherSetKey, destSetKey); stringRedisTemplate.expire(destSetKey, ttl, TimeUnit.SECONDS); return count; } /** * 求两个set集合的差集 * @param setKey * @param otherSetKey * @return */ public Set<String> sdiff(String setKey, String otherSetKey) { return setOpe.difference(setKey, otherSetKey); } /** * 求两个set集合的差集,并将差集保存到指定的set集合 * @param setKey * @param otherSetKey * @param destSetKey * @return */ public Long sdiffstore(int ttl, String setKey, String otherSetKey, String destSetKey) { Long count = setOpe.differenceAndStore(setKey, otherSetKey, destSetKey); stringRedisTemplate.expire(destSetKey, ttl, TimeUnit.SECONDS); return count; } /** * 求两个set集合的合集,并将差集保存到指定的set集合 * @param ttl * @param setKey * @param otherSetKey * @param destSetKey * @return */ public Long unionAndStore(int ttl, String setKey, String otherSetKey, String destSetKey) { Long count = setOpe.unionAndStore(setKey, otherSetKey, destSetKey); stringRedisTemplate.expire(destSetKey, ttl, TimeUnit.SECONDS); return count; } /** * set的size * @param setKey * @return */ public Long ssize(String setKey) { return setOpe.size(setKey); } /** * 返回set的所有集合 * @param setKey * @return */ public Set<String> smembers(String setKey) { return setOpe.members(setKey); }
关于redis集合(Set)操作的的相关命令,Redis集合命令相关资料
转载:https://blog.csdn.net/qq_33144861/article/details/79467888