使用JAVA的ExecutorService来限制线程数量

这篇博客介绍了如何利用Java的ExecutorService来控制线程并发数量,详细讲解了ExecutorService的概念,并通过测试代码展示了其在线程管理和资源调度上的应用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

前言
诸如利用多线程并行访问数据库可以提高系统的并发性能,但是线程变多伴随而来的是,当线程数大于DBMS(数据库管理系统)设置的最大DB连接数时,程序就挂掉了。在JAVA中如何避免这种问题呢?
解决方法
使用ExecutorService,限制最大线程数量
ExecutorService是?
ExecutorService是JAVA标准的并行计算库[java.util.concurrent]里包含的接口,封装原来难以使用的JAVA线程,使其简单化。
测试代码
线程
根据给定的生命周期,记录生命周期前后的时间点并输出信息
package sample;

import java.text.SimpleDateFormat;
import java.util.Calendar;

public class SampleThread implements Runnable {
    private int no;
    private int time;

    SimpleDateFormat sdf = new SimpleDateFormat("HH:mm:ss");

    SampleThread(int no, int time) {
        this.no = no;
        this.time = time;
    }

    @Override
    public void run() {
        System.out.println("No." + no + " start ID:" + Thread.currentThread().getId() + " 生命周期:" + time + " 现在时间:" + sdf.format(Calendar.getInstance().getTime()));
        try {
            Thread.sleep(time * 1000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        System.out.println("No." + no + " end ID:" + Thread.currentThread().getId() + " 生命周期:" + time + " 现在时间:" + sdf.format(Calendar.getInstance().getTime()));
    }
}

使用ExecutorService的线程池
package sample;

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class Main {

    public static void main(String[] args) {
        final int MAX_THREADS = 3; //定义线程数最大值

        ExecutorService executorService = Executors.newFixedThreadPool(MAX_THREADS);

        for (int i = 0; i < 10; i++) {
            int no = i;

            int lifeTime = (int)(Math.random() * 9 + 1);

            executorService.submit(new SampleThread(no, lifeTime));
        }


        System.out.println("executor.shutdown();");
        executorService.shutdown();
    }
}
代码执行结果
No.0 start ID:10 生命周期:8 现在时间:22:37:08
No.1 start ID:11 生命周期:4 现在时间:22:37:08
executor.shutdown();
No.2 start ID:12 生命周期:6 现在时间:22:37:08
No.1 end ID:11 生命周期:4 现在时间:22:37:12
No.3 start ID:11 生命周期:6 现在时间:22:37:12
No.2 end ID:12 生命周期:6 现在时间:22:37:14
No.4 start ID:12 生命周期:4 现在时间:22:37:14
No.0 end ID:10 生命周期:8 现在时间:22:37:16
No.5 start ID:10 生命周期:3 现在时间:22:37:16
No.3 end ID:11 生命周期:6 现在时间:22:37:18
No.6 start ID:11 生命周期:4 现在时间:22:37:18
No.4 end ID:12 生命周期:4 现在时间:22:37:18
No.7 start ID:12 生命周期:6 现在时间:22:37:18
No.5 end ID:10 生命周期:3 现在时间:22:37:19
No.8 start ID:10 生命周期:6 现在时间:22:37:19
No.6 end ID:11 生命周期:4 现在时间:22:37:22
No.9 start ID:11 生命周期:5 现在时间:22:37:22
No.7 end ID:12 生命周期:6 现在时间:22:37:24
No.8 end ID:10 生命周期:6 现在时间:22:37:25
No.9 end ID:11 生命周期:5 现在时间:22:37:27
项目含义
No.线程启动时,所给的序号参数
ID线程在系统中的ID
生命周期线程sleep时间
代码结果分析
  • 现在时间同一时间点,不会超过MAX_THREADS所定义的最大值
  • 当线程池中,有空闲的线程配额时,才会执行待执行的下一个线程
  • 从ID占用的ID来看,当前线程池中占用系统3个进程资源
  • 当线程池3个线程起来后,executorService#shutdown也随即执行
  • executorService#submit方法并不会阻塞程序,当利用的线程数到达最大值时,任务会被全部推FIFO队列中,从结果中也可以验证此点。
参考链接
https://qiita.com/nogitsune413/items/ce2f4f2e9c577552220b

查看原文:https://www.huuinn.com/archives/392
更多技术干货:风匀坊
关注公众号:风匀坊
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值