登录
首页 >  文章 >  java教程

Thread.start() 与直接调用 run() 的主要区别在于线程的执行方式和生命周期管理。以下是详细对比:1. start() 方法作用:启动一个新线程,使线程进入就绪状态,等待 CPU 调度执行。执行方式:start() 会调用 JVM 的底层机制(如操作系统线程),创建一个新的线程。新线程会独立于主线程运行,实现真正的并行执行。生命周期:线程从“新建”状态进入“就绪”状态,最终执行 r

时间:2026-05-22 09:39:17 163浏览 收藏

你是否曾困惑于为什么调用 `Thread.start()` 能实现真正的并发,而直接调用 `run()` 却只是“假多线程”?关键在于:`start()` 是 JVM 向操作系统申请资源、创建独立线程并自动触发 `run()` 的入口开关,它让代码在全新线程(如 Thread-0)中并行执行,不阻塞主线程,支持多核调度;而 `run()` 仅仅是普通方法调用,所有逻辑仍在当前线程(永远是 main)中串行执行,毫无并发可言——更需警惕的是,`start()` 只能调一次,重复调用会抛异常,而 `run()` 却可无限重放。理解这一本质区别,是写出高效、健壮多线程程序的第一步。

如何理解Thread类中start方法与直接调用run方法的区别

关键就一点:start() 是启动新线程,run() 只是普通方法调用。

start() 真正开启并发执行

调用 start() 后,JVM 会向操作系统申请资源,创建一个独立的线程,把当前 Thread 对象的状态从 NEW 改为 RUNNABLE,并安排它排队等待 CPU。一旦获得时间片,这个新线程就会自动执行 run() 方法里的代码——但不是你手动调的,是系统在新线程上下文中触发的。

  • 输出中能看到线程名是 Thread-0、Thread-1,不是 main
  • t1.start() 和 t2.start() 的执行顺序不确定,可能交叉打印
  • 主线程不会卡住等它,可以继续往下跑自己的逻辑

run() 调用不产生新线程

直接写 thread.run(),跟调用 thread.doSomething() 没区别。JVM 完全不介入线程调度,所有代码都在当前线程(比如 main)里顺序执行,栈帧也压在当前线程的调用栈上。

  • 输出线程名永远是 main
  • t1.run(); t2.run(); 一定是先完 t1 再完 t2,严格串行
  • 主线程会一直等到 run() 执行完才执行下一行

调用规则完全不同

start() 是“一次性开关”,run() 是“可反复执行的函数”。

  • 同一个 Thread 实例,start() 只能调一次;第二次调就抛 IllegalThreadStateException
  • run() 可以调十次、一百次,只要对象没被回收,每次都是在当前线程重跑一遍逻辑
  • 线程一旦终止(run 执行完或异常退出),就不能再 start —— 它不是重启按钮

底层机制差异明显

start() 是 native 方法,背后调的是操作系统的 pthread_create 或类似 API;run() 就是纯 Java 字节码,走的是常规方法调用流程。

  • start() 成功后,你会看到两个独立的调用栈:main 线程一个,新线程一个
  • run() 调用时,整个过程只在一个调用栈里展开
  • 多核 CPU 能同时跑多个线程,但永远无法靠多次 run() 实现并行

到这里,我们也就讲完了《Thread.start() 与直接调用 run() 的主要区别在于线程的执行方式和生命周期管理。以下是详细对比:1. start() 方法作用:启动一个新线程,使线程进入就绪状态,等待 CPU 调度执行。执行方式:start() 会调用 JVM 的底层机制(如操作系统线程),创建一个新的线程。新线程会独立于主线程运行,实现真正的并行执行。生命周期:线程从“新建”状态进入“就绪”状态,最终执行 run() 方法。优点:支持多线程并发,适合处理耗时任务。避免阻塞主线程,提升程序响应性。2. 直接调用 run() 方法作用:直接在当前线程中执行 run() 方法的内容,不创建新线程。执行方式:run() 是一个普通方法,直接调用相当于顺序执行代码。所有操作都在主线程中完成,不会产生并发。生命周期:线程始终处于“运行”状态,没有真正启动。缺点:无法实现真正的多线程,可能阻塞主线程》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于的知识点!

资料下载
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>