java多线程之线程的六种状态
线程的六种状态
- (1) NEW(初始状态)
- (2) TERMINATED(终止状态 / 死亡状态)
- (3) RUNNABLE(运行时状态)
- (4) TIMED_WAITING(超时等待状态)
- (5) WAITING(等待状态)
- (6) BLOCK(阻塞状态)
- sleep和wait的区别:
操作系统里的线程自身是有一个状态的,但是java Thread 是对系统线程的封装,把这里的状态又进一步细化了~~
状态 | 说明 |
---|---|
NEW | 线程还没有创建,但是线程对象已经创建出来了 |
TERMINATED | 线程结束了,但是线程对象还存在 |
RUNNABLE | 就绪状态,可以细分为两个状态 |
TIMED_WAITING | 指超时等待状态 |
BLOCK | 等待状态 |
WAITING | 表示阻塞时出现的状态 |
下面将通过代码运行结果来带大家细致了解线程运行的状态.
(1) NEW(初始状态)
public static void main(String[] args) {Thread t = new Thread(()->{System.out.println("t 线程");});//获取线程的状态System.out.println("当前线程是 " +t.getState() + "状态");//调用start(),创建这个线程t.start();}
NEW状态指的是 : 线程在刚刚被new出来的时候,还没有调用start()的状态.
此时可以称这个状态为: (1) 初始状态 (2) 创建状态
运行结果:
(2) TERMINATED(终止状态 / 死亡状态)
public static void main(String[] args) {Thread t = new Thread(()->{System.out.println("t 线程");});//调用start(),创建这个线程t.start();try {//由于计算机的执行速度是很快的,所以1000ms足够t线程执行完Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}//获取线程的状态System.out.println("当前线程是 " +t.getState() + "状态");t.start();}
如果一个线程的 run()
方法执行结束 该线程就会死亡,对于已经死亡的线程,无法再使用 start()
方法令其进入就绪.
运行结果:
(3) RUNNABLE(运行时状态)
运行时状态可以细分为两个状态:(1) 正在运行状态RUNNING
(2) 就绪状态READY
我们可以这么理解 :
(1) 例如我女朋友下班了,我正在带我女朋友去吃饭,此时就是正在运行状态.
(2)因为我提前下班了,但是我女朋友还没有下班,所以此时我在女朋友公司门口等她,只要她下班我随时可以带她吃饭去,此时我就是就绪状态.
通过代码实现(1) :
public static void main(String[] args) {Thread t = new Thread(()->{while(true){//什么都不打印,防止把下面的打印信息冲走}});//调用start(),创建这个线程t.start();//运行线程时判断此时的状态System.out.println("当先线程是 "+t.getState()+"状态");}
由于此时的 t 线程一直在执行死循环,所以 t 线程的状态是正在运行的~~
运行结果:
(4) TIMED_WAITING(超时等待状态)
可以理解为,具有指定等待时间的,正在等待(阻塞)线程的线程状态,由于调用具有指定等待时间的以下方法之一,线程处于定时等待状态.
具有指定等待时间的方法:
(1) Thread.sleep(参数)指定时间,单位为ms
该方法会让当前线程暂停一段时间,其他线程有机会获得 CPU 时间片。
(2) t.join(参数)
调用 t.join
的线程需要等待线程 t 执行指定时间后,才可以运行,等待的过程中是处于阻塞状态的.
(3) wait(参数) :wait 方法提供一个带有 timeout 参数的版本, 来指定等待时间.超过这个时间之后无需其他线程调用该对象的 notify()
或 notifyAll()
方法唤醒该线程,该线程自己就会唤醒.
代码实现(1)
public static void main(String[] args) {Thread t = new Thread(()->{try {//睡眠1000msThread.sleep(1000);System.out.println("hello t");} catch (InterruptedException e) {e.printStackTrace();}});//调用start(),创建这个线程t.start();try {//等待1000ms,此时t线程就创建结束,系统自动执行run方法里面的逻辑Thread.sleep(1000);} catch (InterruptedException e) {e.printStackTrace();}//运行线程时判断此时的状态System.out.println("当先线程是 "+t.getState()+"状态");}
运行结果:
代码实现(2)
public static void main(String[] args) throws InterruptedException {Thread t1 = new Thread(()->{while (true){try {//每500ms打印一次Thread.sleep(500);System.out.println("t1");} catch (InterruptedException e) {e.printStackTrace();}}});Thread t2 = new Thread(()->{try {//等待t1线程执行完1000ms//t2线程再开始执行t1.join(1000);System.out.println("t2");} catch (InterruptedException e) {e.printStackTrace();}});//创建t1线程t1.start();//创建t2线程t2.start();//等待500ms此时两个线程都创建完毕Thread.sleep(500);System.out.println("当先线程是 "+t2.getState()+"状态");}
运行结果:
代码实现(3)
public class ThreadDemo8 {//自己指定的锁对象static Object object = new Object();public static void main(String[] args) throws InterruptedException {Thread t1 = new Thread(()->{//两个锁对象相同synchronized (object){for (int i = 0; i < 10; i++) {if(i == 5) {try {object.wait(1000);//虽然过了1000ms,但是还是需要等待t2线程中锁里面的程序执行完// 才可以继续执行锁.(这里是指,仅有两个锁,且锁对象相等的情况)} catch (InterruptedException e) {e.printStackTrace();}}System.out.println(i);}}});Thread t2 = new Thread(()->{//两个锁对象相同synchronized (object){for (int i = 0; i <10 ; i++) {try {System.out.println("t2");Thread.sleep(100);} catch (InterruptedException e) {e.printStackTrace();}}}});//t1线程先创建t1.start();//等待100ms是防止t2先抢到锁Thread.sleep(100);//创建t2线程t2.start();//查看当前线程的状态System.out.println("当先线程是 "+t1.getState()+"状态");}
}
运行结果:
(5) WAITING(等待状态)
处于这种状态的线程不会被CPU分配执行时间,他们要等待的显示被唤醒,否则会处于无限期的等待状态
具有阻塞的方法:
(1)使用 Thread.join()
方法。该方法会让当前线程等待另一个线程终止
(2) 使用 Object.wait()
方法。该方法会让当前线程等待,直到其他线程调用该对象的 notify() 或 notifyAll() 方法唤醒该线程。
我们这里只演示第一种就好~
public static void main(String[] args) throws InterruptedException {Thread t1 = new Thread(()->{while (true){}});Thread t2 = new Thread(()->{try {t1.join();} catch (InterruptedException e) {e.printStackTrace();}},"猪猪侠");//创建这t1线程t1.start();//等待200ms让t1线程优先创建好Thread.sleep(200);//创建这t2线程t2.start();//等待200ms让t2线程优先创建好Thread.sleep(200);System.out.println("当先线程 "+t2.getName()+"的状态是"+t2.getState()+"状态");}
运行结果是:
(6) BLOCK(阻塞状态)
阻塞状态(Blocked):阻塞状态是线程因为某种原因放弃cpu的使用权,暂停或停止运行,直到线程进入就绪状态,才有机会获得cpu的青睐从而转入运行状态。
我们可以通俗点理解为~ 等待锁的状态.
例如: t1 和 t2 需要的锁对象相同,如果此时 t1 拿到了锁,当 t2 执行到进入锁的代码的时候, t2 就需要等 t1 释放锁之后才可以拿到这把锁.
我们通过代码演示:
public class ThreadDemo8 {static Object object = new Object();public static void main(String[] args) throws InterruptedException {Thread t1 = new Thread(()->{//两个锁对象相同synchronized (object){while (true){}}});Thread t2 = new Thread(()->{//两个锁对象相同//由于先执行的t1所以需要等t1释放锁才可以执行t2//此时t2处于阻塞状态synchronized (object){System.out.println("t2");}},"猪猪侠");//创建这t1线程t1.start();//等待200ms让t1线程优先创建好Thread.sleep(200);//创建这t2线程t2.start();//等待200ms让t2线程优先创建好Thread.sleep(200);//查看t2线程的状态System.out.println("当先线程 "+t2.getName()+"的状态是"+t2.getState()+"状态");}
}
运行结果:
关于线程状态的关系图:
sleep和wait的区别:
sleep()和wait()方法有什么区别:
sleep()睡眠时,保持对象锁,仍然占有该锁;
而wait()睡眠时,释放对象锁。
但是wait()和sleep()都可以通过interrupt()方法打断线程的暂停状态,从而使线程立刻抛出InterruptedException(但不建议使用该方法)。
相关文章:

java多线程之线程的六种状态
线程的六种状态(1) NEW(初始状态)(2) TERMINATED(终止状态 / 死亡状态)(3) RUNNABLE(运行时状态)(4) TIMED_WAITING(超时等待状态)(5) WAITING(等待状态)(6) BLOCK(阻塞状态)sleep和wait的区别:操作系统里的线程自身是有一个状态的,但是java Thread 是对系统线程的封装,把这里的…...

UnixBench----x86架构openEuler操作系统上进行性能测试
【原文链接】UnixBench----x86架构openEuler操作系统上进行性能测试 (1)打开github上 UnixBench 地址,找到发布的tag (2)找到tar.gz包,右键复制链接 比如这里是 https://github.com/kdlucas/byte-unix…...

于Java8 Stream教程之collect()
目录 前言正文第一个小玩法 将集合通过Stream.collect() 转换成其他集合/数组:第二个小玩法 聚合(求和、最小、最大、平均值、分组)总结前言 本身我是一个比较偏向少使用Stream的人,因为调试比较不方便。 但是, 不得不说&#…...

Python
1、str 三个关键点: 正着数,0,1,2 反着数,0,-1,-2 str[a,b] 左闭右开 [a,b) str123456789 print(str) # 输出字符串 print(str[0:-1]) # 输…...

Spring框架中IOC和DI详解
Spring框架学习一—IOC和DI 来源黑马Spring课程,觉得挺好的 目录 文章目录Spring框架学习一---IOC和DI目录学习目标第一章 Spring概述1、为什么要学习spring?2、Spring概述【了解】【1】Spring是什么【2】Spring发展历程【3】Spring优势【4】Spring体系…...

本地快速搭建Kubernetes单机版实验环境(含问题解决方案)
Kubernetes是一个容器编排系统,用于自动化应用程序部署、扩展和管理。本指南将介绍Kubernetes的基础知识,包括基本概念、安装部署和基础用法。 一、什么是Kubernetes? Kubernetes是Google开发的开源项目,是一个容器编排系统&…...

FPGA控制DDS产生1CLK周期误差的分析(二)
前文简短的介绍了DDS的产生原理,其实相当的简单,所以也不需要多做解释,本文详细阐述一下在调试DDS的过程中所产生的一个bug 问题发现 正如上文所述,再用FPGA控制存储在rom中的波形信号输出之后,在上板之前࿰…...

这一次,吃了Redis的亏,也败给了GPT
关注【离心计划】,一起离开地球表面 背景 组内有一个系统中有一个延迟任务的需求,关于延迟任务常见的做法有时间轮、延迟MQ还有Redis Zset等方案,关于时间轮,这边小苏有一个大学时候做的demo: https://github.com/JA…...

第一章 信息化知识
1、信息是客观事物状态和运动特征的一种普遍形式,信息的概念存在两个基本的层次,即本体论层次和认识论层次: 本体论层次:就是事物的运动状态和状态变化方式的自我表述认识论层次:就是主体对于该事物的运动状态以及状态…...

如何用matlab工具箱训练一个SOM神经网络
本站原创文章,转载请说明来自《老饼讲解-BP神经网络》bp.bbbdata.com本文展示如何用matlab工具箱训练一个SOM神经网络的DEMO并讲解其中的代码含义和相关使用说明- 01.SOM神经网络DEMO代码 -- 本文说明 -下面,我们先随机初始化一些样本点,然后…...

音视频技术开发周刊 | 285
每周一期,纵览音视频技术领域的干货。新闻投稿:contributelivevideostack.com。GPT-4 Office全家桶发布谷歌前脚刚宣布AI工具整合进Workspace,微软后脚就急匆匆召开了发布会,人狠话不多地祭出了办公软件王炸——Microsoft 365 Cop…...

安装flume
flume最主要的作用就是实时读取服务器本地磁盘的数据,将数据写入到hdfs中架构:开始安装一,上传压缩包,解压并更名解压:[rootsiwen install]# tar -zxf apache-flume-1.9.0-bin.tar.gz -C ../soft/[rootsiwen install]#…...

为工作排好优先级
工作,是干不完的,因此我们需要分清轻重缓急,为它们划分优先级,这样才不至于让自己手忙脚乱。 给手头的事情排上正确的优先级,是一项很重要的工作能力。 优先级有很多考量,并不是简单的先来后到的线性时间…...

超专业解析!10分钟带你搞懂Linux中直接I/O原理
我们先看一张图: 这张图大体上描述了 Linux 系统上,应用程序对磁盘上的文件进行读写时,从上到下经历了哪些事情。 这篇文章就以这张图为基础,介绍 Linux 在 I/O 上做了哪些事情。 文件系统 什么是文件系统 文件系统࿰…...

【C++】面试101,用两个栈实现队列,包含min函数的栈,有效括号序列,滑动窗口的最大值,最小的K个数,倒置字符串,排序子序列,跳跃,数字三角形,蓝肽子序列
目录 1. 用两个栈实现队列 2.包含min函数的栈 3.有效括号序列 4.滑动窗口的最大值 5.最小的K个数 6.倒置字符串 7.排序子序列 8.数字三角形(蓝桥杯,学习一个大佬的博客....) 9.跳跃(蓝桥杯) 10.蓝肽子序列 1. 用…...

WPF 认识WPF
什么是WPF?WPF是Windows Presentation Foundation(Windows展示基础)简称,顾名思义是专门编写表示层的技术。WPF绚丽界面如下:GUI发展及WPF历史?Windows系统平台上从事图形用户界面GUI(Graphic User Interface)已经经历了多次换代,…...

【建议收藏】PHP单例模式详解以及实际运用
PHP单例模式详解以及实际运用 什么是单例模式? 首先我们百度百科他怎么说? 单例模式,属于创建类型的一种常用的软件设计模式。通过单例模式的方法创建的类在当前进程中只有一个实例(根据需要,也有可能一个线程中属于单例,如&a…...

【十二天学java】day04-流程控制语句
第一章 流程控制语句 在一个程序执行的过程中,各条语句的执行顺序对程序的结果是有直接影响的。所以,我们必须清楚每条语句的执行流程。而且,很多时候要通过控制语句的执行顺序来实现我们想要的功能。 1.1 流程控制语句分类 顺序结构 判断…...

Pandas 与 PySpark 强强联手,功能与速度齐飞
Pandas做数据处理可以说是yyds!而它的缺点也是非常明显,Pandas 只能单机处理,它不能随数据量线性伸缩。例如,如果 pandas 试图读取的数据集大于一台机器的可用内存,则会因内存不足而失败。 另外 pandas 在处理大型数据…...

【Zabbix实战之部署篇】docker部署Zabbix+grafana监控平台
【Zabbix实战之部署篇】docker部署Zabbix+grafana监控平台 一、Zabbix介绍1.Zabbix简介2.Zabbix的优点3.Zabbix各组件介绍4.Zabbix架构图二、grafana介绍1.grafana简介2.grafana特点三、实践环境规划四、检查本地docker环境1.检查操作系统版本2.检查docker版本3.检查docker服务…...

acm省赛:高桥和低桥(三种做法:区间计数、树状数组、线段树)
题目描述 有个脑筋急转弯是这样的:有距离很近的一高一低两座桥,两次洪水之后高桥被淹了两次,低桥却只被淹了一次,为什么?答案是:因为低桥太低了,第一次洪水退去之后水位依然在低桥之上ÿ…...

stm32-定时器详解
0. 概述 本文针对STM32F1系列,主要讲解了其中的8个定时器的原理和功能 1. 定时器分类 STM32F1 系列中,除了互联型的产品,共有 8 个定时器,分为基本定时器,通用定时器和高级定时器基本定时器 TIM6 和 TIM7 是一个 16 位…...

《硬件架构的艺术》读书笔记:Chapter 1 亚稳态的世界
Chapter 1 亚稳态的世界 一、简介 同步系统中,数据和时钟有固定的因果关系(在同一时钟域(Clock Domains))中,只要数据和时钟满足建立时间和保持时间的要求,不会产生亚稳态(meastable) 静态时序分析(STA) 就是基于同步电路设计模型而出现的&am…...

开箱即用的密码框组件
写了一个小玩具,分享一下 - 组件功能: 初次进入页面时,密码隐藏显示,且无法查看真实密码 当修改密码时,触发键盘,输入框则会直接清空 此时输入密码,可以设置密码的隐藏或显示: …...

ChatGPT能否取代程序员?
目录ChatGPT能否取代程序员?ChatGPT和程序员的工作内容和工作方式ChatGPT和程序员的共同点程序员的优势程序员的实力ChatGPT和程序员的关系结论惊喜ChatGPT能否取代程序员? ChatGPT是一种非常普遍的人工智能(AI)系统,…...

案例分享 | 金融微服务场景下如何提升运维可观测性
云原生环境下金融业务的微服务化改造以及分布式架构的部署,使得业务与开发部门的关联更为紧密,传统运维监控已满足不了业务运营需求,亟需建设具备可观测性的运维体系。所以这次我们以某金融客户的实践案例为例,跟大家说一说在金…...

CentOS8提高篇3:Centos8安装播放器(mplayer vlc)
1. 准备工作(需要配置epel, rpmfusion源); 配置epel源 下载epel dnf install epel-release 配置rpmfusion源 下载rpmforge dnf install rpmfusion-free-release-8.noarch.rpm 2. 安装mplayer和vlc 直接dnf安装 # dnf install mplayer # dnf install v…...

MySQL-存储过程
什么是存储过程我们前面所学习的MySQL语句都是针对一个表或几个表的单条 SQL 语句,但是在数据库的实际操作中,并非所有操作都那么简单,经常会有一个完整的操作需要多条SQL语句处理多个表才能完成。例如,为了确认学生能否毕业&…...

经典七大比较排序算法 · 下 + 附计数和基数排序
经典七大比较排序算法 下 附计数和基数排序1 插入排序1.1 算法思想1.2 代码实现1.3 插入排序特性2 希尔排序2.1 算法思想2.2 代码实现2.3 希尔排序特性3 七大比较排序特性总结4 计数排序4.1 算法思想4.2 代码实现4.3 计数排序特性5 基数排序5.1 算法思想5.2 代码实现1 插入排…...

HTTPS协议,看这篇就够了
不安全的HTTP 近些年来,越来越多的网站使用 HTTPS 协议进行数据传输,原因在于 HTTPS 相较于 HTTP 能够提供更加安全的服务。 很多浏览器对于使用 HTTP 协议的网站会加上『警告』的标志表示数据传输不安全,而对于使用 HTTPS 协议的网站会加上…...