Java 信号量机制的实践用途之——流量控制

信号量机制

使用java.util.concurrent下的semaphore类作为信号量

用途

设置一个信号量Map,需要时从此Map中get(特定ID)信号量来控制访问请求数量,以实现对单个请求的流量控制,增加系统稳定性。

用法

  1. 编写SemaphoreManager实现对信号量统一管理
    1. 声明信号量Map Map<Long, Semaphore> semaphoreMap
    2. 编写初始化信号量的方法init(args...),设置初始大小的信号量Map
    3. 编写获取信号量的方法(getSemaphore(args...))、创建信号量方法(createSemaphore(size)),创建信号量时,直接new Semaphore(size)对象即可。

使用时可以在项目初始化方法中、aop中、或相应业务代码中引入使用。

以下是代码部分

SemaphoreManager

java 复制代码
@Component
public class SemaphoreManager {

    @Resource
    private DisConfig disConfig;

    private Map<Long, Semaphore> semaphoreMap = new HashMap<>();

    /**
     * 初始化信号量对象
     *
     * @param list 对象信息列表
     */
    public void init(Collection<ObjectDO> list) {
        if (CollectionUtils.isEmpty(list)) {
            return;
        }
        // 为每个第三方系统生成一个信号量对象
        for (ObjectDO objectDO: list) {
            // 创建信号量对象
            semaphoreMap.put(objectDO.getId(), createSemaphore(objectDO.getMaxConcurrentNum()));
        }
    }

    /**
     * 获取信号量对象
     *
     * @return 信号量对象
     */
    public Semaphore getSemaphore() {
        ObjectContext context = ContextManager.getContext();
        if (context == null) {
            return null;
        }
        return getSemaphore(context.getObjectId(), context.getMaxConcurrentNum());
    }

    /**
     * 获取信号量对象
     *
     * @param objectId 对象ID
     * @param maxConcurrentNum 最大并发数
     * @return 信号量对象
     */
    private Semaphore getSemaphore(Long objectId, Integer maxConcurrentNum) {

        if (semaphoreMap == null) {
            return null;
        }

        // 获取当前对接通道的信号量对象
        Semaphore semaphore = semaphoreMap.get(objectId);
        if (semaphore != null) {
            return semaphore;
        }

        synchronized (objectId.toString().intern()) {
            semaphore = createSemaphore(maxConcurrentNum);
            semaphoreMap.put(objectId, semaphore);
        }

        return semaphore;
    }

    /**
     * 创建信号量对象
     *
     * @param maxConcurrentNum 最大并发数
     * @return 信号量对象
     */
    private Semaphore createSemaphore(Integer maxConcurrentNum) {

        // 如果DB中没有设置则使用disconf中设置的默认值
        if (maxConcurrentNum == null || maxConcurrentNum == 0) {
            maxConcurrentNum = disConfig.getMaxSemp();
        }

        if (maxConcurrentNum == 0) {
            maxConcurrentNum = 100;
        }

        return new Semaphore(maxConcurrentNum);
    }

}

代码使用实践

java 复制代码
                Boolean isAcq = false;
                // 获取信号量对象
                semaphore = semaphoreManager.getSemaphore();

                if (semaphore != null) {
                    // 获取信号量
                    isAcq = semaphore.tryAcquire();
                    // 获取失败时说明请求量过大, 直接返回异常信息
                    AssertUtil.isTrue(isAcq, EnumResultCode.THIRD_SLOWLY);
                }
相关推荐
风铃儿~15 分钟前
Spring AI 入门:Java 开发者的生成式 AI 实践之路
java·人工智能·spring
Code_流苏16 分钟前
C++课设:简易日历程序(支持传统节假日 + 二十四节气 + 个人纪念日管理)
开发语言·c++·stl容器·课设·期末大作业·日历程序·面向对象设计
斯普信专业组20 分钟前
Tomcat全方位监控实施方案指南
java·tomcat
忆雾屿31 分钟前
云原生时代 Kafka 深度实践:06原理剖析与源码解读
java·后端·云原生·kafka
道剑剑非道42 分钟前
QT开发技术【ffmpeg + QAudioOutput】音乐播放器 完善
开发语言·qt·ffmpeg
武昌库里写JAVA44 分钟前
iview Switch Tabs TabPane 使用提示Maximum call stack size exceeded堆栈溢出
java·开发语言·spring boot·学习·课程设计
gaoliheng0061 小时前
Redis看门狗机制
java·数据库·redis
我是唐青枫1 小时前
.NET AOT 详解
java·服务器·.net
lexiangqicheng1 小时前
JS-- for...in和for...of
开发语言·前端·javascript
我是老孙1 小时前
windows10 php报错
开发语言·php