javaEE 初阶 — JUC(java.util.concurrent) 的常见类

news2025/1/17 3:52:12

文章目录

  • 1. Callable 接口
    • 1.1 Callable 的用法
  • 2. ReentrantLock
    • 2.1 ReentrantLock 的缺陷
    • 2.1 ReentrantLock 的优势
  • 3. 原子类
  • 4. 信号量 Semaphore
  • 5. CountDownLatch
  • 6. 相关面试题

1. Callable 接口


类似于 Runnable 一样。
Runnable 用来描述一个任务,描述的任务没有返回值。
Callable 也是用来描述一个任务,描述的任务是有返回值的。

如果需要使用一个线程单独的计算出某个结果,此时使用 Callable 是比较合适的。

1.1 Callable 的用法


代码示例:创建线程计算 1 + 2 + 3 + … + 1000,不使用 Callable 版本

  • 创建一个类 Result,包含一个 sum 表示最终结果,lock 表示线程同步使用的锁对象。
  • main 方法中先创建 Result 实例,然后创建一个线程 t。在线程内部计算 1 + 2 + 3 + … + 1000。
  • 主线程同时使用 wait 等待线程 t 计算结束。(注意,如果执行到 wait 之前,线程 t 已经计算完了,就不
    必等待了)。
  • 当线程 t 计算完毕后,通过 notify 唤醒主线程,主线程再打印结果。
static class Result {
    public int sum = 0;
    public Object lock = new Object();
}

public static void main(String[] args) throws InterruptedException {
    Result result = new Result();
    
    Thread t = new Thread() {
        @Override
        public void run() {
            int sum = 0;
            for (int i = 1; i <= 1000; i++) {
                sum += i;
            }
            synchronized (result.lock) {
                result.sum = sum;
                result.lock.notify();
            }
        }
    };
    t.start();

    synchronized (result.lock) {
        while (result.sum == 0) {
            result.lock.wait();
        }
        System.out.println(result.sum);
    }
}


可以看到,上述代码需要一个辅助类 Result,还需要使用一系列的加锁和 wait notify 操作,代码复
杂,容易出错。


代码示例:创建线程计算 1 + 2 + 3 + … + 1000,使用 Callable 版本

package thread;

import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask;

public class ThreadDemo9 {

    public static void main(String[] args) throws InterruptedException, ExecutionException {
        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 thread = new Thread(futureTask);
        thread.start();

        Integer result = futureTask.get();
        System.out.println(result);
    }
}


call() 相当于是 Runnable 的 run() 方法,run方法返回的是 void 此处返回值泛型参数。

FutureTask 表示这是未来的任务。
可以把 FutureTask 简单理解为点餐时的小票,这个小票就是 FutureTask。
后面我们可以随时凭这张小票去查看自己点的餐做出来了没。

get() 方法就是获取结果。
get 会发生阻塞,直到 callable 执行完毕,get 才阻塞完成,才获取到结果。

可以看到,使用 Callable 和 FutureTask 之后,代码简化了很多,也不必手动写线程同步代码了。

2. ReentrantLock


这里的 ReentrantLock 是标准库给我们提供的另一种锁,也是可重入的。

synchronized 是直接基于代码块的方式来加锁和解锁的。
ReentrantLock 使用了 lock 方法和 unlock 方法来加锁和解锁。

package thread;

import java.util.concurrent.locks.ReentrantLock;

public class ThreadDemo10 {
    
    public static void main(String[] args) {
        ReentrantLock reentrantLock = new ReentrantLock();
        reentrantLock.lock();
        
        reentrantLock.unlock();
    }
}


reentrantLock.lock()reentrantLock.unlock() 之间的代码就被锁给保护起来了。
但是这样的写法有很大的弊端:

unlock 有可能会执行不到

2.1 ReentrantLock 的缺陷


如果代码中间存在 return 或者异常,就有可能会导致 unlock 不能顺利执行。

package thread;

import java.util.concurrent.locks.ReentrantLock;

public class ThreadDemo10 {

    public static void main(String[] args) {
        ReentrantLock reentrantLock = new ReentrantLock();
        int num = 0;
        reentrantLock.lock();

        if (num == 0) {
            return;
        }

        if (num == 1) {
            return;
        }

        throw new Exception();

        reentrantLock.unlock();
    }
}


这时就要把 unlock 写在 finally

package thread;

import java.util.concurrent.locks.ReentrantLock;

public class ThreadDemo10 {

    public static void main(String[] args) {
        ReentrantLock reentrantLock = new ReentrantLock();
        int num = 0;
        try {
            if (num == 0) {
                return;
            }

            if (num == 1) {
                return;
            }

            throw new Exception();
        } finally {
            reentrantLock.unlock();
        }
    }
}

2.1 ReentrantLock 的优势


1、ReentrantLock 提供了公平锁版本的实现。

  ReentrantLock reentrantLock = new ReentrantLock(true);
 ReentrantLock reentrantLock = new ReentrantLock();

ReentrantLock 括号里加上 true 表示这是一个公平锁。
什么都不加,或者加 false 表示这是一个非公平的锁。


2、更加灵活的阻塞等待方式

对于 synchronized 来说,提供的加锁操作就是 “死等” ,只要获取不到锁,就会一直等待。
而 ReentrantLock 提供更加灵活的等待方式:trylock

这里的 trylock 分为有参数和无参数的版本。

无参数版本:能加锁就加,加不上就放弃。

有参数的版本:指定了超时时间,加不上锁就等待一段时间。如果时间到了也没加上就放弃。


3、ReentrantLock 提供了一个更加强大的等待机制。

synchronized 搭配的是 wait 和 notify ,notify 的时候随机唤醒一个 wait 的线程。

ReentrantLock 搭配的是一个 Condition 类,进行唤醒的时候可以指定唤醒的线程。

3. 原子类


原子类内部用的是 CAS 实现的,所以性能要比加锁实现 i++ 高很多。

原子类有以下几个:

  • AtomicBoolean
  • AtomicInteger
  • AtomicIntegerArray
  • AtomicLong
  • AtomicReference
  • AtomicStampedReferenc

以 AtomicInteger 举例,常见方法有:

  • addAndGet(int delta); i += delta;
  • decrementAndGet(); --i;
  • getAndDecrement(); i–;
  • incrementAndGet(); ++i;
  • getAndIncrement(); i++

基于 CAS 确实是更高效的解决了线程安全问题,但是 CAS 不能代替锁。
CAS 的适用范围是有限的,不像锁的适用范围那么广。

4. 信号量 Semaphore


操作系统上提到的信号量和此处这个信号量是一个东西,只不过此处的这个信号量是 java 把操作系统原生的信号量封装了一下。

信号量在生活中经常可以见到。

比如说停车场,因为停车场的空闲位置个数都是固定的,当位置到达上限的时候就不能停车了。

停车场的入口位置会有一个牌子,牌子上显示空闲位置的个数。
每当有车进去,牌子上的显示的个数就减少一个;有车出来,个数就加一个。
这个牌子就相当于是一个计数器,当这个计数器为 0 的时候,也就是停车场空闲位置达到上限了。

当没有位置的时候要停车,就只能在这里等待或者去别处找停车场。

信号量本质上就是一个 计数器,描述了 “可用资源的个数”

P操作(acquire) :申请一个可用资源,计时器就要 -1。
V操作() :释放一个可用资源,计数器个数就要 +1。

如果此时的计数器为 0 了,继续执行 P 操作,就会发生阻塞等待。

考虑一个计数初始值为 1 的信号量。
针对这个信号量的值,就只有1 和 0 两种取值(信号量不能是负的)

执行一次 P 操作,1 就变成了 0 。
执行一次 V 操作,0 就变成了 1 。

如果已经执行过一次 P 操作了,继续执行 P 操作,就会阻塞等待。

有没有让你想到锁,(锁可以视为是计数器为 1 的信号量,二元信号量)
锁是一种信号量的特殊情况,信号量是锁的一般表达。

Semaphore 的使用

package thread;

import java.util.concurrent.Semaphore;

public class ThreadDemo11 {

    public static void main(String[] args) throws InterruptedException{
        Semaphore semaphore = new Semaphore(3); //指定计数器个数是3
        semaphore.acquire(); //执行P操作
        System.out.println("P操作一次");

        semaphore.acquire(); //执行P操作
        System.out.println("P操作一次");

        semaphore.acquire(); //执行P操作
        System.out.println("P操作一次");

        semaphore.acquire(); //执行P操作
        System.out.println("P操作一次");
    }
}




当前的计数器个数是 3 ,当计数器为 0 的时候继续执行 acquire 操作就会阻塞等待。

package thread;

import java.util.concurrent.Semaphore;

public class ThreadDemo11 {

    public static void main(String[] args) throws InterruptedException{
        Semaphore semaphore = new Semaphore(3); //指定计数器个数是3
        semaphore.acquire(); //执行P操作
        System.out.println("P操作一次");

        semaphore.acquire(); //执行P操作
        System.out.println("P操作一次");

        semaphore.acquire(); //执行P操作
        System.out.println("P操作一次");

        semaphore.release(); //执行V操作

        semaphore.acquire(); //执行P操作
        System.out.println("P操作一次");
    }
}




当计数器为 0 的时候,执行 V 操作后,计数器的个数就加了一个。

5. CountDownLatch


假如有一个跑步比赛。



这场跑步比赛,开始的时间确定的(发号枪)
但是结束的时间是不明确的。(所有的选手都冲过终点后)

为了等待跑步比赛的结束,就引入了这个 CountDownLatch
主要是有两个方法:

1、await (a 是 all wait 是等待),主线程来调用这个方法。

2、countDown 表示选手冲过了终点线。


CountDownLatch 在构造的时候,会指定一个计数(选手的个数)

例如,指定四个选手进行比赛,初始情况下,调用 await 就会阻塞。
每个选手都冲过终点,都会调用 countDown 方法。

第三次调用 countDown ,await 没有任何影响。
第四次调用 countDown ,await 就会被唤醒,返回。(解除阻塞队列)
此时就可以认为是整个比赛都结束了。

package thread;

import java.util.concurrent.CountDownLatch;

public class ThreadDemo12 {

    public static void main(String[] args) throws InterruptedException{
        CountDownLatch latch = new CountDownLatch(10);
        Runnable runnable = new Runnable() {
            @Override
            public void run() {
                try {
                    Thread.sleep(1000);
                    latch.countDown();
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        };
        for (int i = 0; i < 10; i++) {
            new Thread(runnable).start();
        }

        // 必须等到 10 人全部回来
        latch.await();
        System.out.println("比赛结束");
    }
}

6. 相关面试题


1、介绍下 Callable 是什么

Callable 是一个 interface,相当于把线程封装了一个 “返回值”。
方便程序猿借助多线程的方式计算结果。

Callable 和 Runnable 相对,都是描述一个 “任务”。
Callable 描述的是带有返回值的任务,Runnable 描述的是不带返回值的任务。

Callable 通常需要搭配 FutureTask 来使用。FutureTask 用来保存 Callable 的返回结果。
因为 Callable 往往是在另一个线程中执行的,啥时候执行完并不确定。
FutureTask 就可以负责这个等待结果出来的工作。


2、线程同步的方式有哪些?

synchronized,ReentrantLock,Semaphore 等都可以用于线程同步。


3、为什么有了 synchronized 还需要 juc 下的 lock?

以 juc 的 ReentrantLock 为例。

  • synchronized 使用时不需要手动释放锁,ReentrantLock 使用时需要手动释放,使用起来更
    灵活。
  • synchronized 在申请锁失败时,会死等,ReentrantLock 可以通过 trylock 的方式等待一段时
    间就放弃。
  • synchronized 是非公平锁,ReentrantLock 默认是非公平锁。可以通过构造方法传入一个
    true 开启公平锁模式。
  • synchronized 是通过 Object 的 wait / notify 实现等待-唤醒。每次唤醒的是一个随机等待的
    线程。ReentrantLock 搭配 Condition 类实现等待-唤醒,可以更精确控制唤醒某个指定的线
    程。

4、AtomicInteger 的实现原理是什么?

基于 CAS 机制。

伪代码如下:

class AtomicInteger {
    private int value;
    public int getAndIncrement() {
        int oldValue = value;
        while ( CAS(value, oldValue, oldValue+1) != true) {
            oldValue = value;
        }
       return oldValue;
    }
}



5、信号量听说过么?之前都用在过哪些场景下?

信号量,用来表示 “可用资源的个数”,本质上就是一个计数器。

使用信号量可以实现 “共享锁”,比如某个资源允许 3 个线程同时使用,那么就可以使用 P 操作作为
加锁,V 操作作为解锁,前三个线程的 P 操作都能顺利返回,后续线程再进行 P 操作就会阻塞等待,
直到前面的线程执行了 V 操作。


6、解释一下 ThreadPoolExecutor 构造方法的参数的含义

参考关于 ThreadPoolExecutor 的篇章

篇章链接:

https://blog.csdn.net/m0_63033419/article/details/128586070?spm=1001.2014.3001.5501

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

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

相关文章

【Spring源码】21. 关于循环依赖的N个问题

完成了applyMergedBeanDefinitionPostProcessors()方法&#xff0c;后面有一段关于判断Bean是否需要提前曝光的逻辑&#xff08;如下图红框框中部分&#xff09;在这段逻辑中涉及到了著名的循环依赖&#xff0c;提到循环依赖基本必讲三级缓存&#xff0c;好吧&#xff0c;这篇就…

CANOpen中SDO和PDO的COB-ID理解

CAN 总线是一种串行通信协议&#xff0c;具有较高的通信速率的和较强的抗干扰能力&#xff0c;可以作为现场总线应用于电磁噪声较大的场合。由于 CAN 总线本身只定义ISO/OSI 模型中的第一层&#xff08;物理层&#xff09;和第二层&#xff08;数据链路层&#xff09;&#xf…

(8)go-micro微服务Mysql配置

文章目录一 gorm介绍二 gorm安装1.1 下载依赖1.2 使用MySQL驱动三 CURD操作1. 查询1.1 单行查询1.2 多行查询2. 插入数据3. 更新数据4. 删除数据四 初始化连接五 使用六 最后一 gorm介绍 Go语言中的database/sql包提供了保证SQL或类SQL数据库的泛用接口&#xff0c;并不提供具…

redis: jedis连接超时(需要手动注入连接超时检测的配置)

相关版本说明 服务端&#xff1a; redis_version: 6.2.8 客户端&#xff1a; springBoot: 2.7.7 jedis: 3.8.0 问题 偶发redis连接超时&#xff0c;刷新就又好了&#xff0c;服务日志错误信息如下&#xff1a; JedisConnectionException: Unexpected end of stream.原因 …

Linux利用httpd搭建局域网yum源

本例环境&#xff1a;vmwareworkstation16 proCentOS7.9 mast节点&#xff1a;192.168.195.110 用于配置httpd并发布本地yum源 node节点&#xff1a;192.168.195.111 用于验证mast节点的yum源是否可用 思路&#xff1a;1.在mast节点挂载/上传镜像后配置本地yum源 2.利用本…

JSP三种脚本

脚本可以编写Java语句、变量、方法或表达式。 1.普通脚本 语法: <% Java代码%> <% page contentType"text/html;charsetUTF-8" language"java" %><html><head> <title>Title</title></head><body>&l…

对u盘的分区进行删除和格式化

一、说明 当usb盘&#xff0c;或者SD卡用作启动盘后&#xff0c;将出现多个盘符、多个分区&#xff1b;若将此盘重新当文件盘&#xff0c;需要删除以前的分区&#xff0c;并重新格式化后&#xff0c;才能使用。 二、使用Diskpart在Windows 10中对USB进行分区删除 2.1 尝试磁盘…

重启之后,台式机网络不能连接怎么办

目录 1.问题 2.排查过程 3.心得 1.问题 前天电脑意外断电后,再启动发现网络变成了未连接状态.查看本地连接显示已启动,但IPv4和IPv6未连接.当时做了一些尝试,没有收到效果,直到今天问题才得以解决. 2.排查过程 Windows网络诊断为:DNS服务器未响应.后来花了一部分时间在DNS…

ruoyi-vue集成magic-api(一)

ruoyi虽然带了强大的代码生成器&#xff0c;面对比较通用的CRUD还是游刃有余的&#xff0c;但在项目开发阶段&#xff0c;需求总是经常变化的&#xff0c;数据结构和逻辑也经常变化&#xff0c;我们需要的是快速验证功能逻辑&#xff0c;代码生成器可帮不上忙&#xff0c;每次需…

一、java编写登录功能

java编写登录功能 文章目录java编写登录功能前言编程学习记录一、登录逻辑简述二、代码实现1.创建USER表2.前端代码3.创建User类4.创建LoginServlet类5.创建JDBCUtils类6.创建UserDao类7.创建FailServlet类9.创建SuccessServlet 类11.配置tomcat 服务12.启动服务前言 编程学习…

SpringCloud Netfllix复习之Hystrix

文章目录写作背景Hystrix是什么Hystrix的核心功能上手实战RestTemplate整合HystrixOpenFeign整合HystrixOpenFeign与Hystrix整合的各种参数如何配置&#xff1f;源码验证基于HystrixCommand注解实现熔断源码分析初始化资源线程池的源码OpenFeign与Hystrix整合执行请求的源码写作…

Java多线程:创建多线程的“四种“ 方式

Java多线程&#xff1a;创建多线程的"四种" 方式 每博一文案 白马笑西风写道&#xff1a;江南有杨柳&#xff0c;有燕子&#xff0c;金鱼......汉人中有的是英俊勇武的少年&#xff0c;倜傥潇洒的少年...... 但这个美丽的姑娘就像故高昌国人那样固执&#xff1a;&qu…

buctoj-2023寒假集训-进阶训练赛(八)

问题 A: 分离出整数n从右边数第k个数字&#xff0c;递归实现 题目描述 在程序中定义一函数digit(n,k)&#xff0c;它能分离出整数n从右边数第k个数字。 输入 正整数n和k。 输出 第k个数字(若不存在则输出0&#xff09; 样例输入 31859 3 样例输出 8 #include<bits/stdc.h&g…

电商直播小程序核心功能有哪些?电商直播小程序代码分析

一个优质的电商直播小程序&#xff0c;必须带有后台管理&#xff0c;模块功能分工明确&#xff0c;可以让商家及时管理商品。在管理后台端又分为会员、商品、订单、店铺、直播、分销、优惠券、物流、数据等功能列表栏&#xff0c;基本功能较完善。下文小编将为大家讲解一下电商…

Linux命令行中 git 的使用

文章目录&#xff1a;什么是gitgitee新建仓库git提交代码1.同步远程仓库代码 - git pull2.查看本地仓库的状态 - git status3.添加代码到本地.git缓冲区 - git add4.推送代码到本地仓库.git中 - git commit5.同步本地仓库.git的内容到远程仓库 - git push什么是git Git 是一个…

2023年了,浏览器竟然还有新玩法,能看热搜能领券

在移动互联网时代&#xff0c;手机浏览器是手机中不可缺少的APP之一。我们经常使用手机浏览器查资料&#xff0c;看新闻&#xff0c;看小说等等。如今&#xff0c;手机浏览器的功能越来越强大&#xff0c;玩法也越来越多。最近&#xff0c;发现一款手机浏览器&#xff0c;竟然聚…

立创EDA入门3 通过51单片机最小板学习PCB设计

立创EDA入门3 通过51单片机最小板学习PCB设计一、本文目的二、原理图设计1. 新建工程&#xff0c;命名为51系统2. 各模块原理图3. 一些常用操作&#xff08;1&#xff09;放置普通元器件&#xff08;2&#xff09;封装、标签设置&#xff08;3&#xff09;在线库中查找元器件&a…

硬件系统工程师宝典(4)-----传输过程的信号要如何描述?

各位同学大家好&#xff0c;欢迎继续做客电子工程学习圈&#xff0c;今天我们继续来讲这本书&#xff0c;硬件系统工程师宝典。上篇我们说到为实现信号的有效传输&#xff0c;需要保证信号波形的完整和信号时序的完整&#xff0c;并且知道了从时域、频域两个角度去分析信号。那…

高压功率放大器在诱发肌电运动阈值对比研究中的应用

实验名称&#xff1a;经颅磁声刺激与经颅超声刺激诱发肌电运动阈值的对比研究 研究方向&#xff1a;生物医学 测试目的&#xff1a; 在脑科学与神经科学研究中&#xff0c;物理刺激是目前应用最广泛的电磁刺激技术。该技术利用变化的磁场诱发感应电流进而对神经组织进行调控&am…

递归算法实例应用(一)

递归算法实例应用&#xff08;一&#xff09; 递归简笔 递归和普通函数调用一样&#xff0c;都是通过函数栈实现。 以斐波那契数列递归调用为例 递归时函数调用栈的进栈、出栈过程可以由上述图示直观的体现出来&#xff0c; 因此可以得出递归的几个作用&#xff1a; ​ …