­

看完本文,再也不怕问java线程创建的几种方式了

  • 2019 年 10 月 4 日
  • 筆記
常规面试必问,线程创建的方式有哪几种,那么回答有如下几种:
  1. 继承Thread类
  2. 实现Runnable接口
  3. 实现Callable接口(JDK1.5+)

如果想了解具体怎么创建的,请往下看:

继承Thread类

//继承Thread  public class ExtendThread extends Thread{      //线程执行体      @Override      public void run() {          //do something          System.out.println("继承Thread创建线程");          //无返回值      }  }  public class ThreadCreateDemo {      public static void main(String[] args) {          //创建一个线程          ExtendThread extendThread = new ExtendThread();          //调用start方法启动线程          extendThread.start();           //没有返回值      }  }

输出:

继承Thread创建线程

注意

使用继承Thread类的方法来创建线程类时候,多个线程之间是无法共享线程类的实例变量的。

实现Runnable接口

覆写Runnable接口实现多线程可以避免单继承局限, 当子类实现Runnable接口,此时子类和Thread的代理模式(子类负责真实业务的操作,thread负责资源调度与线程创建辅助真实业务)。

//实现Runnable接口  public class ImplRunnable implements Runnable {      //线程实行体      @Override      public void run() {          //do something          System.out.println("实现Runnable创建线程");          //没有返回值      }  }  public class ThreadCreateDemo {      public static void main(String[] args) {          ImplRunnable implRunnable = new ImplRunnable();          Thread thread = new Thread(implRunnable);          //启动线程          thread.start();      }  }

输出:

实现Runnable创建线程

注意

Runnable对象仅仅作为Thread对象的target,Runnable实现类里包含的run方法仅仅作为线程的执行体,而实际的线程对象依旧是Thread实例,只是该Thread线程负责执行器target的方法。

覆写Callable接口

//实现Callable返回值类型为Integer类型  public class ImplCallable implements Callable<Integer> {      //该call()方法将作为线程执行体,并且有返回值      @Override      public Integer call() throws Exception {          //do something          System.out.println("实现Callable接口创建线程,返回类型为Integer类型");          return 999;      }  }  public class ThreadCreateDemo {      public static void main(String[] args) throws ExecutionException, InterruptedException {          Callable<Integer> callable = new ImplCallable();          FutureTask<Integer> futureTask = new FutureTask<>(callable);          Thread thread = new Thread(futureTask);          thread.start();          //获取返回值futureTask.get()          System.out.println(futureTask.get());      }  }

输出:

999

注意

Callable接口有泛型限制,Callable接口里的泛型形参类型与call方法返回值类型相同,而且Callable接口是函数式接口,因此可以使用Lambda表达式创建Callable对象。

三种方式的对比

通过继承Thread类或者实现Runnable接口、Callable接口都可以实现多线程,不过实现Runnable接口与实现Callable接口的方式基本相同,只是Callabl接口里定义的方法返回值,可以声明抛出异常而已。因此将实现Runnable接口和实现Callable接口归为一种方式。这种方式与继承Thread方式之间的主要差别如下。

采用实现Runnable、Callable接口的方式创建线程的优缺点:

优点

线程类只是实现了Runnable或者Callable接口,还可以继承其他类。这种方式下,多个线程可以共享一个target对象,所以非常适合多个相同线程来处理同一份资源的情况,从而可以将CPU、代码和数据分开,形成清晰的模型,较好的体现了面向对象的思想。

缺点

编程稍微复杂一些,如果需要访问当前线程,则必须使用

Thread.currentThread()方法

采 用继承Thread类的方式创建线程的优缺点:

缺点

因为线程类已经继承了Thread类,Java语言是单继承的,所以就不能再继承其他父类了。

优点

编写简单,如果需要访问当前线程,则无需使用

Thread.currentThread()方法,直接使用this即可获取当前线程

鉴于上面分析,因此一般建议采用实现Runnable或Callable接口来创建多线程