线程的状态
-
创建线程:
-
创建就进入新生状态
-
-
就绪状态:
-
调用start方法,线程立即进入就绪状态
-
-
运行状态:
-
cpu调度
-
-
阻塞状态:
-
调用sleep,wait或同步锁进入阻塞状态
-
-
死亡状态:
-
线程中断或者结束,一旦进入死亡状态,就不能再次启动。
-
停止线程 (stop)
-
不建议使用jdk提供的方法停止线程。
-
线程自己停止下来
-
建议使用标志位进行停止线程。
线程停止测试代码:
//测试stop //1.建议线程正常停止 --->利用次数,不建议死循环, //2.建议使用标志位--->设置一个标志位 //3.不要使用stop或者destroy等过时或者jdk不建议使用的方法 public class TestStop implements Runnable { //1.设置一个标志位 private boolean flag = true; @Override public void run() { int i = 0; while (flag){ System.out.println("run...Thread"+i++); } } //2.设置一个公开的方法停止线程,转换标志位 public void stop(){ this.flag = false; } public static void main(String[] args) { TestStop testStop = new TestStop(); new Thread(testStop).start(); for (int i = 0; i < 1000; i++) { System.out.println("main"+i); if(i == 900){ //调用stop方法切换标志位,让线程停止 testStop.stop(); System.out.println("线程该停止了"); } } } }
线程休眠:(sleep)
-
sleep(时间)指定当前线程阻塞的毫秒数;
-
sleep存在异常InterruptedException;
-
sleep时间到达后线程进入就绪状态。
-
sleep可以模拟网络延时,倒计时等。
-
每一个对象都有一个锁,sleep不会释放锁。
模拟倒计时,获取时间测试代码:
//模拟倒计时 public class TestSleep2 { public static void main(String[] args) { //打印当前系统时间 Date startTime = new Date(System.currentTimeMillis());//获取系统当前时间 while (true){ try { Thread.sleep(1000); System.out.println(new SimpleDateFormat("HH:mm:ss").format(startTime)); startTime = new Date(System.currentTimeMillis());//更新当前时间 } catch (InterruptedException e) { e.printStackTrace(); } } } //模拟倒计时 public static void tenDown() throws InterruptedException { int num = 10; while (true){ Thread.sleep(1000); System.out.println(num--); if(num<=0){ break; } } } }
线程礼让(yield)
-
礼让线程,让当前正在执行的线程暂停 ,但不阻塞
-
将线程从运行状态转为就绪状态
-
让cpu重新调度,礼让不一定成功。看cpu心情
测试礼让线程代码:
//测试礼让线程 //礼让不一定成功 ,看cpu心情 public class TestYield { public static void main(String[] args) { MyYield myYield = new MyYield(); new Thread(myYield,"a").start(); new Thread(myYield,"b").start(); } } class MyYield implements Runnable{ @Override public void run() { System.out.println(Thread.currentThread().getName()+"线程开始执行"); Thread.yield();//礼让 System.out.println(Thread.currentThread().getName()+"线程停止执行"); } }
线程强制执行:(Join)
-
Join合并线程,待次线程执行完成后,再执行其他线程,其他线程阻塞
-
可以想象成插队
测试代码:
//测试Join方法,想象为插队 public class TestJoin implements Runnable { @Override public void run() { for (int i = 0; i < 1000; i++) { System.out.println("线程vip来了"+i); } } public static void main(String[] args) throws InterruptedException { //启动线程 TestJoin testJoin = new TestJoin(); Thread thread = new Thread(testJoin); thread.start(); //主线程 for (int i = 0; i < 500; i++) { if(i==200){ thread.join();//插队 } System.out.println("main"+i); } } }
线程状态观测
new->runnable->blocked->waiting->dead
jdk帮助文档查看Thread.State
-
线程状态。
线程可以处于以下状态之一:
-
NEW
尚未启动的线程处于此状态。 -
RUNNABLE
在Java虚拟机中执行的线程处于此状态。 -
BLOCKED
被阻塞等待监视器锁定的线程处于此状态。 -
WAITING
正在等待另一个线程执行特定动作的线程处于此状态。 -
TIMED_WAITING
正在等待另一个线程执行动作达到指定等待时间的线程处于此状态。 -
TERMINATED
已退出的线程处于此状态。
一个线程可以在给定时间点处于一个状态。 这些状态是不反映任何操作系统线程状态的虚拟机状态。
线程测试状态代码:线程只能启动一次
//观察测试线程的状态 public class TestState { public static void main(String[] args) throws InterruptedException { Thread thread = new Thread(()->{ for (int i = 0; i < 5; i++) { try { Thread.sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } } System.out.println("///////"); }); //观察状态 Thread.State state = thread.getState(); System.out.println(state); //new //观察启动后 thread.start();//启动线程 state = thread.getState(); System.out.println(state); while (state != Thread.State.TERMINATED){ //只要线程不终止,就一直输出状态 Thread.sleep(100); state = thread.getState(); //更新线程状态 System.out.println(state); //输出状态 } } }
-
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 使用C#创建一个MCP客户端
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列1:轻松3步本地部署deepseek,普通电脑可用
· 按钮权限的设计及实现