• Redis 最佳实践


    Redis键值设计

    优雅的key的结构

     Redis的版本低于4.0则长度为embstr的上限为39

     拒绝BigKey

     String的key可以使用以下指令查看key所占的内存大小,但是不推荐使用对cpu的使用率比较高,所以String类型可以通过strlen key来查看长度大概估算大小,集合-拿list举例可以使用llen key来查看集合长度,大概估算大小

    memory usage key
    

     

     Scan扫描的简单实现

    1. final static int STR_MAX_LEN = 10 * 1024;
    2. final static int HASH_MAX_LEN = 500;
    3. @Test
    4. void testScan() {
    5. int maxLen = 0;
    6. long len = 0;
    7. String cursor = "0";
    8. do {
    9. // 扫描并获取一部分key
    10. ScanResult result = jedis.scan(cursor);
    11. // 记录cursor
    12. cursor = result.getCursor();
    13. List list = result.getResult();
    14. if (list == null || list.isEmpty()) {
    15. break;
    16. }
    17. // 遍历
    18. for (String key : list) {
    19. // 判断key的类型
    20. String type = jedis.type(key);
    21. switch (type) {
    22. case "string":
    23. len = jedis.strlen(key);
    24. maxLen = STR_MAX_LEN;
    25. break;
    26. case "hash":
    27. len = jedis.hlen(key);
    28. maxLen = HASH_MAX_LEN;
    29. break;
    30. case "list":
    31. len = jedis.llen(key);
    32. maxLen = HASH_MAX_LEN;
    33. break;
    34. case "set":
    35. len = jedis.scard(key);
    36. maxLen = HASH_MAX_LEN;
    37. break;
    38. case "zset":
    39. len = jedis.zcard(key);
    40. maxLen = HASH_MAX_LEN;
    41. break;
    42. default:
    43. break;
    44. }
    45. if (len >= maxLen) {
    46. System.out.printf("Found big key : %s, type: %s, length or size: %d %n", key, type, len);
    47. }
    48. }
    49. } while (!cursor.equals("0"));
    50. }

    恰当的数据类型

     

     

     

     

     总结

     批处理优化

    Pipeline

     

     总结

     集群下的批处理

     并行slot - jedis实现

    1. public class JedisClusterTest {
    2. private JedisCluster jedisCluster;
    3. @BeforeEach
    4. void setUp() {
    5. // 配置连接池
    6. JedisPoolConfig poolConfig = new JedisPoolConfig();
    7. poolConfig.setMaxTotal(8);
    8. poolConfig.setMaxIdle(8);
    9. poolConfig.setMinIdle(0);
    10. poolConfig.setMaxWaitMillis(1000);
    11. HashSet nodes = new HashSet<>();
    12. nodes.add(new HostAndPort("192.168.150.101", 7001));
    13. nodes.add(new HostAndPort("192.168.150.101", 7002));
    14. nodes.add(new HostAndPort("192.168.150.101", 7003));
    15. nodes.add(new HostAndPort("192.168.150.101", 8001));
    16. nodes.add(new HostAndPort("192.168.150.101", 8002));
    17. nodes.add(new HostAndPort("192.168.150.101", 8003));
    18. jedisCluster = new JedisCluster(nodes, poolConfig);
    19. }
    20. @Test
    21. void testMSet() {
    22. jedisCluster.mset("name", "Jack", "age", "21", "sex", "male");
    23. }
    24. @Test
    25. void testMSet2() {
    26. Map map = new HashMap<>(3);
    27. map.put("name", "Jack");
    28. map.put("age", "21");
    29. map.put("sex", "Male");
    30. Map>> result = map.entrySet()
    31. .stream()
    32. .collect(Collectors.groupingBy(
    33. entry -> ClusterSlotHashUtil.calculateSlot(entry.getKey()))
    34. );
    35. for (List> list : result.values()) {
    36. String[] arr = new String[list.size() * 2];
    37. int j = 0;
    38. for (int i = 0; i < list.size(); i++) {
    39. j = i<<2;
    40. Map.Entry e = list.get(0);
    41. arr[j] = e.getKey();
    42. arr[j + 1] = e.getValue();
    43. }
    44. jedisCluster.mset(arr);
    45. }
    46. }
    47. @AfterEach
    48. void tearDown() {
    49. if (jedisCluster != null) {
    50. jedisCluster.close();
    51. }
    52. }
    53. }

    并行slot - spring实现

    1. #配置分片集群
    2. spring:
    3. redis:
    4. cluster:
    5. nodes: #配置分片集群的每一个节点信息
    6. - 192.168.99.100:7001
    7. - 192.168.99.100:7002
    8. - 192.168.99.100:7003
    9. - 192.168.99.100:8001
    10. - 192.168.99.100:8002
    11. - 192.168.99.100:8003
    1. @Test
    2. void testMSetInCluster() {
    3. Map map = new HashMap<>(3);
    4. map.put("name", "Rose");
    5. map.put("age", "21");
    6. map.put("sex", "Female");
    7. stringRedisTemplate.opsForValue().multiSet(map);
    8. List strings = stringRedisTemplate.opsForValue().multiGet(Arrays.asList("name", "age", "sex"));
    9. strings.forEach(System.out::println);
    10. }

    服务端优化

    持久化配置

     慢查询

     

     上述修改都是一次性的,如果想要永久生效,修改redis.conf

    或者在客户端软件中也可以查看到慢查询

     命令及安全配置

     

     rename-command

    bind

    不要配置bind 为0.0.0.0全部都可以访问,配置成局域网ip

    内存配置

     

     

     可以使用client list,查看每一个客户端的信息,其中就包含缓冲区的信息

     集群最佳实践

     集群完整性问题

     当配置为yes时,假设集群3主3从,如果一个master节点和它的slaver节点同时down了.则表明一部分插槽不可用,这时在往这个集群插入数据时,无论它的插槽值是否是在正常运行的节点上还是down的节点上,整个集群都不可用。

    当配置为no时,假设集群3主3从,如果一个master节点和它的slaver节点同时down了..则表明一部分插槽不可用,这时在往这个集群插入数据时,如果它的插槽值是在正常运行的节点上则可以正常存储以及获取,但如果插槽值在down的节点上,则对该key的指令执行失败。

    集群带宽问题

     数据倾斜问题

    如果存在bigkey,并且存储时使用了相同的值hash“{hashkey}”时,会导致存储这些bigkey存储到同一台的集群节点上,导致这个节点相对于其他节点数据过大。

    客户端性能问题

    客户端在访问集群时,都会对插槽判断,对读写分离服务器的判断,会对客户端的性能带来影响

    集群的兼容性问题

    Lua和事务问题

    因为lua和事务都是对操作的原子性,但是在集群的情况下,lua脚本中每一行命令的操作的key,可能插槽值,读写分离都不能保证在同一个节点上,所以在集群下没有办法执行Lua和事务

    总结 

     

  • 相关阅读:
    Java工具库——FastJson的40个常用方法
    Tensorflow2 深度学习十必知
    SSM学习48:日期型请求参数
    网络安全(黑客)自学
    使用Python操作CSV文件,方便又快捷
    Vue后台管理系统框架推荐
    Servlet中Session会话追踪的实现机制
    呼叫中心和电话营销系统相关知识--中继线路
    Altium Designer学习笔记7
    PB数据库开发技术(二)-PowerBuilder数据定义
  • 原文地址:https://blog.csdn.net/qq_33753147/article/details/126570324