多线程(三):Thread 类的基本属性

news/2025/1/12 22:01:34/

上一个篇章浅浅了解了一下 线程的概念,进程与线程的区别,如何实现多线程编程。

而且上一章提到一个重要的面试点: start 方法和 run 方法的区别。

start 方法是从系统那里创建一个新的线程,这个线程会自动调用内部的run 方法;

run 方法是一个线程的执行入口,所有的实现逻辑都写在该方法里。

从概念上来说,这两个方法是属于线程的属性。我们接着这个线程属性继续来认识线程。

线程属性

就像我们之前写的代码:

我们如何让某个线程中断呢?

这里的中断意味着:让线程停下来,线程的终止。

本质上来说让线程终止的办法只有一种,让线程的入口方法 执行完毕。无论是是 return 、抛出异常还是别的方法都是执行完毕。 

这里就需要提到 给线程设定一个结束标志。

给线程设定一个结束标志

来看下面的这段代码: 

我们可以选择这种方式来控制循环,不让他写的那么死。

 但这仍是我们自己设置的变量,而不属于线程的属性,线程中也有个方法:interru方法

interru方法

我们来看看这一段代码:

 我们这里就用到了几个线程的属性, 

Thread.currentThread()、isInterrupted()、t.interrupt()

上述的e.printStackTrace() 就是打印出异常信息。 

我们来看看结果:

我们发现,在main线程睡眠了 3s 之后并没有打印完异常就结束,而是继续线程 t 的运行。

这是为什么呢?

 这就需要来聊聊 interrupt 方法的作用了:

  1. 将标识符设置为 true 
  2. 如果该线程正在阻塞状态中(例如:sleep,后面还有其他方法也会使线程阻塞),会直接被interrupt 方法唤醒,此时通过抛出异常就会让 sleep 立即结束。

按照 interrupt 方法的作用,此时应该结束了啊,t 线程这么还会继续呢?

这里有个非常重要的点;

  当 sleep 被唤醒以后,会自动将 isInterrupted 设置为 false,也就是说将异常标志位清空了(true -> false);

sleep 为什么要清空异常标志位呢?

目的是为了让这个线程能够对于线程什么时候结束有一个更精准的控制。

当前的 interrpt 方法是告诉线程你该结束了,至于什么时候结束,都是由代码灵活控制的。

interrpt 方法 是个通知而不是命令。

等待线程 -- join 方法

线程与线程之间的的并发执行的,线程间的调度是无序的,我们无法判断两个线程之间谁先结束谁后结束。

例如: 

但是我们可以通过 join 方法进行控制,例如:

 在t 执行 join 方法时,如果 t 线程尚未结束,那么main线程就会进入 Blocking (阻塞状态),因为t.join 在 main线程中执行。

main 线程代码走到这一行 就不参与 cpu 的执行调度了。

如果这个有多个线程并发执行, 那么 只是main 线程不参与线程的调度,其余的线程任然是抢占执行的。

我们使用的 join 是无参的版本,我们还存在另一个带参的版本。

这个带参的版本,参数作为等待的最大时间,超过了这个最大时间,t 线程还没有结束,那么main 线程 就会被唤醒,不再是Blocking 状态 就会继续执行。

而上面不带参的版本,就是死等,等不到 t 线程结束,mian 线程就不会开始。

既然上面都提到了线程状态,那么就简单理解一下线程状态。

线程状态

Java中线程的状态分为6种:

1. 初始(NEW):新创建了一个线程对象,但还没有调用start()方法。
2. 运行(RUNNABLE):Java线程中将就绪(ready)和运行中(running)两种状态笼统的称为“运行”。
线程对象创建后,其他线程(比如main线程)调用了该对象的start()方法。该状态的线程位于可运行线程池中,等待被线程调度选中,获取CPU的使用权,此时处于就绪状态(ready)。就绪状态的线程在获得CPU时间片后变为运行中状态(running)。
3. 阻塞(BLOCKED):表示线程阻塞于锁。
4. 等待(WAITING):进入该状态的线程需要等待其他线程做出一些特定动作(通知或中断)。
5. 超时等待(TIMED_WAITING):该状态不同于WAITING,它可以在指定的时间后自行返回。
6. 终止(TERMINATED):表示该线程已经执行完毕。

 我们简单理解就是一条主线,三条支线。

我们就简单了解一下就好,后面还会再聊到。

简单了解线程安全

对于多线程编程雀氏提供了很多便利,我们可以再同一时间完成多个任务,相比于多进程编程多线程还降低了 cpu 资源的占用。

这几章都提到了,多线程之间的线程调度是抢占式执行的,所以在此基础之上会产生很多的线程安全问题。

本章就稍稍介绍认识一下线程安全的问题就行。

本章以一个不安全的例子来认识

例如:我们设置一个 count ,两个线程各执行 5 w 次自增运算。

代码如下:

class Count {private int count = 0;public void add() {count++;}public int get() {return count;}
}
public class demo10 {public static void main(String[] args) throws InterruptedException {Count count = new Count();Thread t1 = new Thread(() -> {for (int i = 0; i < 50000; i++) {count.add();}});Thread t2 = new Thread(() -> {for (int i = 0; i < 50000; i++) {count.add();}});t1.start();t2.start();t1.join();t2.join();System.out.println(count.get());}
}

运行一次看看结果:

再来试试看:

 再来一次:

从结果上来看它是一个随机数。

至于为什么会产生一个随机数,我们需要了解在这执行的内部发生了什么事情:

对于这个 count++ 这个操作,本质上是由 cpu 上三个指令构成的:

  1. load : 从内存中将数据读取到 cpu 寄存器中
  2. add :在 cpu 中的值进行 + 1 运算
  3. save : 把寄存器中的值写入到内存中

由于是抢占式执行的,那么就会产生如下等问题:

这个 3 的执行顺序就变成不确定的了,产生多种排列组合结果。

只有 25 亿分之 1 的可能得到 10w 这个数字,也有可能会产生结果小于 5w 具体的可以自己排列组合看看。

对于 t1 和 t2 这两个线程可能式运行在同一个 cpu 核心上执行,也有可能在不同的 cpu 核心上执行。

归根结底,线程安全问题,全是因为线程之间的无序调度,导致了执行顺序不确定,结果不确定。


http://www.ppmy.cn/news/35545.html

相关文章

面试热点题:回溯算法 递增子序列与全排列 II

前言&#xff1a; 如果你一点也不了解什么叫做回溯算法&#xff0c;那么推荐你看看这一篇回溯入门&#xff0c;让你快速了解回溯算法的基本原理及框架 递增子序列 给你一个整数数组 nums &#xff0c;找出并返回所有该数组中不同的递增子序列&#xff0c;递增子序列中 至少有两…

【第13届蓝桥杯】C/C++组B组省赛题目+详解

A.九进制转十进制 题目描述 九进制正整数(2022)9转换成十进制等于多少&#xff1f; 解&#xff1a; 2*9^02*9^12*9^321814581478; B.顺子日期 题目描述 小明特别喜欢顺子。顺子指的就是连续的三个数字&#xff1a;123、456等。顺子日期指的就是在日期的yyyymmdd表示法中&a…

【javaEE】阻塞队列、定时器、线程池

目录 &#x1f334;一、阻塞队列 1.概念 2.生产者消费者模型 3.阻塞队列的实现 &#x1f3f9;二、定时器 1.引出定时器 2.定时器的实现 &#x1f525;三、线程池 1.引出线程池 2.ThreadPoolExecutor 构造方法 3.标准数据库的4种拒绝策略【经典面试题】【重点掌握】 …

进程间通信----信号量

文章目录信号量1. 问题2. 什么是信号量3. 信号量的使用4. 信号量的控制6. 实例信号量 1. 问题 程序中&#xff0c;有时存在一种特殊代码&#xff0c;最多只允许一个进程执行该部分代码。这部分区域&#xff0c;称为“临界区” 然而在多进程并发执行时&#xff0c;当一个进程进…

【云原生|Docker】06-dokcerfile详解

目录 前言 Dockerfile基础示例 Dockerfile简介 1. Dockerfile概念 2. Dokcer镜像分层理解 ​3. Doker build构建原理 Dockerfile参数解析 1. Dokcerfile组成 2. 指令说明 2.1 FROM引入基础镜像 2.2 LABEL 2.3 ENV 2.4 RUN 2.5 COPY 2.6 ADD 2…

【十二天学java】day09常用api介绍

1.API 1.1API概述 什么是API API (Application Programming Interface) &#xff1a;应用程序编程接口 java中的API 指的就是 JDK 中提供的各种功能的 Java类&#xff0c;这些类将底层的实现封装了起来&#xff0c;我们不需要关心这些类是如何实现的&#xff0c;只需要学习这…

软件测试面试题 —— 整理与解析(3)

&#x1f60f;作者简介&#xff1a;博主是一位测试管理者&#xff0c;同时也是一名对外企业兼职讲师。 &#x1f4e1;主页地址&#xff1a;&#x1f30e;【Austin_zhai】&#x1f30f; &#x1f646;目的与景愿&#xff1a;旨在于能帮助更多的测试行业人员提升软硬技能&#xf…

RPA机器人能做什么?自动化办公、简化工作流程……还有很多事情等着你挖掘

看到这篇文章&#xff0c;首先我要恭喜你&#xff0c;因为我会带领你进入一个全新的领域&#xff0c;这将极大地方便你的日常工作和生活。 你听说过RPA吗&#xff1f;也许你会认为这是一个非常陌生的名词&#xff0c;但它可能会改变你的工作方式&#xff0c;提高你的工作效率&a…