Java EE 进阶--多线程(二)

news2025/1/4 19:03:00

目录

一、JUC(java.util.concurrent) 的常见类

1.1 信号量 Semaphore

1.2 CountDownLatch

1.3 CyclicBarrier -循环栅栏

二、线程安全的集合类

 2.1 多线程环境使用 ArrayList

2.2 多线程环境使用队列

2.3 多线程环境使用哈希表

 三、死锁

3.1 死锁是什么

3.2 如何避免死锁

3.3 避免死锁的解决方案

四、ThreadLocal


一、JUC(java.util.concurrent) 的常见类

1.1 信号量 Semaphore

信号量 , 用来表示 " 可用资源的个数 ". 本质上就是一个计数器 .
理解信号量
可以把信号量想象成是停车场的展示牌 : 当前有车位 100 . 表示有 100 个可用资源 .
当有车开进去的时候 , 就相当于申请一个可用资源 , 可用车位就 -1 ( 这个称为信号量的 P 操作 )
当有车开出来的时候 , 就相当于释放一个可用资源 , 可用车位就 +1 ( 这个称为信号量的 V 操作 )
如果计数器的值已经为 0 , 还尝试申请资源 , 就会阻塞等待 , 直到有其他线程释放资源 .
Semaphore PV 操作中的加减计数器操作都是原子的 , 可以在多线程环境下直接使用 .
代码示例
创建 Semaphore 示例 , 初始化为 4, 表示有 4 个可用资源 .
acquire 方法表示申请资源 (P 操作 ), release 方法表示释放资源 (V 操作 )
创建 20 个线程 , 每个线程都尝试申请资源 , sleep 1 秒之后 , 释放资源 . 观察程序的执行效果 .

Semaphore semaphore = new Semaphore(4);
Runnable runnable = new Runnable() {
    @Override
    public void run() {
        try {
            System.out.println("申请资源");
            semaphore.acquire();
            System.out.println("我获取到资源了");
            Thread.sleep(1000);
            System.out.println("我释放资源了");
            semaphore.release();
       } catch (InterruptedException e) {
            e.printStackTrace();
       }
   }
};
for (int i = 0; i < 20; i++) {
    Thread t = new Thread(runnable);
    t.start();
}

 1.2 CountDownLatch

同时等待 N 个任务执行结束 .
好像跑步比赛,4 个选手依次就位,哨声响才同时出发;所有选手都通过终点,才能公布成绩

 CountDownLatch的作用,是可以设置所有的线程必须都到达某一个关键点然后再执行后续的操作

 

 应用场景:把一个大任务分成若干个小任务,或是等待一些前置资源时,可以考虑使用CountDownLatch

1.3 CyclicBarrier -循环栅栏

CountDownLatch的升级版,可以实现线程间的相互等待,计数重置

二、线程安全的集合类

原来的集合类 , 大部分都不是线程安全的 .

 

 2.1 多线程环境使用 ArrayList

1) 自己使用同步机制 (synchronized 或者 ReentrantLock)
前面做过很多相关的讨论了 . 此处不再展开 .不推荐
2) Vector, Stack, HashTable, 是线程安全的(不建议用), 其他的集合类不是线程安全的.
3)Collections.synchronizedList(new ArrayList);
synchronizedList 是标准库提供的一个基于 synchronized 进行线程同步的 List.
synchronizedList 的关键操作上都带有 synchronized

 实现方式是在普通集合对象外层又包裹了一层synchronized完成的线程安全,也不推荐

4) 使用 CopyOnWriteArrayList
CopyOnWrite 容器即 写时复制 的容器。
当我们往一个容器添加元素的时候,不直接往当前容器添加,而是先将当前容器进行 Copy
复制出一个新的容器,然后新的容器里添加元素,
添加完元素之后,再将原容器的引用指向新的容器。
这样做的好处是我们可以对 CopyOnWrite 容器进行并发的读,而不需要加锁,因为当前容器不会
添加任何元素。
所以 CopyOnWrite 容器也是一种读写分离的思想,读和写不同的容器。
优点 :
在读多写少的场景下 , 性能很高 , 不需要加锁竞争 .
缺点 :
1. 占用内存较多 .
2. 新写的数据不能被第一时间读取到 .
在多线程环境中如果需要使用集合类那么优先考虑 CopyOnWriteArrayList

 

2.2 多线程环境使用队列

1) ArrayBlockingQueue
基于数组实现的阻塞队列
2) LinkedBlockingQueue
基于链表实现的阻塞队列
3) PriorityBlockingQueue
基于堆实现的带优先级的阻塞队列
4) TransferQueue
最多只包含一个元素的阻塞队列

2.3 多线程环境使用哈希表

HashMap 本身不是线程安全的 .
在多线程环境下使用哈希表可以使用 :
  • Hashtable
  • ConcurrentHashMap
1) Hashtable
只是简单的把关键方法加上了 synchronized 关键字 .
这相当于直接针对 Hashtable 对象本身加锁 .
  • 如果多线程访问同一个 Hashtable 就会直接造成锁冲突.
  • size 属性也是通过 synchronized 来控制同步, 也是比较慢的.
  • 一旦触发扩容, 就由该线程完成整个扩容过程. 这个过程会涉及到大量的元素拷贝, 效率会非常低.

 

2) ConcurrentHashMap
相比于 Hashtable 做出了一系列的改进和优化 . Java1.8 为例
  • 读操作没有加锁(但是使用了 volatile 保证从内存读取结果), 只对写操作进行加锁. 加锁的方式仍然是用 synchronized, 但是不是锁整个对象, 而是 "锁桶" (用每个链表的头结点作为锁对象), 大大降低了锁冲突的概率.
  • 充分利用 CAS 特性. 比如 size 属性通过 CAS 来更新. 避免出现重量级锁的情况.  
  • 优化了扩容方式: 化整为零 发现需要扩容的线程, 只需要创建一个新的数组, 同时只搬几个元素过去. 扩容期间, 新老数组同时存在. 后续每个来操作 ConcurrentHashMap 的线程, 都会参与搬家的过程. 每个操作负责搬运一小部分元素.搬完最后一个元素再把老数组删掉. 这个期间, 插入只往新数组加. 这个期间, 查找需要同时查新数组和老数组
  • 典型的以空间换时间的用例

 


三、死锁

3.1 死锁是什么

死锁是这样一种情形:多个线程同时被阻塞,它们中的一个或者全部都在等待某个资源被释放。由于线程被无限期地阻塞,因此程序不可能正常终止。
举个栗子:车钥匙锁家里了,家里的钥匙锁车里了
为了进一步阐述死锁的形成 , 很多资料上会谈论到 " 哲学家就餐问题 ".
有个桌子 , 围着一圈 哲 , 桌子中间放着一盘意大利面 . 每个哲学家两两之间 , 放着一根筷子. 每个 哲 家 只做两件事 : 思考人生 或者 吃面条 . 思考人生的时候就会放下筷子 . 吃面条就会拿起左 右两边的筷子 ( 先拿起左边 , 再拿起右边 ).
如果 哲 家 发现筷子拿不起来了 ( 被别人占用了 ), 就会阻塞等待.
[关键点在这] 假设同一时刻 , 五个 哲 家 同时拿起左手边的筷子 , 然后再尝试拿右手的筷子 , 就会 发现右手的筷子都被占用了. 由于 哲 家 们互不相让 , 这个时候就形成了 死锁

 

死锁是一种严重的 BUG!! 导致一个程序的线程 " 卡死 ", 无法正常工作 !

 3.2 如何避免死锁

死锁产生的四个必要条件:
  • 互斥使用,即当资源被一个线程使用(占有)时,别的线程不能使用
  • 不可抢占,资源请求者不能强制从资源占有者手中夺取资源,资源只能由资源占有者主动释放。
  • 请求和保持,即当资源请求者在请求其他的资源的同时保持对原有资源的占有。
  • 循环等待,即存在一个等待队列:P1占有P2的资源,P2占有P3的资源,P3占有P1的资源。这样
就形成了一个等待环路。
当上述四个条件都成立的时候,便形成死锁。当然,死锁的情况下如果打破上述任何一个条件,便可让死锁消失。
其中最容易破坏的就是 " 循环等待".
最常用的一种死锁阻止技术就是 锁排序 . 假设有 N 个线程尝试获取 M 把锁 , 就可以针对 M 把锁进行编号 (1, 2, 3...M).
N 个线程尝试获取锁的时候 , 都按照固定的按编号由小到大顺序来获取锁 . 这样就可以避免环路等待 .
两个线程对于加锁的顺序没有约定 , 就容易产生环路等待
Object lock1 = new Object();
Object lock2 = new Object();
Thread t1 = new Thread() {
    @Override
    public void run() {
        synchronized (lock1) {
            synchronized (lock2) {
                // do something...
           }
       }
   }
};
t1.start();
Thread t2 = new Thread() {
    @Override
    public void run() {
        synchronized (lock2) {
            synchronized (lock1) {
                // do something...
           }
       }
   }
};
t2.start();
不会产生环路等待的代码:
约定好先获取 lock1, 再获取 lock2 , 就不会环路等待
Object lock1 = new Object();
Object lock2 = new Object();
Thread t1 = new Thread() {
    @Override
    public void run() {
        synchronized (lock1) {
            synchronized (lock2) {
                // do something...
           }
       }
   }
};
t1.start();
Thread t2 = new Thread() {
    @Override
    public void run() {
        synchronized (lock1) {
            synchronized (lock2) {
                // do something...
           }
       }
   }

3.3 避免死锁的解决方案

 

 

四、ThreadLocal

场景:多个班级,每个班级要统计班里的人数,根据人数去做校服

 

 

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

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

相关文章

Linux文本三剑客之sed

Linux文本三剑客之sed 一、sed简介二、工作流程三、sed的常见用法1、常见的sed命令选项2、常见的操作3、基本用法实例3.1 sed查询3.2 sed删除3.3 sed替换sed ‘s/旧字符/新字符/’ &#xff1a;替换每行匹配到的第一个旧字符3.4 sed插入 一、sed简介 sed&#xff08;Stream ED…

Chrome浏览器竟然也可以用ChatGPT了!

最近这段时间想必 和我一样&#xff0c;都被chatGPT刷屏了。 在看到网上给出的一系列chatGPT回答问题的例子和自己亲自体验之后&#xff0c;的确发现它效果非常令人惊艳。 chatGPT的火热程度在开源社区也有很明显的体现&#xff0c;刚推出不久&#xff0c;围绕chatGPT的开源项…

Redis:发布订阅

发布订阅到底是什么功能&#xff1f; 在CSDN这个app中有一个关注的功能&#xff0c;其实这个功能与redis的发布订阅有着异曲同工之处 订阅就相当于关注&#xff0c;关注之后&#xff0c;就相当于加入博主的专属的频道里&#xff0c;只要博主在这个频道里发布了什么信息&#…

VMware虚拟机安装OpenEuler欧拉系统

原文地址&#xff1a;https://program-park.top/2023/05/17/linux_7/ OpenEuler 镜像下载&#xff1a;https://www.openeuler.org/zh/download/   我这里以x86_64架构为示例&#xff0c;使用的23.03版本&#xff1a; 准备好镜像文件&#xff1a; 创建新虚拟机&#xff1a; 选…

HBuilder开发uniapp添加android的模拟器的方法

我们知道使用uniapp开发多端app非常方便&#xff0c;开发过程中的模拟器也可以提高我们测试代码的效率。但我们按uniapp官网的方法&#xff0c;上google的官网下载模拟器&#xff0c;往往非常不方便。 下面我们来看一下使用其他模拟器的方法。 我们知道android开发中&#xf…

Java生成jni.h头文件,java调用C方法 图文详解

环境搭建 1. android studio2021.2.1 2. JDK版本1.8 一、创建一个android项目 File ——> New ——> New Project ——> Empty Activity 创建后如下图所示 二、创建一个java调用C的类 2.1 java类命名为JNITest&#xff0c;创建一个两数之和的方法sums 大概需求…

5.Golang、Java面试题—Spring Cloud、Docker、kubernets(k8s)

本文目录如下&#xff1a; Golang、Java面试题二十、Spring Cloud什么是微服务架构&#xff1f;服务拆分 有哪些注意事项&#xff1f;什么是分布式集群?分布式的 CAP 原则&#xff1f;组件 - Spring Cloud 哪几个组件比较重要&#xff1f;组件 - 为什么要使用这些组件&#xf…

低代码开发ERP:从行业应用到自我价值的思考

随着数字化时代的到来&#xff0c;企业管理软件变得越来越重要。而最为重要的企业管理软件之一便是ERP&#xff08;Enterprise Resource Planning&#xff09;&#xff0c;也就是企业资源计划&#xff0c;它集成了企业内部各个部门的信息&#xff0c;帮助企业进行全面的资源管理…

几个优秀的Wordpress主题汇总(精选免费WP主题)

DNSHH主题 单栏的 WordPress 博客主题&#xff0c;其模板风格&#xff0c;从DNSHH上移植到Typecho&#xff0c;再从Typecho移植到了 WordPress 上&#xff0c;相信这款 WordPress 博客主题这么受欢迎是因为被其简单大气的风格所吸引人吧&#xff01; frontopen 扁平化页面风格…

Java进程(基础)

基本概念 1、进程&#xff1a;程序的执行过程 2、线程&#xff1a;一个进程可以有单个线程也就是我们说的单线程&#xff0c;还可以有多个线程也就是我们说的多线程&#xff0c; 线程 1、当一个类继承了Thread类就可以当成一个线程用 2、我们会重写run方法写上我们自己的业务…

plsql 安装和连接配置

首先下载plsql 安装&#xff0c; 然后 下载oracle 客户端 配置连接(如果安装了oracle 数据库&#xff0c;可以直接配置数据库则可跳过此步骤),下载后解压(解压密码为1) 然后找到 \instantclient_supper\network\admin 目录创建 tnsnames.ora 文件配置数据库连接 例如配置本地…

pynvme操作流程

步骤一&#xff1a;检查本地windows是否安装ssh 检查方式&#xff1a;windows本地打开windows powershell&#xff0c;输入ssh&#xff0c;若打印usage &#xff1a;ssh等一些信息&#xff0c;则已安装ssh&#xff0c;否则需要安装&#xff0c;安装方式如下&#xff0c;一般系…

Java 基础核心总结

目录 前言 介绍 1、基本语法 2、面向对象编程 3、异常处理 4、集合框架 5、IO 流 6、多线程 专栏地址 前言 Java 是一种广泛使用的程序设计语言&#xff0c;具有跨平台、面向对象、安全性高、灵活性强等特点&#xff0c;广泛应用于企业级应用程序和移动应用程序等领域…

win10锁屏或登录时会自动弹出触摸按键的问题解决办法

本篇文章主要讲解win10锁屏或登录时会自动弹出触摸按键的问题解决方式。 日期:2023年5月17日 作者:任聪聪 问题情况截图 屏幕按键说明 如下截图为屏幕按键 设置路径:设置–>轻松使用–>键盘 说明:见图中右侧第一个选择项即使用屏幕键盘的方法,但这并不是本次我们…

《花雕学AI》35:如何一次性和17个AI聊天机器人交流?ChatALL让你轻松实现

聊天机器人&#xff0c;也称为对话机器人&#xff0c;是一种能够通过自然语言与人类进行交流的人工智能系统。聊天机器人的应用领域非常广泛&#xff0c;从客服、娱乐、教育、医疗、社交等&#xff0c;到科研、商业、政治、军事等&#xff0c;几乎无所不包。随着深度学习和自然…

NXP MCUXPresso - 确定冰沙主板工程需要编译的确切文件集合

文章目录 NXP MCUXPresso - 确定冰沙主板工程需要编译的确切文件集合概述END NXP MCUXPresso - 确定冰沙主板工程需要编译的确切文件集合 概述 在尝试迁移 openpnp - Smoothieware project 从gcc命令行 MRI调试方式 到NXP MCUXpresso工程. 先搭了一个MCUXpresso C工程, MCU选…

别去外包,干了三年,废了....

先说一下自己的情况&#xff0c;大专生&#xff0c;18年通过校招进入湖南某软件公司&#xff0c;干了接近4年的功能测试&#xff0c;今年年初&#xff0c;感觉自己不能够在这样下去了&#xff0c;长时间呆在一个舒适的环境会让一个人堕落!而我已经在一个企业干了四年的功能测试…

祖传渣屏退休季,五月份高性价比显示器推荐

眼看五月中旬&#xff0c;又快到了电商狂欢 618&#xff0c;不少伙伴开始将升级电脑配置、全新装机提上日程。 趁着这个节骨眼&#xff0c;咱们正好出一期当前各段位值得购买的高性价比显示器推荐。 入门办公 小米 Redmi 1A 主要参数&#xff1a;23.8 英寸、1920*1080 分辨…

java: 无法访问org.springframework.boot.SpringApplication

SpringBoot启动报错&#xff1a; 原因 根据错误提示&#xff0c;可以看出是类文件版本错误导致的。Spring Boot 3.06 是基于 JDK 17 编译的&#xff0c;而我的 JDK 版本低于此&#xff0c;是JDK8版本&#xff0c;所以无法访问该类文件。因此&#xff0c;解决这个问题需要将 JD…

C++类和对象再探

文章目录 const成员再谈构造函数成员变量的定义函数体内赋值初始化列表 隐式类型转换explicitstatic成员 const成员 我们知道在调用类的成员函数时,会有一个默认的this指针且这个this指针时不可以被修改的,例如在日期类中,会有隐式的Date * const this;注意这里默认会在this前…