news 2026/5/19 16:24:27

Thread类的基本用法

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Thread类的基本用法

1.线程创建

见上篇博客https://blog.csdn.net/weixin_69059394/article/details/155943050?fromshare=blogdetail&sharetype=blogdetail&sharerId=155943050&sharerefer=PC&sharesource=weixin_69059394&sharefrom=from_link

2.是否是守护线程/设置守护线程

守护线程其实就是后台线程。

我们平时代码中所创建的所有线程默认都是前台线程。如需更换为后台线程,使用.setDaemon()。

判断是否是后台线程使用.isDaemon()。

JVM会在⼀个进程的所有非后台线程结束后,才会结束运行。

3.线程是否存活

java中创建的Thread对象和系统中的线程,是一一对应的关系。

但是Thread对象的生命周期和系统线程的生命周期,是不同的。很有可能Thread对象还存在,但是系统中的线程已经销毁了,判断线程是否还存活使用.isAlive()方法。

4.线程启动

.start()而非.run()。run()是线程的入口方法。

每个Thread对象只能.start()一次。(日抛)

5.进程中断

中断进程其实就是线程终止,意味着线程终止以后就不会再恢复了。

只要将线程的入口方法执行(run()方法)执行完毕,线程就随之结束了。

public class demo6 { public static boolean isFinished=false; public static void main(String[] args) throws InterruptedException { Thread t=new Thread(()->{ while(!isFinished){ System.out.println("hello thread"); try { Thread.sleep(1000); } catch (InterruptedException e) { throw new RuntimeException(e); } } System.out.println("t线程结束"); }); t.start(); Thread.sleep(5000); isFinished=true; } }

注意,这里我们创建的isFinished为静态成员变量,线程的入口方法是可以通过内部类访问外部类成员属性的方法,得到isFinished变量的值的。

但是当我们将isFinished写入main方法内,定义为局部变量时,程序运行失败。

以上是由于lambda表达式是回调函数,执行时机是真正创建出线程之后,因此很有可能出现线程在要创建时,main线程已经执行结束,其中的isFinished局部变量已经销毁。

所以lambda里面想要使用外部变量时,会触发变量捕获机制,将外面的局部变量拷贝一份到lambda里面,这样无论外面的局部变量是否销毁,都不影响线程的创建了。

但是变量捕获需要满足该变量为final或者实际上final,变量是不允许修改的。

java的Thread类中提供了现成的方法.interrupt()和变量isInterrupted()来中断线程和判断线程是否中断。

public class demo6 { //public static boolean isFinished=false; public static void main(String[] args) throws InterruptedException { Thread t=new Thread(()->{ while(!Thread.currentThread().isInterrupted()){ System.out.println("hello thread"); try { Thread.sleep(1000); } catch (InterruptedException e) { //throw new RuntimeException(e); break; } } System.out.println("t线程结束"); }); t.start(); Thread.sleep(6000); t.interrupt(); //isFinished=true; } }

使用.interrupt()方法不仅能设置中断标志位,还能唤醒sleep()这样的阻塞方法,唤醒sleep()以后就会抛出异常。

但是当我们不去break循环时,程序还会继续执行循环:

public class demo6 { //public static boolean isFinished=false; public static void main(String[] args) throws InterruptedException { Thread t=new Thread(()->{ while(!Thread.currentThread().isInterrupted()){ System.out.println("hello thread"); try { Thread.sleep(1000); } catch (InterruptedException e) { //throw new RuntimeException(e); //break; System.out.println("main线程尝试中断t线程"); } } System.out.println("t线程结束"); }); t.start(); Thread.sleep(6000); t.interrupt(); //isFinished=true; } }

这是由于sleep在被提前唤醒的情况下,把isInterrupted()标志位重新设置为false。这样t线程的终止权就在t线程自己手上。

6.线程等待

t.join() 表示main线程等待t线程结束。(可以理解为t线程插队到main线程前面,只有t结束,main才会执行。),

t.join(1000) 表示main线程至多等待t线程1000ms,如果不设置时间,那就等到天荒地老。

7.获取当前线程引用

public static Thread currentThread();

哪个线程调用这个静态方法就返回哪个线程的引用,作用类似于this。

8.线程休眠

需要注意的是,线程调度是不可控的,所以,实际上的休眠时间是大于参数设置的休眠时间的。

这是因为代码调用sleep,相当于线程让出了cpu资源,当设置的休眠时间到达以后,操作系统需要把线程重新调度到cpu上,才能继续执行。

时间到了,意味着允许被调度了,但不代表立刻被执行。

sleep(0)的用法,使当前进程立刻放弃cpu资源,等待操作系统重新调度。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/12 19:53:53

AI代码生成的PDCA框架实践指南

关键要点 将结构化目标设定循环应用于AI编码会话:运用计划-执行-检查-行动原则为每次会话设定明确、可观察的成功标准,并根据结果调整方向。对AI使用结构化任务级规划:让代理分析代码库,并将大型功能分解为可在短迭代内完成的小型…

作者头像 李华
网站建设 2026/5/12 7:10:41

SQL入门

一、什么是数据库数据库是按照数据结构来组织、存储和管理数据库的仓库。每个数据库都有一个活多个不同的API用于创建,访问,管理,搜索和复制所保存的数据。 二、术语数据库:数据库是一些关联表的集合。数据表:表是数…

作者头像 李华
网站建设 2026/5/12 20:48:41

Qwen3-VL-8B + Ollama下载:本地化多模态推理环境搭建

Qwen3-VL-8B Ollama下载:本地化多模态推理环境搭建 在智能应用日益依赖“看图说话”能力的今天,如何让一台普通工作站也能具备图像理解与自然语言交互的能力?这不再是大型科技公司的专属特权。随着轻量化多模态模型和本地运行框架的发展&…

作者头像 李华
网站建设 2026/5/19 5:30:19

此扩展程序不再受支持?vLLM社区活跃度更高

vLLM社区活跃度更高:为何它正在重塑大模型推理格局 在今天的AI服务部署中,一个现实问题摆在许多团队面前:曾经依赖的推理扩展工具逐渐停滞更新,GitHub仓库长时间无提交,文档陈旧,社区提问无人回应。与此同时…

作者头像 李华
网站建设 2026/5/19 1:45:52

处理机调度

目录 调度的概念、层次 进程调度的时机、方式、切换与过程 调度器、闲逛进程 调度算法的评价指标 CPU利用率:​编辑 系统吞吐量:​编辑 周转时间:​编辑 等待时间:​编辑 响应时间: ​编辑 调度算法 先来先服…

作者头像 李华