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 {
         //睡眠1000ms
             Thread.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(但不建议使用该方法)。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

基于C++的AI五子棋游戏项目开发教程

项目资源下载 基于C的AI五子棋游戏项目源码压缩包下载地址基于C的AI五子棋游戏项目源码Github下载地址基于C的AI五子棋游戏项目所需素材基于C的AI五子棋游戏项目所需要的EasyX 项目简介 本项目基于C开发&#xff0c;整体来说比较简单&#xff0c;实现了人与AI之间的五子棋对弈…

Java实习生------Redis常见面试题汇总(AOF持久化、RDB快照、分布式锁、缓存一致性)⭐⭐⭐

“年轻人&#xff0c;就要勇敢追梦”&#x1f339; 参考资料&#xff1a;图解redis 目录 谈谈你对AOF持久化的理解&#xff1f; redis的三种写回策略是什么&#xff1f; 谈谈你对AOF重写机制的理解&#xff1f;AOF重写机制的具体过程&#xff1f; 谈谈你对RDB快照的理解&a…

面试官:说一下MySQL中的锁机制吧

5. 1MySQL有哪些锁&#xff1f; 为保证数据的一致性&#xff0c;需要对并发操作进行控制&#xff0c;因此产生了锁。同时锁机制也为实现MySQL的各个隔离级别提供了保证。 锁冲突 也是影响数据库并发访问性能的一个重要因素。所以锁对数据库而言显得尤其重要&#xff0c;也更加…

seata服务搭建

它支持两种存储模式&#xff0c;一个是文件&#xff0c;一个是数据库&#xff0c;下面我们分别介绍一下这两种配置nacos存储配置&#xff0c;注意如果registry.conf中注册和配置使用的是file&#xff0c;就会去读取file.config的配置&#xff0c;如果是nacos则通过nacos动态读取…

Kafka和RabbitMQ有哪些区别,各自适合什么场景?

目录标题1. 消息的顺序2. 消息的匹配3. 消息的超时4. 消息的保持5. 消息的错误处理6. 消息的吞吐量总结1. 消息的顺序 有这样一个需求&#xff1a;当订单状态变化的时候&#xff0c;把订单状态变化的消息发送给所有关心订单变化的系统。 订单会有创建成功、待付款、已支付、已…

Cookie和Session详解

目录 前言&#xff1a; Session详解 Cookie和Session区别和关联 服务器组织会话的方式 使用Tomcat实现登录成功跳转到欢迎页面 登录前端页面 登录成功后端服务器 重定向到欢迎页面 抓包分析交互过程 小结&#xff1a; 前言&#xff1a; Cookie之前博客有介绍过&#x…

音视频技术开发周刊 | 285

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

使用GPT-4生成QT代码

一、概述最近ChatGPT火爆起来了&#xff0c;ChatGPT是一种基于GPT的自然语言处理模型&#xff0c;可以用于生成自然语言文本&#xff0c;例如对话、文章等。最近又发现了一个优秀且免费的代码生成工具Cursor.so &#xff0c;Cursor.so集成了 GPT-4 &#xff0c;可以帮助你快速编…

Python3,5行代码,生成自动排序动图,这操作不比Excel香?

5行代码生成自动排序动图1、引言2、代码实战2.1 pynimate介绍2.2 pynimate安装2.3 代码示例3、总结1、引言 小屌丝&#xff1a;鱼哥&#xff0c;听说你的excel段位又提升了&#xff1f; 小鱼&#xff1a;你这是疑问的语气&#xff1f; 小屌丝&#xff1a;没有~ 吧… 小鱼&…

计算机网络复习重点

文章目录计算机网络复习重点第一章 计算机网络和因特网概念与应用1、什么是因特网2、协议protocol3、入网方式4、物理媒介5、数据交换模式6、延时与丢包什么时候发生延时&#xff1f;延时的类型丢包何时发生7、协议层次与模型因特网协议栈TCP / IP模型ISO/OSI参考模型协议数据单…

MySQL-存储过程

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

归并排序和快速排序

目录 归并排序 思路&#xff1a; 代码执行&#xff1a; 快速排序 运行流程图&#xff1a; 代码思路&#xff1a; 代码执行&#xff1a; 归并排序 定义&#xff1a;归并排序是建立在归并操作上的一种有效&#xff0c;稳定的排序算法&#xff0c;该算法是采用分治法&#x…

【JavaSE】类和对象(中)

类和对象&#xff08;中&#xff09;4. this引用4.1 为什么要有this引用4.2 什么是this引用4.3 this引用的特性5. 对象的构造及初始化5.1 如何初始化对象5.2 构造方法&#xff08;构造器&#xff09;5.2.1 概念5.2.2 特性5.3 默认初始化5.4 就地初始化6. 封装6.1 封装的概念6.2…

常用hook

Hook 是 React 16.8 的新增特性。它可以让你在不编写 class 的情况下使用 state 以及其他的 React 特性。理解&#xff1a;hook是react提供的函数API官方提供的hook基础hookuseState APIconst [state, setState] useState(initialState); //返回state值 以及更新state的方法 …

壹沓科技完成近2亿元B轮融资:构建数字机器人,实现业务超自动化

RPA中国获悉&#xff0c;全球领先的数字机器人公司壹沓科技近期宣布完成近2亿元B轮融资&#xff0c;本轮融资由鼎晖VGC(创新与成长基金)领投&#xff0c;创享欢聚投资基金、IDG资本、钟鼎资本跟投&#xff0c;指数资本继续担任独家财务顾问。 壹沓科技CEO卞晓瑜表示&#xff1…

栈、队列、优先级队列的模拟实现

优先级队列的模拟实现栈stack的模拟实现push()pop()top()size()empty()swap()stack总代码队列queue的模拟实现push()pop()front()back()empty()size()swap()queue总代码优先级队列(堆)push()pop()top()empty()size()swap()priority_queue总代码deque的了解栈 在CSTL中栈并不属…

吃透Java面试题,建议收藏

本文已经收录到Github仓库&#xff0c;该仓库包含计算机基础、Java基础、多线程、JVM、数据库、Redis、Spring、Mybatis、SpringMVC、SpringBoot、分布式、微服务、设计模式、架构、校招社招分享等核心知识点&#xff0c;欢迎star~ Github地址&#xff1a;https://github.com/…

LeetCode:27. 移除元素

&#x1f34e;道阻且长&#xff0c;行则将至。&#x1f353; &#x1f33b;算法&#xff0c;不如说它是一种思考方式&#x1f340;算法专栏&#xff1a; &#x1f449;&#x1f3fb;123 一、&#x1f331;27. 移除元素 题目描述&#xff1a;给你一个数组 nums 和一个值 val&am…

Chapter6.2:其他根轨迹及综合实例分析

该系列博客主要讲述Matlab软件在自动控制方面的应用&#xff0c;如无自动控制理论基础&#xff0c;请先学习自动控制系列博文&#xff0c;该系列博客不再详细讲解自动控制理论知识。 自动控制理论基础相关链接&#xff1a;https://blog.csdn.net/qq_39032096/category_10287468…

使用vite创建vue3工程

定义 什么是vite&#xff1f;-----新一代前端构建工具 优势 开发环境中&#xff0c;无需打包操作&#xff0c;可快速的冷启动---最牛的地方轻量快速的热重载&#xff08;HMR&#xff09;---一修改代码就局部刷新&#xff0c;webpack也具备&#xff0c;但vite更快真正的按需编…