美文网首页技术方案
容量限制设计方案

容量限制设计方案

作者: Waisam | 来源:发表于2018-10-14 14:07 被阅读0次

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

背景

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


消息流程.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;
    }
}

相关文章

  • 容量限制设计方案

    服务通常需要考虑速度和容量限制,增强系统的鲁棒性。 背景 笔者曾负责过某公司内公众号服务开发。公众号接口服务接收到...

  • 集合框架

    ArrayList 以数组实现。节约空间,但数组有容量限制。超出限制时会增加50%容量,用System.array...

  • Java基础之ArrayList

    一、概述 以数组实现。节约空间,但数组有容量限制。超出限制时会增加50%容量,用System.arraycopy(...

  • Java ArrayList初探

    一、概述 arraylist是以数组实现。节约空间,但是数组容量有限制。超出限制会主动增加50%容量,用Syste...

  • Java集合——ArrayList

    一、概述 以数组实现。节约空间,但数组有容量限制。超出限制时会增加50%容量,用System.arraycopy(...

  • 【阿里P8大牛教你Android入门之路(java篇)】Java

    一、概述 以数组实现。节约空间,但数组有容量限制。超出限制时会增加50%容量,用System.arraycopy(...

  • ArrayList

    ArrayList工作原理 以数组实现,有数组容量的限制,超出限制时会增加一半的容量,用System.arrayc...

  • 从程序员转型到架构师的Java集合小抄

    1.List 1.1 ArrayList 以数组实现。节约空间,但数组有容量限制。超出限制时会增加50%容量,用S...

  • ArrayList工作原理及实现(动态数组)

    一、概述 1.以数组实现。节约空间,但数组有容量限制。超出限制时会增加50%容量,用System.arraycop...

  • java集合总结

    List ArrayList 以数组实现,有容量限制,超出限制时会自动扩充容量,为避免不必要的复制操作,使用时尽量...

网友评论

    本文标题:容量限制设计方案

    本文链接:https://www.haomeiwen.com/subject/coezaftx.html