容量限制设计方案

服务通常需要考虑速度和容量限制,增强系统的鲁棒性。

背景

笔者曾负责过某公司内公众号服务开发。公众号接口服务接收到用户的推送请求后会构造公众号消息并写入消息队列,路由服务异步接收到消息后进行消息存储后,再交由推送服务向用户推送消息。基本流程如下图所示:


消息流程.png

消息存储过程:

  1. 路由服务发起消息存储请求,并将消息缓存到本地;
  2. 存储服务成功存储消息后异步发送成功通知;
  3. 路由服务接收到成功通知后从本地缓存获取消息内容后进行后续推送处理;

问题

若存储服务异常,系统会出现什么问题?

  • 路由服务使用local cache临时存储消息。当存储服务异常时,若不加限制,路由服务极有可能导致内存溢出,路由服务不可用;
  • 路由服务发起消息存储请求为异步过程,很有可能会一直消费MQ里的消息,导致存储服务承受更大的服务压力。同时会存在消息可能丢失的风险;

方案

基于信号量实现限制容量的本地缓存。容量大小为信号量个数,当路由服务发起消息存储请求时,信号量减1。当路由服务接收到存储成功通知后,信号量加1。

  • 存储服务正常时,容量限制机制不会起作用,服务性能不会受到影响;
  • 存储服务异常时,本地缓存的容量会越来越小。最后再无可用的信号量时,服务会阻塞等待。此时不再对消息队列进行消费。既避免了服务OOM的状况,也降低了服务继续恶化的可能;

实施

基于信号量实现的限容数据结构BlockingHashMap

public class BlockingHashMap<K, V> {

    private static final int DEFAULT_MAX_AVAILABLE = 1000;
    private final ConcurrentHashMap<K, V> inmap = new ConcurrentHashMap<>(DEFAULT_MAX_AVAILABLE);
    private Semaphore sem;


    public BlockingHashMap() {
        this(DEFAULT_MAX_AVAILABLE);
    }

    public BlockingHashMap(int permits) {
        sem = new Semaphore(permits);
    }

    public V put(K key, V value) {
        boolean wasAdded = false;
        try {
            sem.acquire();
            V v = inmap.putIfAbsent(key, value);
            if (v != null) {
                return v;
            }
            wasAdded = true;
        } catch (Exception e) {
        } finally {
            if (!wasAdded) {
                // 若添加失败,需要释放信号量
                sem.release();
            }
        }
        return value;
    }

    public V remove(K key) {
        V value = inmap.remove(key);
        if (value != null) {
            // 只有当成功移除元素时才释放信号量
            sem.release();
        }
        return value;
    }
}
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,288评论 19 139
  • 关于Mongodb的全面总结 MongoDB的内部构造《MongoDB The Definitive Guide》...
    中v中阅读 32,101评论 2 89
  • 1.ios高性能编程 (1).内层 最小的内层平均值和峰值(2).耗电量 高效的算法和数据结构(3).初始化时...
    欧辰_OSR阅读 29,795评论 8 265
  • 终于找到了一个能自言自语的地方。真好。完美地藏起来了呢。
    _listen_to_me_阅读 312评论 0 0
  • 当苦恼足够 再也说不出宣言 忍受和沉默在夜里 散发着扭曲的光彩 窗外的阳光 有种不真实的好 我坐着 站着 用忙碌填...
    张百酒阅读 207评论 0 3