当前位置: 首页 > news >正文

【Java多线程】JUC其他常用组件

在这里插入图片描述

🔍 开发者资源导航 🔍
🏷️ 博客主页: 个人主页
📚 专栏订阅: JavaEE全栈专栏

Callable

我们知道在JAVA的线程中我们只能执行任务,而这个任务不能返回一个值,而Callable的出现就是为了弥补这一缺陷。

Callable:定义一个带有返回值的任务, 并没有真的在执行,需要搭配Thread对象来使用。

Thread本身不提供获取结果的方法,需要凭FurureTask对象来得到结果,这么做是为了解耦合。

public static void main(String[] args) throws ExecutionException, InterruptedException {Callable<Integer> callable = new Callable<Integer>() {@Overridepublic Integer call() throws Exception {Integer res = 0;for (int i = 0; i < 10; i++) {res += i;}return res;}};FutureTask<Integer> futureTask = new FutureTask<>(callable);Thread thread = new Thread(futureTask);thread.start();System.out.println(futureTask.get());
}
//匿名写法
FutureTask<Integer> futureTask = new FutureTask<>(()->{Integer res = 0;for (int i = 0; i < 10; i++) {res += i;}return res;
});

get方法用于获取FutureTask的返回值,如果FutureTask没有运行完成get方法会产生阻塞,直到线程运行完毕,get拿到返回结果。

ReentrantLock (可重入锁)

通过lock()和unlock()方法加解锁,效率比synchronized低,远古时期用的较多。

如果要使用ReentrantLock进行加锁,可以通过此方式防止忘记解锁

locker.lock();
try {//代码逻辑
} finally {locker.unlock();
}

ReentrantLock和synchronized的区别:

  1. synchronized 是 关键字(内部实现是 JVM 内部通过 C++ 实现的),ReentrantLock 标准库的类(Java)。
  2. synchronized 通过代码块控制加锁解锁,ReentrantLock 需要 lock/unlock 方法,需要注意 unlock 不被调用的问题。
  3. ReentrantLock 除了提供 lock, unlock 之外,还提供了一个方法trylock(),这个方法不会产生阻塞,加锁成功返回true,失败返回false,可以由调用者决定接下来怎么做。
  4. ReentrantLock提供了公平锁的实现,默认是非公平,需要在创建的时候传入true。
  5. ReentrantLock搭配的等待通知机制是Condition类,相比于wait notify来说要更加强大灵活一些。

semaphore(信号量)

信号量表示的是“可用资源的个数”,申请一个资源,计数器就会+1,释放一个就会-1,如果为0,继续申请就会陷入阻塞等待。

//设置初始数量
Semaphore semaphore = new Semaphore(5);
//申请一个资源
semaphore.acquire();
//获取多少个资源可以申请
System.out.println(semaphore.availablePermits());
//释放一个资源
semaphore.release();
System.out.println(semaphore.availablePermits());

作用:

  1. 限制同时访问某资源的线程数(如接口限流、数据库连接池)。
  2. 管理有限的资源(如线程池、数据库连接池)。

特殊场景:
当初始值为1时,等价于“锁”。

Semaphore mutex = new Semaphore(1);  // 二进制信号量void criticalSection() throws InterruptedException {mutex.acquire();try {// 临界区代码(同一时间仅一个线程可进入)} finally {mutex.release();}
}

CountDownLatch

在使用多线程,经常把一个大任务拆分成多个子任务,可以提高程序的效率。

那么我们如何判定这个大任务已经完成了呢?Java引入了CountDownLatch类帮助我们判定。

  1. 在构造时传入参数,表示有多少个任务。
  2. 在每个任务完成时,调用一次countDown方法,当次数足够时,说明任务已经全部完成。
  3. 在主线程中调用await方法会等待任务全部完成。
public static void main(String[] args) throws InterruptedException {CountDownLatch countDownLatch = new CountDownLatch(10);//重写工厂类方法ThreadFactory threadFactory = new ThreadFactory() {private int cnt = 0;@Overridepublic Thread newThread(Runnable r) {Thread thread = new Thread(r, "Thread" + cnt);thread.setDaemon(true);return thread;}};//使用工厂类创建线程池ExecutorService executorService = Executors.newFixedThreadPool(1, threadFactory);//提交任务for (int i = 0; i < 10; i++) {int id = i;executorService.submit(()->{System.out.println("开始" + Thread.currentThread().getName() + id);try {Thread.sleep(100);} catch (InterruptedException e) {throw new RuntimeException(e);}System.out.println("结束" + Thread.currentThread().getName() + id);countDownLatch.countDown();});}//等待全部完成countDownLatch.await();System.out.println("全部结束了");
}

感谢各位的观看Thanks♪(・ω・)ノ,如果觉得满意的话留个关注再走吧。

相关文章:

  • 系统架构设计(十八):ATAM
  • Linux的启动流程
  • Win11怎样禁止程序开机启动
  • vue3项目动态路由的相关配置踩坑记录
  • 每日算法 -【Swift 算法】Z 字形变换(Zigzag Conversion)详解与实现
  • 第二十章 TIM——基本定时器
  • Javascript 学习
  • leetcode2947. 统计美丽子字符串 I-medium
  • AGI大模型(32):LangChain实现RAG
  • 探秘 C++ 计数器类:从基础实现到高级应用
  • (Git) 稀疏检出(Sparse Checkout) 拉取指定文件
  • SpringCloud (3) 配置中心
  • 【AT32】 at32 软复位
  • 利用GeoTools实现导出MySQL地理数据表为shp格式
  • DHT11介绍
  • 对于Python虚拟环境的理解
  • YOLOv8模型剪枝笔记(DepGraph和Network Slimming网络瘦身)
  • C++(29):结构体
  • SpringBoot Day_03|数据校验|异常处理|日志级别|定时器
  • 位运算及其算法
  • 英文版网站制作/免费自媒体网站
  • 南通网站制作公司哪家好/四平网络推广
  • 那个网站适合学生做兼职/东莞谷歌推广公司
  • 有哪些做电子商务的网站/宁波网站推广怎么做
  • 郑州工装定制/进行优化
  • 女性时尚资讯+淘宝客模式系列网站源码/怎么优化网站性能