在Java中如何创建线程并启动执行_Java线程创建方式与执行流程解析

最简单线程创建方式是继承Thread类重写run()并调用start();直接调用run()不会开启新线程;一个Thread实例只能start()一次;推荐使用Runnable+Thread更灵活;需返回值或异常处理时用Callable+Future。

直接用 Thread 类创建并启动线

程最简单

Java 中最基础的线程创建方式是继承 Thread 类,重写 run() 方法,然后调用 start() 启动。注意:不能直接调用 run(),否则只是普通方法调用,不会开启新线程。

常见错误现象:thread.run() 看似执行了,但主线程阻塞等待、CPU 时间片没切走、isAlive() 始终返回 false

  • start() 才真正触发 JVM 创建 OS 级线程,并调度执行 run()
  • 一个 Thread 实例只能 start() 一次,重复调用抛 IllegalThreadStateException
  • 子类中若需复用父类逻辑,别忘了在 run() 开头加 super.run()(虽然 Thread.run() 默认空实现)
class MyThread extends Thread {
    public void run() {
        System.out.println("线程 " + Thread.currentThread().getName() + " 正在运行");
    }
}
// 使用
MyThread t = new MyThread();
t.start(); // ✅ 正确
// t.run();   ❌ 不会新建线程

Runnable + Thread 更灵活,推荐日常使用

相比继承,实现 Runnable 接口更符合面向对象设计原则(避免单继承限制),也更适合传递任务逻辑。实际开发中绝大多数场景都走这条路。

使用场景:需要把任务逻辑和线程生命周期解耦;多个线程执行同一份任务代码;配合线程池使用(ExecutorService.submit(Runnable))。

  • Runnable 是函数式接口,可用 lambda 表达式简化:() -> System.out.println("hello")
  • 传入 Thread(Runnable) 构造器后,仍需调用 start(),不是自动执行
  • lambda 中若访问局部变量,该变量必须是 final 或“事实 final”(Java 8+)
Runnable task = () -> {
    System.out.println("任务由 " + Thread.currentThread().getName() + " 执行");
};
Thread t = new Thread(task, "worker-1");
t.start();

CallableFuture 用于需要返回值或异常处理的线程

如果线程执行完要返回结果,或可能抛出受检异常,Runnable 就不够用了——它没有返回值,且 run() 方法不能声明抛异常。

这时候改用 Callable:它的 call() 方法有返回值、可抛异常;再配合 Future 获取结果(支持阻塞等待、超时、取消)。

  • Future.get() 是阻塞调用,不设超时可能永久挂起,生产环境务必用 get(long, TimeUnit)
  • ExecutorService.submit(Callable) 返回 Future,而 execute(Runnable) 不返回任何东西
  • Callable 不能直接传给 Thread 构造器,必须包装成 FutureTask(它同时实现了 RunnableFuture
Callable task = () -> {
    Thread.sleep(100);
    return 42;
};
FutureTask ft = new FutureTask<>(task);
new Thread(ft).start();
System.out.println(ft.get()); // 阻塞直到结果就绪

线程启动失败的几个典型原因和检查点

即使代码语法正确,线程也可能“看似没执行”或“启动即死”,多数不是 Java 层问题,而是资源或状态误判。

容易被忽略的地方:

  • JVM 已接近最大线程数(-Xss 太大或系统 ulimit -u 限制过小),start()OutOfMemoryError: unable to create native thread
  • 线程 run() 方法体为空或瞬间结束,导致 isAlive() 查不到存活态,误以为没启动
  • 日志输出被缓冲(尤其 System.out),加上线程退出太快,看不到打印——加 System.out.flush() 或换 System.err 测试
  • IDE 调试时启用了“suspend on exception”,而线程内未捕获异常导致静默终止(建议在线程内加顶层 try-catch 打印堆栈)

复杂点在于:线程的生命周期(NEW → RUNNABLE → BLOCKED/WAITING → TIMED_WAITING → TERMINATED)不是全由 Java 代码控制,OS 调度、GC 暂停、锁竞争都会影响状态观测。别依赖 isAlive() 的瞬时结果做关键判断。