Java 线程创建的 3 种方式

228 阅读2分钟

Java 线程创建的 3 种方式

1 通过继承 Thread 类

//匿名 Thread 类
new Thread(){
    //重写 run 方法
     @Override
    public void run() {
       //do
    }
}.start();
//继承 Thread 类
class MyThread extends Thread {
    @Override
    public void run() {
       //do
    }
}
//假设把 run 方法看成一个任务,那么每创建一个线程,就有一个新的任务
 MyThread myThread = new MyThread();
 myThread.start();
  • Thread 也实现了 Runnable 接口,需要在 run 方法内捕获处理异常,出现异常会导致线程终止
  • 多个线程相互独立,因为创建了多个任务,无法实现资源共享(资源写在 MyThread 类里)
  • Java 单继承的特性导致不够灵活

2 通过实现 Runnable 接口,然后传入 Thread 启动

//需要借助 Thread 类
new Thread(new Runnable() {
   //重写 run 方法  
     @Override
     public void run() {
       //do
     }
}).start();
class MyRunnable implements Runnable {
          @Override
         public void run() {
          //do
         }
}
//其实还是通过 Thread 类对象来创建线程的,每创建一个线程,传入的可以是同一个任务
new Thread(new MyRunnable()).start();
  • 多个线程协调作业,完成同一个任务,实现了资源共享(资源写在 MyRunnable 类里)
  • 可以避免由于 java 单继承特征带来的局限性
  • Runnable 可以配合 ExecutorService 线程池的 execute 方法使用

3. 通过实现 Callable 接口,借助 FutureTask 类,然后传入 Thread 启动

//同样需要借助 Thread 类
 new Thread(new FutureTask<String>(new Callable<String>() {
            @Override
            public String call() throws Exception {
                return null;
            }
})).start();
class MyCallable implements Callable<String>{
     @Override
     public String call() throws Exception {
            return null;
     }
}
MyCallable myCallable = new MyCallable();
//FutureTask 实现了 RunnableFuture<V> 接口,而 RunnableFuture 继承了 Runnable 和 Future<V>
FutureTask<String> futureTask = new FutureTask<>(myCallable);
//Runnable 的特性
new Thread(futureTask).start();
//获取返回值, Future<V> 的特性,注意 get 会阻塞当前线程
String result = futureTask.get();
  • Callable 类似于 Runnable 接口
  • 可以有返回值,可以抛出异常
  • 可以配合 ExecutorService 线程池的 submit 方法使用

小结:三种方式最终都是通过 Thread 类对象来真正创建启动线程的