【Java多线程】面试常考 —— JUC(java.util.concurrent) 的常见类

这篇具有很好参考价值的文章主要介绍了【Java多线程】面试常考 —— JUC(java.util.concurrent) 的常见类。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

目录

1、JUC(java.util.concurrent)

1.1、Callable 接口

1.2、ReentrantLock 可重入锁

1.3、Semaphore 信号量

1.4、CountDownLatch



【Java多线程】面试常考 —— JUC(java.util.concurrent) 的常见类,java,面试,开发语言,笔记,JUC

1、JUC(java.util.concurrent)

这是java中的一个包,存放着多线程编程中常见的一些类。

1.1、Callable 接口

【Java多线程】Thread类的基本用法-CSDN博客https://blog.csdn.net/zzzzzhxxx/article/details/136121421?spm=1001.2014.3001.5501

往期的博客中对创建线程的几种方式进行过说明,有如下几种:

1、继承 Thread(包含了匿名内部类的方式)

2、实现 Runnable(包含了匿名内部类的方式)

3、基于 lambda 表达式

其实除了以上方式,还有一个方式可以创建线程,即基于 Callable 的创建方法

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

        //引入 FutureTask 类,作为 Thread 和 callable 的 “粘合剂”
        FutureTask<Integer> futureTask = new FutureTask<>(callable);
        Thread t = new Thread(futureTask);
        t.start();

        //取餐号,使用 futureTask 获取到结果,具有阻塞功能
        System.out.println(futureTask.get());
    }

基于 Callable 同样可以创建线程,那么它与 Runnable 有什么区别呢?

  • Runnable 关注的是整个过程,不关注执行结果,Runnable 提供的是 run 方法,返回值类型是 void。
  • Callable 关注的是执行结果,Callable 提供的是 call 方法,返回值就是线程执行任务得到的结果。

因此,如果编写多线程代码时,希望关注线程中代码的返回值时,此时使用基于 Callable 实现的线程更为方便。

1.2、ReentrantLock 可重入锁

与 synchronized 有区别,上古时期的 Java,synchronized 还不够强壮,功能也不够强大,也没有各种优化,当时 ReenactmentLock 就是实现可重入锁的选择。

ReenactmentLock 的用法具有传统锁的风格,上锁使用lock(),解锁使用unlock()。
由于需要手动解锁,触发 return 或者 异常时 容易忘记解锁,为了避免这种问题,正确使用 ReenactmentLock 时需要把 unlock 操作放到 finally 中。

问题:既然有 synchronized,为啥还要用 ReenactmentLock ?(面试题)

1、ReenactmentLock 提供了 tryLock 操作。

  • lock 尝试进行加锁,如果加锁不成,就会阻塞;
  • tryLock 尝试进行加锁,如果加锁不成,不阻塞,直接返回 false;(更多的可操作空间)

2、ReenactmentLock 提供了公平锁的实现,通过队列记录加锁线程的先后顺序。

3、搭配的等待通知机制不同。

  • 对于 synchronized,搭配 wait/notify
  • 对于 ReenactmentLock,搭配 Condition 类,功能比 wait/notify 略强一些

1.3、Semaphore 信号量

首先举例说明 semaphore 信息量的概念:停车场入口立着的那个显示屏,每有一辆车进入停车场显示屏就会显示剩余车位减1,每有一辆车从停车场出去,显示屏上显示的剩余车辆就会加1,当显示屏上的剩余车位为0时,停车场入口的栏杆就不会再打开,车辆就无法进入停车场了,直到有一辆车从停车场出去为止。

而这个场景中提及的“显示屏”这个东西,就可以认为是信息量,表示“可用资源的个数”,申请一个可用资源,就会使数字 - 1,这个操作就成为 p 操作,释放一个可用资源,就会使数字 + 1,这个操作就是 v 操作,如果信息量数值为 0,继续 p 操作就会进行阻塞。

信号量是更广义的“锁”。所谓锁,本质上也是一种特殊的信息量,可以把锁认为是计数值为 1 的信息量。释放状态,就是1,加锁状态就是0,对于这种非0即1的信息量,称为“二元信息量”。除此之外,同样 Semaphore 也可以用来实现生产者消费者模型

acquire() 表示申请资源(P操作);
release() 表示释放资源(V操作);

使用 semaphore 模式锁并使用模拟锁实现两个线程同时对 i 自增 5w 的操作

package thread;

import java.util.concurrent.Semaphore;

public class ThreadDemo38 {
    private static int count = 0;

    public static void main(String[] args) throws InterruptedException {
        //信息量设置为 1
        Semaphore semaphore = new Semaphore(1);

        Thread t1 = new Thread(() -> {
            for (int i = 0; i < 50000; i++) {
                try {
                    semaphore.acquire();
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
                count++;
                semaphore.release();
            }
        });
        Thread t2 = new Thread(() -> {
            for (int i = 0; i < 50000; i++) {
                try {
                    semaphore.acquire();
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
                count++;
                semaphore.release();
            }
        });
        t1.start();
        t2.start();
        t1.join();
        t2.join();

        System.out.println("count = " + count);
    }
}

1.4、CountDownLatch

针对特定场景下解决问题的小工具。作用:同时等待 N 个任务执行结束。

比如,多线程执行一个任务,把大的任务拆分成几个部分,由每个线程分别执行。“多线程下载”:idm,比特彗星

package thread;

import java.util.Random;
import java.util.concurrent.CountDownLatch;

public class ThreadDemo39 {
    public static void main(String[] args) throws InterruptedException {
        // 1. 此处构造方法中写 10, 意思是有 10 个线程/任务
        CountDownLatch latch = new CountDownLatch(10);

        // 创建出 10 个线程负责下载.
        for (int i = 0; i < 10; i++) {
            int id = i;
            Thread t = new Thread(() -> {
                Random random = new Random();
                // [0, 5)
                int time = (random.nextInt(5) + 1) * 1000;
                System.out.println("线程 " + id + " 开始下载");
                try {
                    Thread.sleep(time);
                } catch (InterruptedException e) {
                    throw new RuntimeException(e);
                }
                System.out.println("线程 " + id + " 结束下载");
                // 2. 告知 countDownLatch 我执行结束了.
                latch.countDown();
            });
            t.start();
        }

        // 3. 通过这个 await 操作来【等待所有任务结束】. 也就是 countDown 被调用 10 次了.
        latch.await();
        System.out.println("所有任务都已经完成了!");
    }
}
  • 每个任务执行完毕,都调用 latch.countDown()。在 CountDownLatch 内部的计数器同时自减。
  • 主线程中使用 latch.await(); 阻塞等待所有任务执行完毕,相当于计数器为 0 了。

【Java多线程】面试常考 —— JUC(java.util.concurrent) 的常见类,java,面试,开发语言,笔记,JUC

【博主推荐】 

【网络原理】TCP 协议中比较重要的一些特性(三)-CSDN博客https://blog.csdn.net/zzzzzhxxx/article/details/136597348?spm=1001.2014.3001.5501【Java多线程】关于多线程的一些案例 —— 单例模式中的饿汉模式和懒汉模式以及阻塞队列-CSDN博客https://blog.csdn.net/zzzzzhxxx/article/details/136688859?spm=1001.2014.3001.5501【网络原理】TCP 协议中比较重要的一些特性(二)-CSDN博客https://blog.csdn.net/zzzzzhxxx/article/details/136579333?spm=1001.2014.3001.5501

如果觉得作者写的不错,求给博主一个大大的点赞支持一下,你们的支持是我更新的最大动力!文章来源地址https://www.toymoban.com/news/detail-846589.html

如果觉得作者写的不错,求给博主一个大大的点赞支持一下,你们的支持是我更新的最大动力!

如果觉得作者写的不错,求给博主一个大大的点赞支持一下,你们的支持是我更新的最大动力!

到了这里,关于【Java多线程】面试常考 —— JUC(java.util.concurrent) 的常见类的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 如何解决java.util.concurrent.RejectedExecutionException: Task java.util.concurrent.FutureTask@7566d7cf r...

    Java中的 java.util.concurrent.RejectedExecutionException 异常表示无法将任务提交到线程池中执行。这通常是因为线程池处于关闭状态或者已经达到了最大线程数,无法再接受新的任务。 要解决这个异常,你可以考虑以下几种方法: 检查线程池的状态,确保它处于可以接受新任务的状态

    2024年02月13日
    浏览(54)
  • java.util.concurrent.Executionexception 异常

    今天运行时发生了如下报错。自己捣鼓半天也没发现问题出在哪儿,感谢大佬的帮助,记录下来防止再犯。。 caused by org.apache.flink.client.program.programInvocationException: Job failed。程序调用异常。网上找了很多解决方法,都没有能够解决这个问题。 直到在报错中发现了这一行: C

    2024年02月19日
    浏览(40)
  • 已解决java.util.concurrent.ExecutionException异常的正确解决方法,亲测有效!!!

    已解决java.util.concurrent.ExecutionException异常的正确解决方法,亲测有效!!! java.util.concurrent.ExecutionException java.util.concurrent.ExecutionException是Java多线程编程中常见的异常之一,它表示在执行一个Callable或者Runnable任务时,任务抛出了一个异常。 下滑查看解决方法 具体解决方法可

    2024年02月11日
    浏览(37)
  • 关于报错java.lang.reflect.InaccessibleObjectException: Unable to make field private java.util.concurrent

    java.lang.reflect.InaccessibleObjectException: Unable to make field private java.util.concurrent.Callable java.util.concurrent.FutureTask.callable accessible: module java.base does not \\\"opens java.util.concurrent\\\" to unnamed module @32eebfca 假如报这种错误,只需要在Run-Edit Configurations-Modify Options-add VM options中加  假如你报的是下面

    2024年02月15日
    浏览(92)
  • 【问题已解决】Unrecognized option: --add-opens=java.base/java.util.concurrent.atomic=ALL-UNNAMED

    今天在创建java项目时,运行报错,说无法成功创建java程序。 Unrecognized option: --add-opens=java.base/java.util.concurrent.atomic=ALL-UNNAMED Error: Could not create the Java Virtual Machine. Error: A fatal exception has occurred. Program will exit. 解决办法: 1、使用最新的jdk版本 2、在第三处,选择最新的jdk版本

    2024年02月13日
    浏览(51)
  • 【JUC-1】java多线程线程基础知识

    继承Thread类. 实现Runable接口. 实现Callable接口. Runable/Callable接口的实现, 都是重写其中的run/call方法, 实现任务逻辑, 再由线程执行器(可以是Thread类,也可以是线程池)并发执行run/call的逻辑. 而Thread类中的包含start方法, 可以控制线程启动,执行任务. 当发生线程上下文切换时, 操作系

    2024年02月11日
    浏览(40)
  • Java基础-多线程&JUC-线程池和自定义线程池

    主要核心原理 不推荐Executors创建没有上线的线程池,建议使用自定义的线程池; Java工具类创建线程池; 当只有3个任务时,直接上处理机运行; 当有6个任务时,任务1-3上处理机运行,任务4-6进入阻塞队列等待; 当有9个任务时,任务1-3上处理机运行,任务4-6进入阻塞队列等

    2024年02月12日
    浏览(37)
  • idea启动报错Internal error. Please refer to https://jb.gg/ide/critical-startup-errors java.util.concurr

    java.util.concurrent.CompletionException: org.picocontainer.PicoRegistrationException: Key com.tang.intellij.lua.luacheck.LuaCheckSettings duplicated) Internal error. Please refer to https://jb.gg/ide/critical-startup-errors java.util.concurrent.CompletionException: org.picocontainer.PicoRegistrationException: Key com.tang.intellij.lua.luacheck.LuaCheckSetti

    2023年04月17日
    浏览(73)
  • Java进阶(ConcurrentHashMap)——面试时ConcurrentHashMap常见问题解读 & 结合源码分析 & 多线程CAS比较并交换 初识

    List、Set、HashMap作为Java中常用的集合,需要深入认识其原理和特性。 本篇博客介绍常见的关于Java中线程安全的ConcurrentHashMap集合的面试问题,结合源码分析题目背后的知识点。 关于List的博客文章如下: Java进阶(List)——面试时List常见问题解读 结合源码分析 关于的Set的博

    2024年02月06日
    浏览(61)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包