1. 创建线程的三种方法及其区别

1.1 继承Thread类

首先,定义Thread类的子类并重写run()方法:

  1. package com.zwwhnly.springbootaction.javabase.thread;
  2. public class MyFirstThread extends Thread {
  3. @Override
  4. public void run() {
  5. for (int i = 0; i < 5; i++) {
  6. System.out.printf("[MyFirstThread]输出:%d,当前线程名称:%s\n",
  7. i, getName());
  8. }
  9. }
  10. }

然后,创建该子类的实例并调用start()方法启动线程:

  1. package com.zwwhnly.springbootaction.javabase.thread;
  2. public class ThreadTest {
  3. public static void main(String[] args) {
  4. System.out.println("主线程开始执行,当前线程名称:" +
  5. Thread.currentThread().getName());
  6. Thread firstThread = new MyFirstThread();
  7. firstThread.start();
  8. System.out.println("主线程执行结束,当前线程名称:" +
  9. Thread.currentThread().getName());
  10. }
  11. }

运行结果如下所示:

主线程开始执行,当前线程名称:main

主线程执行结束,当前线程名称:main

[MyFirstThread]输出:0,当前线程名称:Thread-0

[MyFirstThread]输出:1,当前线程名称:Thread-0

[MyFirstThread]输出:2,当前线程名称:Thread-0

[MyFirstThread]输出:3,当前线程名称:Thread-0

[MyFirstThread]输出:4,当前线程名称:Thread-0

从运行结果可以看出以下2个问题:

  1. 程序中存在2个线程,分别为主线程main和自定义的线程Thread-0。
  2. 调用firstThread.start();,run()方法体中的代码并没有立即执行,而是异步执行的。

查看Thread类的源码,可以发现Thread类实现了接口Runnable:

  1. public class Thread implements Runnable {
  2. // 省略其它代码

这里是重点,面试常问!

1.2 实现Runnable接口(推荐)

首先,定义Runnable接口的实现类并实现run()方法:

  1. package com.zwwhnly.springbootaction.javabase.thread;
  2. public class MySecondThread implements Runnable {
  3. @Override
  4. public void run() {
  5. for (int i = 0; i < 5; i++) {
  6. System.out.printf("[MySecondThread]输出:%d,当前线程名称:%s\n",
  7. i, Thread.currentThread().getName());
  8. }
  9. }
  10. }

然后,调用Thread类的构造函数创建Thread实例并调用start()方法启动线程:

  1. package com.zwwhnly.springbootaction.javabase.thread;
  2. public class ThreadTest {
  3. public static void main(String[] args) {
  4. Runnable target = new MySecondThread();
  5. Thread secondThread = new Thread(target);
  6. secondThread.start();
  7. }
  8. }

运行结果如下所示:

主线程开始执行,当前线程名称:main

主线程执行结束,当前线程名称:main

[MySecondThread]输出:0,当前线程名称:Thread-0

[MySecondThread]输出:1,当前线程名称:Thread-0

[MySecondThread]输出:2,当前线程名称:Thread-0

[MySecondThread]输出:3,当前线程名称:Thread-0

[MySecondThread]输出:4,当前线程名称:Thread-0

可以看出,使用这种方式和继承Thread类的运行结果是一样的。

1.3 实现Callable接口

首先,定义Callable接口的实现类并实现call()方法:

  1. package com.zwwhnly.springbootaction.javabase.thread;
  2. import java.util.Random;
  3. import java.util.concurrent.Callable;
  4. public class MyThirdThread implements Callable<Integer> {
  5. @Override
  6. public Integer call() throws Exception {
  7. Thread.sleep(6 * 1000);
  8. return new Random().nextInt();
  9. }
  10. }

然后,调用FutureTask类的构造函数创建FutureTask实例:

  1. Callable<Integer> callable = new MyThirdThread();
  2. FutureTask<Integer> futureTask = new FutureTask<>(callable);

最后,调用Thread类的构造函数创建Thread实例并调用start()方法启动线程:

  1. package com.zwwhnly.springbootaction.javabase.thread;
  2. import java.util.concurrent.Callable;
  3. import java.util.concurrent.ExecutionException;
  4. import java.util.concurrent.FutureTask;
  5. public class ThreadTest {
  6. public static void main(String[] args) {
  7. System.out.println("主线程开始执行,当前线程名称:" +
  8. Thread.currentThread().getName());
  9. Callable<Integer> callable = new MyThirdThread();
  10. FutureTask<Integer> futureTask = new FutureTask<>(callable);
  11. new Thread(futureTask).start();
  12. try {
  13. System.out.println("futureTask.isDone() return:" + futureTask.isDone());
  14. System.out.println(futureTask.get());
  15. System.out.println("futureTask.isDone() return:" + futureTask.isDone());
  16. } catch (InterruptedException | ExecutionException e) {
  17. e.printStackTrace();
  18. }
  19. System.out.println("主线程执行结束,当前线程名称:" +
  20. Thread.currentThread().getName());
  21. }
  22. }

运行结果如下所示:

主线程开始执行,当前线程名称:main

futureTask.isDone() return:false

-1193053528

futureTask.isDone() return:true

主线程执行结束,当前线程名称:main

可以发现,使用Callable接口这种方式,我们可以通过futureTask.get()获取到线程的执行结果,而之前的2种方式,都是没有返回值的。

注意事项:调用futureTask.get()获取线程的执行结果时,主线程会阻塞直到获取到结果。

阻塞效果如下图所示:

1.4 区别

以下是重点,面试常问!

  1. Java中,类仅支持单继承,如果一个类继承了Thread类,就无法再继承其它类,因此,如果一个类既要继承其它的类,又必须创建为一个线程,就可以使用实现Runable接口的方式。
  2. 使用实现Runable接口的方式创建的线程可以处理同一资源,实现资源的共享。
  3. 使用实现Callable接口的方式创建的线程,可以获取到线程执行的返回值、是否执行完成等信息。

关于第2点,可以通过如下示例来理解。

假如我们总共有10张票(共享的资源),为了提升售票的效率,开了3个线程来售卖,代码如下所示:

  1. package com.zwwhnly.springbootaction.javabase.thread;
  2. public class SaleTicketThread implements Runnable {
  3. private int quantity = 10;
  4. @Override
  5. public void run() {
  6. while (quantity > 0) {
  7. System.out.println(quantity-- + " is saled by " +
  8. Thread.currentThread().getName());
  9. }
  10. }
  11. }
  1. public static void main(String[] args) {
  2. Runnable runnable = new SaleTicketThread();
  3. Thread saleTicketThread1 = new Thread(runnable);
  4. Thread saleTicketThread2 = new Thread(runnable);
  5. Thread saleTicketThread3 = new Thread(runnable);
  6. saleTicketThread1.start();
  7. saleTicketThread2.start();
  8. saleTicketThread3.start();
  9. }

因为3个线程都是异步执行的,因此每次的运行结果可能是不一样,以下列举2次不同的运行结果。

第1次运行结果:

10 is saled by Thread-0

8 is saled by Thread-0

7 is saled by Thread-0

5 is saled by Thread-0

9 is saled by Thread-1

3 is saled by Thread-1

2 is saled by Thread-1

1 is saled by Thread-1

4 is saled by Thread-0

6 is saled by Thread-2

第2次运行结果:

10 is saled by Thread-0

9 is saled by Thread-0

8 is saled by Thread-0

7 is saled by Thread-0

6 is saled by Thread-0

5 is saled by Thread-0

3 is saled by Thread-0

2 is saled by Thread-0

4 is saled by Thread-2

1 is saled by Thread-1

如果将上面的SaleTicketThread修改成继承Thread类的方式,就变成了3个线程各自拥有10张票,即变成了30张票,而不是3个线程共享10张票。

2. Thread类start()和run()的区别

2.1 示例

因为实现Runnable接口的优势,基本上实现多线程都使用的是该种方式,所以我们将之前定义的MyFirstThread也修改为实现Runnable接口的方式:

  1. package com.zwwhnly.springbootaction.javabase.thread;
  2. public class MyFirstThread implements Runnable {
  3. @Override
  4. public void run() {
  5. for (int i = 0; i < 5; i++) {
  6. System.out.printf("[MyFirstThread]输出:%d,当前线程名称:%s\n",
  7. i, Thread.currentThread().getName());
  8. }
  9. }
  10. }

然后仍然沿用之前定义的MyFirstThread、MySecondThread,我们先看下调用start()的效果:

  1. package com.zwwhnly.springbootaction.javabase.thread;
  2. public class ThreadTest {
  3. public static void main(String[] args) {
  4. System.out.println("主线程开始执行,当前线程名称:" +
  5. Thread.currentThread().getName());
  6. Thread firstThread = new Thread(new MyFirstThread());
  7. Runnable target = new MySecondThread();
  8. Thread secondThread = new Thread(target);
  9. firstThread.start();
  10. secondThread.start();
  11. System.out.println("主线程执行结束,当前线程名称:" +
  12. Thread.currentThread().getName());
  13. }
  14. }

运行结果(注意:多次运行,结果可能不一样):

主线程开始执行,当前线程名称:main

[MyFirstThread]输出:0,当前线程名称:Thread-0

[MyFirstThread]输出:1,当前线程名称:Thread-0

[MySecondThread]输出:0,当前线程名称:Thread-1

主线程执行结束,当前线程名称:main

[MySecondThread]输出:1,当前线程名称:Thread-1

[MySecondThread]输出:2,当前线程名称:Thread-1

[MySecondThread]输出:3,当前线程名称:Thread-1

[MySecondThread]输出:4,当前线程名称:Thread-1

[MyFirstThread]输出:2,当前线程名称:Thread-0

[MyFirstThread]输出:3,当前线程名称:Thread-0

[MyFirstThread]输出:4,当前线程名称:Thread-0

可以看出,调用start()方法后,程序中有3个线程,分别为主线程main、Thread-0、Thread-1,而且执行顺序不是按顺序执行的,存在不确定性。

然后将start()方法修改为run()方法,如下所示:

  1. firstThread.run();
  2. secondThread.run();

此时的运行结果如下所示(多次运行,结果是一样的):

主线程开始执行,当前线程名称:main

[MyFirstThread]输出:0,当前线程名称:main

[MyFirstThread]输出:1,当前线程名称:main

[MyFirstThread]输出:2,当前线程名称:main

[MyFirstThread]输出:3,当前线程名称:main

[MyFirstThread]输出:4,当前线程名称:main

[MySecondThread]输出:0,当前线程名称:main

[MySecondThread]输出:1,当前线程名称:main

[MySecondThread]输出:2,当前线程名称:main

[MySecondThread]输出:3,当前线程名称:main

[MySecondThread]输出:4,当前线程名称:main

主线程执行结束,当前线程名称:main

可以看出,调用run()方法后,程序中只有一个主线程,自定义的2个线程并没有启动,而且执行顺序也是按顺序执行的。

1.2 总结

以下是重点,面试常问!

  • run()方法只是一个普通方法,调用之后程序会等待run()方法执行完毕,所以是串行执行,而不是并行执行。
  • start()方法会启动一个线程,当线程得到CPU资源后会自动执行run()方法体中的内容,实现真正的并发执行。

3. Runnable和Callable的区别

在文章前面的章节中(1.2 实现Runnable接口 和1.3 实现Callable接口),我们了解了如何使用Runnable、Callable接口来创建线程,现在我们分别看下Runable和Callable接口的定义,其中,Runable接口的定义如下所示:

  1. public interface Runnable {
  2. public abstract void run();
  3. }

Callable接口的定义如下所示:

  1. public interface Callable<V> {
  2. V call() throws Exception;
  3. }

由此可以看出,Runnable和Callable的区别主要有以下几点:

  1. Runable的执行方法是run(),Callable的执行方法是call()
  2. call()方法可以抛出异常,run()方法如果有异常只能在内部消化
  3. 实现Runnable接口的线程没有返回值,实现Callable接口的线程能返回执行结果
  4. 实现Callable接口的线程,可以和FutureTask一起使用,获取到线程是否完成、线程是否取消、线程执行结果,也可以取消线程的执行。

4. 源码及参考

源码地址:https://github.com/zwwhnly/springboot-action.git,欢迎下载。

Java中实现多线程的两种方式之间的区别

Java Thread 的 run() 与 start() 的区别

Java Runnable与Callable区别

Callable,Runnable比较及用法

Runnable和Callable的区别和用法

《Java多线程面试题》系列-创建线程的三种方法及其区别的更多相关文章

  1. Java中创建线程的三种方法以及区别

    Java使用Thread类代表线程,所有的线程对象都必须是Thread类或其子类的实例.Java可以用三种方式来创建线程,如下所示: 1)继承Thread类创建线程 2)实现Runnable接口创建线 ...

  2. java创建线程的三种方法

    这里不会贴代码,只是将创建线程的三种方法做个笼统的介绍,再根据源码添加上自己的分析. 通过三种方法可以创建java线程: 1.继承Thread类. 2.实现Runnable接口. 3.实现Callab ...

  3. Java并发编程(二)创建线程的三种方法

    进程与线程 1.  进程 进程和代码之间的关系就像音乐和乐谱之间的关系一样,演奏结束的时候音乐就不存在了但乐谱还在:程序执行结束的时候进程就消失了但代码还在,而计算机就是代码的演奏家. 2. 线程 线 ...

  4. python 多线程编程之threading模块(Thread类)创建线程的三种方法

    摘录 python核心编程 上节介绍的thread模块,是不支持守护线程的.当主线程退出的时候,所有的子线程都将终止,不管他们是否仍在工作. 本节开始,我们开始介绍python的另外多线程模块thre ...

  5. java多线程(一)创建线程的四种方式

    1.   什么是并发与并行 要想学习多线程,必须先理解什么是并发与并行 并行:指两个或多个事件在同一时刻发生(同时发生). 并发:指两个或多个事件在同一个时间段内发生. 2.   什么是进程.线程 进 ...

  6. JAVA中创建线程的三种方法及比较

    JAVA中创建线程的方式有三种,各有优缺点,具体如下: 一.继承Thread类来创建线程 1.创建一个任务类,继承Thread线程类,因为Thread类已经实现了Runnable接口,然后重写run( ...

  7. Java中创建线程的三种方式以及区别

    在java中如果要创建线程的话,一般有3种方法: 继承Thread类: 实现Runnable接口: 使用Callable和Future创建线程. 1. 继承Thread类 继承Thread类的话,必须 ...

  8. 27 多线程(一)——创建进程的三种方法、线程锁(同步synchornized与lock)

    线程的流程 线程的创建 有三种方法,重点掌握前两种: 继承Thread类 实现Runnable接口(推荐使用:避免单继承的局限性) 实现Callable接口 根据java的思想,要少用继承,多用实现. ...

  9. java中创建线程的几种方法及区别

    1,实现Runnable接口创建线程 特点: A:将代码和数据分开,形成清晰的模型 B:线程体run()方法所在的类可以从其它类中继承一些有用的属性和方法 C:有利于保持程序风格的一致性 2,继承Th ...

随机推荐

  1. PhpSpreadsheet 导出特定格式 — 广告请款单

    需求说明 最近需要实现一个导出这种格式的Excel表单,之前都有用过导出Excel的功能,但大都是表头+数据的形式,只用于获取数据,没有太多样式要求,不用合并单元格.合并居中等,也不用对每一行数据特异 ...

  2. Python学习笔记五(读取提取写入文件)

    #Python打开读取一个文件内容,然后写入一个新的文件中,并对某些字段进行提取,写入新的字段的脚本,与大家共同学习. import os import re def get_filelist(dir ...

  3. Java ThreadLocal 的使用与源码解析

    GitHub Page: http://blog.cloudli.top/posts/Java-ThreadLocal-的使用与源码解析/ ThreadLocal 主要解决的是每个线程绑定自己的值,可 ...

  4. PHP array_unshift

    1.函数的作用:在数组的开头插入一个或者多个元素 2.函数的参数: @params  array  &$array @params  mixed $value1 @params  mixed ...

  5. 三种常见字符编码:ASCII、Unicode和UTF-8

    什么是字符编码? 计算机只能处理数字,如果要处理文本,就必须先把文本转换为数字才能处理.最早的计算机在设计时采用8个比特(bit)作为一个字节(byte),所以,一个字节能表示的最大的整数就是255( ...

  6. C# 结合 Golang 开发

    1. 实现方式与语法形式 基本方式:将 Go 程序编译成 DLL 供 C# 调用. 1.1 Go代码 注意:代码中 export 的注释是定义的入口描述不能省略 package main import ...

  7. Ubuntu 19.10 发布 | 云原生生态周报 Vol. 24

    作者 | 木苏.进超.冬岛.元毅.心水.衷源 业界要闻 1.云原生编程语言 Pulumi 1.0 pulumi ,一款中立的开源云开发平台,Pulumi 支持多语言.混合云环境.完全可扩展.初期支持 ...

  8. 你真的了解 volatile 关键字吗?

    今天,让我们一起来探讨 Java 并发编程中的知识点:volatile 关键字 本文主要从以下三点讲解 volatile 关键字: volatile 关键字是什么? volatile 关键字能解决什么 ...

  9. (二)django--带APP的网站

    1.打开终端,进入到django项目,创建APP应用:python manage.py startapp news 2.在settings.py中进行注册 3.在news下新建views.py,和ur ...

  10. Ubuntu svn 安装 Rabbitvcs

    先添加源 sudo add-apt-repository ppa:rabbitvcs/ppa 必要的话在源清单里面也添加一下 sudo gedit /etc/apt/sources.list 内容是 ...