【JavaEE】Callable,Semaphore和CountDownLatch

news2025/1/8 5:11:45

    🔥个人主页: 中草药

🔥专栏:【Java】登神长阶 史诗般的Java成神之路


一.Callable

在Java中,Callable接口是一个可以返回结果的异步任务执行方式。它与Runnable接口类似,都是描述一个“任务”,但最主要的区别在于Callable描述的是带返回值的任务,Runnable描述的是不带返回值的任务

Callable 通常需要搭配 FutureTask 来使⽤. FutureTask ⽤来保存 Callable 的返回结果. 因为Callable 往往是在另⼀个线程中执行的, 啥时候执行完并不确定.
FutureTask 就可以负责这个等待结果出来的⼯作.
想象去吃麻辣烫. 当餐点好后, 后厨就开始做了. 同时前台会给你⼀张 "小票" . 这个小票就是
FutureTask. 后面我们可以随时凭这张小票去查看自己的这份麻辣烫做出来了没.

接口定义

Callable接口位于java.util.concurrent包中,其定义如下:

public interface Callable<V> {
    V call() throws Exception;
}

这里的关键点是call()方法,它会抛出异常,这意味着你可以在call()方法中处理任何需要的逻辑,并可能抛出异常。

使用Callable和Future

使用Callable通常涉及到以下几个步骤:

  1. 创建Callable对象

    首先,你需要创建一个实现了Callable接口的类或者匿名内部类,并重写call()方法。
  2. 获取FutureTask对象

    提交任务后,你会得到一个Future对象,该对象可以用来获取计算的结果或检查任务的状态。
  3. 从FutureTask获取结果

    使用Future.get()方法来获取计算的结果。需要注意的是,这将阻塞调用线程直到结果可用。

下面是一个简单的例子:

ublic static void main(String[] args) throws ExecutionException, InterruptedException {
        Callable<Integer> callable=new Callable<Integer>() {
            @Override
            public Integer call() throws Exception {
                int sum=0;
                for (int i = 0; i < 1000; i++) {
                    sum+=i;
                }
                return sum;
            }
        };

        FutureTask<Integer> futureTask=new FutureTask<>(callable);
        Thread t=new Thread(futureTask);
        t.start();

        // 后续需要通过 FutureTask 拿到最终的结果.
        System.out.println(futureTask.get());
    }

注意事项

  • Callable任务可能会抛出异常,这些异常需要被捕获处理。
  • Future.get()方法是阻塞的,如果不想阻塞主线程,可以使用Future.get(long timeout, TimeUnit unit)来设置超时时间。

二.Semaphore

在Java中,Semaphore是一个用于控制同时访问特定资源的线程数量的同步工具。它可以用来管理一组相关许可的集合。每个许可都代表对某个资源的一次访问权限。

构造函数

  • Semaphore(int permits):创建具有给定数量的许可证的Semaphore
  • Semaphore(int permits, boolean fair):创建具有给定数量的许可证的Semaphore,并指定是否公平地获取许可证。

主要方法

获取许可证

  • acquire():获取一个许可证,在没有可用许可证之前一直等待。
  • acquireUninterruptibly():获取一个许可证,在没有可用许可证之前一直等待,即使线程被中断也不会立即返回。
  • tryAcquire():尝试获取一个许可证,如果许可证可用则立即返回true,否则返回false。
  • tryAcquire(long timeout, TimeUnit unit):尝试获取一个许可证,如果在给定的时间内许可证可用,则返回true,否则返回false。

释放许可证

  • release():释放一个许可证,返回一个许可证到池中。

查询状态

  • availablePermits():返回当前可用的许可证数量。
  • drainPermits():获取所有可用的许可证,并返回获取的许可证数量。
  • hasQueuedThreads():如果有线程正在等待获取许可证,则返回true。
  • getQueueLength():返回正在等待获取许可证的线程数。

示例

 public static void main(String[] args) throws InterruptedException {
        Semaphore semaphore=new Semaphore(3);

        semaphore.acquire();
        System.out.println("申请资源");

        semaphore.acquire();
        System.out.println("申请资源");

        semaphore.acquire();
        System.out.println("申请资源");

        semaphore.acquire();
        System.out.println("申请资源");
    }

如上代码运行结果为

如运行结果所示,尽管申请了四次资源,但semaphore做了限制是3,因此只会显示三次打印

如若,测试以下代码

public static void main(String[] args) throws InterruptedException {
        Semaphore semaphore=new Semaphore(3);

        semaphore.acquire();
        System.out.println("申请资源");

        semaphore.acquire();
        System.out.println("申请资源");

        semaphore.acquire();
        System.out.println("申请资源");

        semaphore.release();
        System.out.println("释放资源");

        semaphore.acquire();
        System.out.println("申请资源");
    }

结果为

注意事项

  • 在使用Semaphore时,一定要确保在完成资源访问后调用release()方法来释放许可证,否则会导致其他线程永远无法获取许可证。
  • 如果不希望线程因等待许可证而被中断,可以使用acquireUninterruptibly()
  • 如果希望在等待获取许可证时能够响应中断信号,应使用acquire()方法,并在catch块中处理InterruptedException
  • Semaphore可以用于多种场景,比如控制并发请求的数量、限制数据库连接池的大小等。

三.CountDownlatch

CountDownLatch是Java并发包java.util.concurrent中的一个实用工具类,它允许一个或多个线程等待其他线程完成操作。简单来说,CountDownLatch可以看作是一个计数器,当计数器的值减到0时,等待的线程就会被释放继续执行。

构造函数

  • CountDownLatch(int count):创建一个新的CountDownLatch,初始化计数器为给定的值。

主要方法

  1. 减少计数器

    • void countDown():将计数器的值减1。当计数器的值变为0时,所有等待的线程都会被释放。
  2. 等待计数器变为0

    • void await():使当前线程等待,直到其他线程调用countDown()方法使计数器的值减到0。

    • boolean await(long timeout, TimeUnit unit):使当前线程等待,直到其他线程调用countDown()方法使计数器的值减到0,或者等待时间超过给定的超时时间。

  3. 查询计数器值

    • long getCount():返回当前计数器的值。

示例

下面是一个简单的使用CountDownLatch的例子,结合线程池展示了如何使用CountDownLatch来等待一些子线程完成它们的工作。

public static void main(String[] args) throws InterruptedException {
        ExecutorService service= Executors.newFixedThreadPool(4);

        // 构造方法的数字, 就是拆分出来的任务个数.
        CountDownLatch countDownLatch=new CountDownLatch(20);

        for (int i = 0; i < 20; i++) {
            int id=i;
            service.submit(()->{
                System.out.println("任务"+id+"开始执行");
                try {
                    Thread.sleep(3000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println("任务"+id+"结束执行");
                countDownLatch.countDown();
            });
        }

        // 当 countDownLatch 收到了 20 个 "完成" , 所有的任务就都完成了.
        // await => all wait
        // await 这个词也是计算机术语. 在 python / js 意思是 async wait (异步等待)
        countDownLatch.await();

        System.out.println("所有任务已完成");
    }

注意事项

  • 当调用await()方法时,当前线程会等待,直到计数器的值变为0。如果当前线程在此期间被中断,那么它将抛出InterruptedException,并且计数器的值不会改变。
  • 如果你想设置一个等待的超时时间,可以使用await(long timeout, TimeUnit unit)方法。如果超时时间到了,线程将继续执行,但计数器的值仍然不变。
  • CountDownLatch只能用于一次性事件。一旦计数器的值减为0,就不能再重置。如果需要多次使用,可以考虑使用CyclicBarrier

🍀🍀🍀🍀🍀🍀🍀🍀🍀🍀🍀🍀🍀🍀🍀🍀🍀🍀🍀🍀🍀🍀🍀🍀🍀🍀🍀🍀

以上,就是本期的全部内容啦,若有错误疏忽希望各位大佬及时指出💐

  制作不易,希望能对各位提供微小的帮助,可否留下你免费的赞呢🌸

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2055627.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

如何为列表元素分配序列号(Java Stream API 实战)

摘要&#xff1a; 本文将详细介绍如何在Java中使用Stream API对列表中的元素进行分组&#xff0c;并为每组元素分配一个唯一的序列号。通过一个具体的示例和测试案例&#xff0c;我们将演示这一技术如何简化数据处理&#xff0c;提高代码效率。 在Java编程中&#xff0c;我们经…

Linux-进程间关系与守护进程

一、进程组 1.1 什么是进程组 之前我们提到了进程的概念&#xff0c; 其实每一个进程除了有一个进程 ID(PID)之外 还属于一 个进程组。进程组是一个或者多个进程的集合&#xff0c; 一个进程组可以包含多个进程。 每一 个进程组也有一个唯一的进程组 ID(PGID)&#xff0c; 并且…

阻抗控制中的dynamic movement primitives(DMP) model

在阻抗控制中&#xff0c;Dynamic Movement Primitives (DMP) 模型被用于实现一种高度灵活且可泛化的轨迹模仿学习方法。DMP模型由美国南加州大学&#xff08;University of Southern California&#xff09;的Stefan Schaal教授团队于2002年提出&#xff0c;它通过将动态系统建…

Linux命令之二

Linux命令之二 VI和VIM编辑器1、VIM的一般模式①、删除复制操作②、光标移动操作 2、VIM的编辑模式3、VIM的命令模式4、三种模式之间的切换关系 权限管理文件类型和权限的表示文件属性介绍chmod 改变文件权限chown 改变所有者 yum 安装软件命令 Linux命令速查平台 VI和VIM编辑器…

IOy系列BL196MQTT远程IO模块推动智能交通信号控制

在当今这个快速发展的时代&#xff0c;城市化进程加速推进&#xff0c;随之而来的是日益严重的交通拥堵问题。为了缓解这一状况&#xff0c;智能交通信号控制系统应运而生&#xff0c;成为解决城市交通问题的有效手段之一。 一、钡铼技术BL196模块简介 钡铼技术推出的IOy系列…

【SQL】商品销售

目录 题目 分析 代码 题目 表&#xff1a; Product ----------------------- | Column Name | Type | ----------------------- | product_id | int | | product_name | varchar | | unit_price | int | ----------------------- product_id 是该表的主键…

Redis与CAP定理—理解数据存储的权衡

一、概述 在分布式系统的设计中&#xff0c;理解CAP定理的重要性是不言而喻的。CAP定理指出&#xff0c;一个分布式系统不可能同时完全满足一致性&#xff08;Consistency&#xff09;、可用性&#xff08;Availability&#xff09;和分区容忍性&#xff08;Partition Toleran…

OCR识别行驶证(阿里云和百度云)

OCR识别行驶证(阿里云和百度云) 一、使用场景 1、通过识别行驶证&#xff0c;获取相关汽车信息&#xff0c;替代手输 二、效果图 三、代码部分&#xff1a; 1、阿里云OCR 1.1、控制层 PostMapping("/ocrCard") public JSONObject ocrCard(RequestPart("fi…

开放式耳机哪种好用?开放式种草测评!

现在很多人都很喜欢用开放式耳机了&#xff0c;因为这种耳机非常的舒服&#xff0c;而且不会压迫我们的耳道&#xff0c;从而给到我们更健康的体验&#xff0c;但是现在开放式耳机的品牌越来越多&#xff0c;我们也越来越难选择&#xff0c;所以我们应该怎么样才能选到一款适合…

ado.net 操作sqlite

新建控制台项目 安装nuget包Microsoft.Data.Sqlite 数据库名字和链接 string dbName "test.db"; SqliteConnection? connection null; try {//创建链接connection new SqliteConnection($"Data Source{dbName}");//打开链接connection.Open(); } ca…

RK33568 android12 背景图片替换

文章目录 前言一、直接文件替换二、使用属性替换1.属性获取图片的代码分析2.设置图片路径前言 项目中,需要替换背景图片,要求是黑色的背景图片 修改前: 修改后: 一、直接文件替换 文件路径 device/rockchip/rk356x/overlay/frameworks/base/core/res/res/drawable-nod…

为什么要用数字化营销管理平台?

数字化营销管理平台是一种利用数字技术来整合和优化营销流程的工具。它能够帮助企业更高效地进行市场推广、客户关系管理以及销售活动。 一、主要功能 1.数据整合与分析 整合多渠道数据&#xff0c;包括网站流量、社交媒体互动、电子邮件营销反馈等。通过数据分析&#xff0…

Docker 存储空间不足无法导入加载镜像

问题:在载入镜像时,发现docker没有空间了 解决办法: 更改docker的存储路径 1.添加新的硬盘 docker info #查看docker的存储位置 df -Th #查看占用以及挂载情况 发现没有可用的剩余空间,我们可以添加一个新的硬盘 在l

EMC学习之接地与平面

1 不同信号地的放置 下图为数字地和射频地的两种放置方式&#xff0c;第一个布局方式射频电流必须要经过数字地才能返回电源GND上&#xff0c;而且射频地与数字地之间的间隙很小&#xff0c;所以更容易通过寄生电容来耦合噪声。第二个布局数字地与射频地没有重叠&#xff0c;平…

iOS 18 Beta 5:苹果的细腻之笔,绘制用户体验新画卷

在苹果的世界里&#xff0c;每一次系统更新都是对用户体验进行的一次精心雕琢。 随着iOS 18 Beta 5的上线&#xff0c;苹果带来了一系列令人耳目一新的功能&#xff0c;同时也在系统的每个细微之处展现了对完美的追求。 Safari浏览器的“干扰控制”功能 在今天信息充斥的数字…

BF算法,KMP算法

前言&#xff1a;今天我们来学习两种算法&#xff0c;BF算法和KMP算法。相信会让许多小伙伴们打开新世界的大门。 1 BF算法 实践是检验真理的唯一标准。举一个例子说明BF算法。现在我们要在一个主串中找子串的位置。那我们该如何解决这个问题呢&#xff1f;最简单的办法自然是…

【数据结构-哈希前缀】力扣2845. 统计趣味子数组的数目

给你一个下标从 0 开始的整数数组 nums &#xff0c;以及整数 modulo 和整数 k 。 请你找出并统计数组中 趣味子数组 的数目。 如果 子数组 nums[l…r] 满足下述条件&#xff0c;则称其为 趣味子数组 &#xff1a; 在范围 [l, r] 内&#xff0c;设 cnt 为满足 nums[i] % mod…

springboot打包找不到主类

1.打包jar包 idea中点击项目结构project stucture ,选择artfacts 2.

机器学习辅助复合材料预测,性能管理优化创新材料,这种王炸般的组合,还真是大开眼界!

在人工智能与复合材料技术融合的背景下&#xff0c;复合材料的研究和应用正迅速发展&#xff0c;创新解决方案层出不穷。从复合材料性能的精确预测到复杂材料结构的智能设计&#xff0c;从数据驱动的材料结构优化到多尺度分析&#xff0c;人工智能技术正以其强大的数据处理能力…

网络空间安全考研方向:5大专业值得选择,你喜欢哪一个?

网络空间安全考研方向包括网络与信息安全、信息安全工程、信息对抗技术、信息安全与管理、网络安全与执法等专业&#xff0c;旨在培养网络安全领域的高级专业人才&#xff0c;涵盖网络攻击与防御、信息加密与解密、信息安全评估与管理等核心知识与技能&#xff0c;为国家和社会…