【Java基础 下】 029 -- 多线程
创始人
2024-05-30 06:10:37
0

目录

一、为什么要有多线程?

1、线程与进程

2、多线程的应用场景

3、小结

二、多线程中的两个概念(并发和并行)

1、并发

2、并行

3、小结

三、多线程的三种实现方式

1、继承Thread类的方式进行实现

2、实现Runnable接口的方式进行实现

3、利用Callable接口和Future接口方式的实现

4、多线程三种实现方式对比

四、常见的成员方法

1、get/setName方法 -- 线程名字

2、currentThread方法 -- 获取当前线程对象

3、sleep方法 -- 线程休眠

4、set/getPriority方法 -- 线程优先级

5、setDaemon方法 -- 守护线程

6、yield方法 -- 礼让线程

7、join方法 -- 插入线程

8、线程的生命周期

五、线程安全的问题

1、练习:设计一个程序模拟电影院卖票

2、买票引发的安全问题

①、重复票的由来:(线程在执行代码的过程中,CPU的执行权随时有可能被抢走)

②、出现了超出范围的票:(和上面的原因相同)

3、安全问题的解决办法 -- 同步代码块

4、同步代码块中的两个小细节

①、细节1:synchronized要写在循环的里面

②、细节2:synchronized中的锁对象一定是唯一的

5、同步方法

6、StringBuilder和StringBuffer的区别

7、Lock锁(手动加锁、释放锁)

①、Lock使用不规范造成的两个安全问题

六、死锁

七、生产者和消费者(等待唤醒机制)

1、消费者等待

2、生产者等待

3、常见方法(wait/notify/notifyAll)

4、消费者与生产者代码实现

①、Cook.java

②、Desk.java

③、Foodie.java

④、ThreadDemo.java

5、阻塞队列方式(另一种等待唤醒机制)

①、阻塞队列的继承结构

②、阻塞队列实现等待唤醒机制

7、多线程的6中状态

八、综合练习

1、多线程练习1(卖电影票)

2、多线程练习2(送礼品)

3、多线程练习3(打印奇数数字)

4、多线程练习4(抢红包)

精确运算:(BigDecimal)

5、多线程练习5(抽奖箱抽奖)

6、多线程练习6(多线程统计并求最大值)

7、多线程练习7(多线程之间的比较)

8、多线程练习8(多线程阶段大作业)

九、线程池

1、吃饭买碗的故事

①、问题

②、解决方案

2、以前写多线程的弊端

3、线程池的核心原理

4、线程池的代码实现

①、Executors工具类

②、线程复用示例

③、创建一个有上限的线程池

5、自定义线程池(ThreadPoolExecutor)

①、任务拒绝策略

②、代码实现

③、小结

6、最大并行数

①、什么是最大并行数?

②、向Java虚拟机返回可用处理器的数目

7、线程池多大才合适?

十、多线程的额外扩展内容


一、为什么要有多线程?

1、线程与进程

举例:在任务管理器中,一个软件运行之后,它就是一个进程

线程:(简单理解,线程就说应用软件中互相独立,可以同时运行的功能)

单线程程序:所有的都在一个线程中执行,耗时长

2、多线程的应用场景

3、小结

二、多线程中的两个概念(并发和并行)

1、并发

2、并行

以2核4线程为例:(如果计算机中只要4条线程,那么它是不用切换的,但如果线程越来越多,那么这个红线就会在多个线程之间随机的进行切换)

3、小结

三、多线程的三种实现方式

1、继承Thread类的方式进行实现

代码实现:

①、自己定义一个类继承Thread并重写run方法

②、创建子类的对象,并启动线程

2、实现Runnable接口的方式进行实现

代码实现:

①、自己定义一个类实现Runnable接口,并重新里面的run方法

②、创建自己的类对象

③、创建一个Thread类的对象,并开启线程

示例代码:

3、利用Callable接口和Future接口方式的实现

代码实现:

①、创建一个类MyCallable实现Callable接口,并重写call

②、创建MyCallable/FutureTask/Thread的对象

完整代码:

4、多线程三种实现方式对比

四、常见的成员方法

1、get/setName方法 -- 线程名字

默认名字的由来:

序号自增

细节:

2、currentThread方法 -- 获取当前线程对象

3、sleep方法 -- 线程休眠

4、set/getPriority方法 -- 线程优先级

抢占式调度:随机性

非抢占式调度:轮流

没有设置,优先级则默认为5,优先级越高,抢到CPU的概率就越高

示例代码:

5、setDaemon方法 -- 守护线程

两个线程执行的代码不同:守护线程是陆续结束的,所以守护线程也叫做备胎线程

守护线程的应用场景:

6、yield方法 -- 礼让线程

但只是尽可能的均匀,不是绝对的

7、join方法 -- 插入线程

插入线程:将土豆插入到main线程之前,只有当土豆线程执行完毕,才会轮到main线程

8、线程的生命周期

五、线程安全的问题

1、练习:设计一个程序模拟电影院卖票

示例代码:

出现了超出票范围或者重复票的情况:

2、买票引发的安全问题

①、重复票的由来:(线程在执行代码的过程中,CPU的执行权随时有可能被抢走)

②、出现了超出范围的票:(和上面的原因相同)

3、安全问题的解决办法 -- 同步代码块

示例代码:(锁对象一定得是唯一的)

4、同步代码块中的两个小细节

①、细节1:synchronized要写在循环的里面

②、细节2:synchronized中的锁对象一定是唯一的

示例代码:(当前类的字节码文件对象)

5、同步方法

示例代码:

将同步代码块改成同步方法:

6、StringBuilder和StringBuffer的区别

两个类的方法都是相同的

但是StringBuffer是线程安全的,它里面所有的方法都是线程同步的

StringBuilder是非线程安全的,所以如果用到多线程则可以使用StringBuffer,没有需求则选择StringBuilder

7、Lock锁(手动加锁、释放锁)

①、Lock使用不规范造成的两个安全问题

Ⅰ、重复票以及超出范围票

我们在使用Thread类实现多线程时,创建自己的类,一定要注意锁对象需要唯一,即在相关变量前加上static关键字

Ⅱ、程序无法正常终止

这是由于当满足条件时,循环直接被终止,导致lock锁没有被释放

Ⅲ、正确代码(标准写法)

即将容易产生异常的代码块放入try…catch中

六、死锁

死锁是指两个或两个以上的进程在执行过程中,由于竞争资源或者由于彼此通信而造成的一种阻塞的现象,若无外力作用,它们都将无法推进下去。

代码实现:(理解过程)

注意事项:千万不要让两个锁嵌套起来!

七、生产者和消费者(等待唤醒机制)

生产者消费者模式是一个十分经典的多线程协作的模式

1、消费者等待

2、生产者等待

3、常见方法(wait/notify/notifyAll)

4、消费者与生产者代码实现

①、Cook.java

②、Desk.java

③、Foodie.java

④、ThreadDemo.java

5、阻塞队列方式(另一种等待唤醒机制)

①、阻塞队列的继承结构

②、阻塞队列实现等待唤醒机制

Cook.java:

put方法的源码中实现了Lock锁

Foodie.java:

take方法的底层也是有锁的

ThreadDemo.java:

打印语句是在锁的外面的,但是不会对数据造成影响,只是影响了控制台的打印阅读体验

7、多线程的6中状态

Java中是没有定义运行状态的,只有以下6种状态,这是因为一旦线程抢夺到CPU执行权之后,线程就会交给操作系统了,Java就不管了

八、综合练习

1、多线程练习1(卖电影票)

待补充~

2、多线程练习2(送礼品)

待补充~

3、多线程练习3(打印奇数数字)

待补充~

4、多线程练习4(抢红包)

示例代码:

测试类:

精确运算:(BigDecimal)

5、多线程练习5(抽奖箱抽奖)

示例代码:

MyThread.java:

测试类:

6、多线程练习6(多线程统计并求最大值)

示例代码一:(在练习5的基础上进行修改)

MyThread.java:

示例代码二:升级版--线程栈(示例一可以用,但不好)

改进后,这里只需要一个ArrayList就搞定了

示例二内存图讲解:

每个线程都有自己独立的空间

7、多线程练习7(多线程之间的比较)

示例代码:(难点在于如何获取两个线程中的最大值★)

调用多线程的第三种方式Callable来实现(可以返回结果)

MyCallable.java:

测试类:

8、多线程练习8(多线程阶段大作业)

待补充~

 

九、线程池

1、吃饭买碗的故事

①、问题

②、解决方案

买个碗柜,买了碗之后不摔,存入碗柜中

2、以前写多线程的弊端

3、线程池的核心原理

当有新的任务出现,且线程池线程不足时,会新建线程以满足需求,其中最大线程的数量可以自行设置

4、线程池的代码实现

①、Executors工具类

示例代码:

MyRunnable.java:

测试类:

②、线程复用示例

测试类:

③、创建一个有上限的线程池

测试类:

5、自定义线程池(ThreadPoolExecutor)

①、任务拒绝策略

以下面示例为例,它会将任务4抛弃,将任务10加入

②、代码实现

③、小结

6、最大并行数

①、什么是最大并行数?

②、向Java虚拟机返回可用处理器的数目

7、线程池多大才合适?

可以通过thread dump来计算CPU的计算时间和等待时间

十、多线程的额外扩展内容

准备面试时可以再突击学习,资料可见《多线程(额外扩展).md》

相关内容

热门资讯

122.(leaflet篇)l... 听老人家说:多看美女会长寿 地图之家总目录(订阅之前建议先查看该博客) 文章末尾处提供保证可运行...
育碧GDC2018程序化大世界... 1.传统手动绘制森林的问题 采用手动绘制的方法的话,每次迭代地形都要手动再绘制森林。这...
育碧GDC2018程序化大世界... 1.传统手动绘制森林的问题 采用手动绘制的方法的话,每次迭代地形都要手动再绘制森林。这...
Vue使用pdf-lib为文件... 之前也写过两篇预览pdf的,但是没有加水印,这是链接:Vu...
PyQt5数据库开发1 4.1... 文章目录 前言 步骤/方法 1 使用windows身份登录 2 启用混合登录模式 3 允许远程连接服...
Android studio ... 解决 Android studio 出现“The emulator process for AVD ...
Linux基础命令大全(上) ♥️作者:小刘在C站 ♥️个人主页:小刘主页 ♥️每天分享云计算网络运维...
再谈解决“因为文件包含病毒或潜... 前面出了一篇博文专门来解决“因为文件包含病毒或潜在的垃圾软件”的问题,其中第二种方法有...
南京邮电大学通达学院2023c... 题目展示 一.问题描述 实验题目1 定义一个学生类,其中包括如下内容: (1)私有数据成员 ①年龄 ...
PageObject 六大原则 PageObject六大原则: 1.封装服务的方法 2.不要暴露页面的细节 3.通过r...
【Linux网络编程】01:S... Socket多进程 OVERVIEWSocket多进程1.Server2.Client3.bug&...
数据结构刷题(二十五):122... 1.122. 买卖股票的最佳时机 II思路:贪心。把利润分解为每天为单位的维度,然后收...
浏览器事件循环 事件循环 浏览器的进程模型 何为进程? 程序运行需要有它自己专属的内存空间࿰...
8个免费图片/照片压缩工具帮您... 继续查看一些最好的图像压缩工具,以提升用户体验和存储空间以及网站使用支持。 无数图像压...
计算机二级Python备考(2... 目录  一、选择题 1.在Python语言中: 2.知识点 二、基本操作题 1. j...
端电压 相电压 线电压 记得刚接触矢量控制的时候,拿到板子,就赶紧去测各种波形,结...
如何使用Python检测和识别... 车牌检测与识别技术用途广泛,可以用于道路系统、无票停车场、车辆门禁等。这项技术结合了计...
带环链表详解 目录 一、什么是环形链表 二、判断是否为环形链表 2.1 具体题目 2.2 具体思路 2.3 思路的...
【C语言进阶:刨根究底字符串函... 本节重点内容: 深入理解strcpy函数的使用学会strcpy函数的模拟实现⚡strc...
Django web开发(一)... 文章目录前端开发1.快速开发网站2.标签2.1 编码2.2 title2.3 标题2.4 div和s...