【JavaEE】JUC的常见类

目录

Callable接口

ReentrantLock

用法

[ReentrantLock与 synchronized的区别](#ReentrantLock与 synchronized的区别)

信号量Semaphore

CountDownLatch

CountDownLatch的两个主要方法


JUC全称java.util.concurrent,这个包里的内容主要就是跟多线程有关的组件。

Callable接口

Callable也是一种创建线程的方式,适合于想让某个线程执行一个逻辑,并返回结果的时候,相比之下,Runable不关注结果

Callable<Integer> callable=new Callable<Integer>() {

@Override

public Integer call() throws Exception {

return null;

}

}

  • Callable<返回结果的类型>
  • call()是Callable的核心方法,与Runable中的run()一样

完整的示例:

//定义了任务

Callable<Integer> callable=new Callable<Integer>() {

@Override

public Integer call() throws Exception {

int sum = 0;

for (int i = 0; i < 100; i++) {

sum += i;

}

return sum;

}

};

//把任务放到线程中进行执行
FutureTask<Integer> futureTask=new FutureTask<>(callable);

Thread t=new Thread(futureTask);

t.start();

//此处的get就能获取到callable里面的返回结果

//由于线程是并发执行的,执行到主线程get的时候,t线程可能还没有执行完

//没执行完的话,get就会阻塞

System.out.println(futureTask.get());

  • FutureTask用来保存Callable的返回结果

ReentrantLock

ReentrantLock也是一个可重入锁,使用效果和synchronized是类似的;

优势:

  1. ReentrantLock在加锁的时候,有两种方式:lock,tryLock,提供更多的操作空间
  2. ReentrantLock提供了公平锁的实现,默认情况下是非公平锁
  3. ReentrantLock提供了更强大的等待通知机制,搭配了Condition类,实现等待通知

用法

  • lock():加锁,如果获取不到锁就死等
  • trylock():超时时间,加锁,如果获取不到锁,等待一定的时间后,放弃加锁
  • unlock():解锁

ReentrantLock lock=new ReentrantLock();

lock.lock();

try{

}

finally {

lock.unlock();

}

ReentrantLock与 synchronized的区别

  • synchronized是一个关键字,是JVM内部实现的,ReentrantLock是标准库中的一个类,是在JVM外实现的
  • synchronized使用时不需要手动释放锁,ReentrantLock使用时需要手动释放锁
  • synchronized在申请锁失败时,会死等,ReentrantLock可以通过trylock的方式等待一段时间后放弃
  • synchronized是非公平锁,ReentrantLock默认是非公平锁,可以通过构造方法传入一个true开启公平锁模式

信号量Semaphore

信号量,就是一个计数器,描述"可用资源"的个数

  • 申请一个资源,让计数器-1(P操作)
  • 释放一个资源,让计数器+1(V操作)

锁本质上就属于一种特殊的信号量,可以看成可用资源为1的信号量,二元信号量(只有加锁和解锁操作)

Semaphore semaphore=new Semaphore(5);

semaphore.acquire();

System.out.println("P操作");

semaphore.acquire();

System.out.println("P操作");

semaphore.release();

System.out.println("V操作");

CountDownLatch

适用于多个线程完成一系列任务的时候,用来衡量任务的进度是否完成,比如需要把一个大的任务,拆分成多个小的任务,让这些任务并发的去执行,就可以使用CountDownLatch来判定说当前的这些任务是否都完成了。

CountDownLatch的两个主要方法

  • await,调用的时候会阻塞,就会等待其他的线程完成任务,所有的线程完成了任务之后,此时这个await才会继续往下走
  • countDown,通知CountDownLatch,我当前这个子任务已经完成了

//await会在10次调用完countDown之后才能继续执行

CountDownLatch countDownLatch=new CountDownLatch(10);

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

{

Thread t=new Thread(()->{

int id=i;

System.out.println("thread "+id);

try {

Thread.sleep(1000);

} catch (InterruptedException e) {

throw new RuntimeException(e);

}

countDownLatch.countDown();

});

}

countDownLatch.await();

System.out.println("所有任务都完成了");

相关推荐
_.Switch9 分钟前
Python 自动化运维持续优化与性能调优
运维·开发语言·python·缓存·自动化·运维开发
徐*红9 分钟前
java 线程池
java·开发语言
尚学教辅学习资料9 分钟前
基于SSM的养老院管理系统+LW示例参考
java·开发语言·java毕设·养老院
2401_8576363910 分钟前
计算机课程管理平台:Spring Boot与工程认证的结合
java·spring boot·后端
1 9 J12 分钟前
Java 上机实践4(类与对象)
java·开发语言·算法
Code apprenticeship12 分钟前
Java面试题(2)
java·开发语言
J不A秃V头A15 分钟前
Python爬虫:获取国家货币编码、货币名称
开发语言·爬虫·python
憨子周1 小时前
2M的带宽怎么怎么设置tcp滑动窗口以及连接池
java·网络·网络协议·tcp/ip
霖雨3 小时前
使用Visual Studio Code 快速新建Net项目
java·ide·windows·vscode·编辑器
SRY122404193 小时前
javaSE面试题
java·开发语言·面试