多线程 - 创建线程的方法
总结
https://www.cnblogs.com/3s540/p/7172146.html;https://zhuanlan.zhihu.com/p/82339746
1 | Thread类是对线程本身的抽象,Runnable/Callable是对业务动作的抽象 |
Java中创建线程主要有三种方式:
- 继承Thread类
- 实现Runnable接口
- 使用Callable和Future
方法1:继承java.lang.Thead类创建线程
(1)继承java.lang.Thread类并重写run方法 (此处的“重写”不是语法强制重写,因为Thread类已经重写过。只不过为了自己的逻辑需要重写)
(2)创建线程对象
(3)调用该线程对象的start()方法来启动线程。注意:不是run方法(), 区别请看这里:多线程 - Thread.start() vs Thread.run()
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 | public class CreateThreadTest { public static void main(String[] args) { new ThreadTest().start(); new ThreadTest().start(); } } class ThreadTest extends Thread { private int i = 0 ; @Override public void run() { for (; i < 100 ; i++) { System.out.println(Thread.currentThread().getName() + " is running: " + i); } } } |
方法2:实现java.lang.Runnable接口创建线程
(1)定义一个类实现java.lang.Runnable接口,并重写该接口的run()方法。(语法强制规定的重写)
(2)创建 Runnable实现类的对象,作为创建Thread对象的target参数,此Thread对象才是真正的线程对象
(3)调用线程对象的start()方法来启动线程
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 | public class CreateThreadTest { public static void main(String[] args) { RunnableTest runnableTest = new RunnableTest(); new Thread(runnableTest, "线程1" ).start(); new Thread(runnableTest, "线程2" ).start(); } } class RunnableTest implements Runnable { private int i = 0 ; @Override public void run() { for (; i < 100 ; i++) { System.out.println(Thread.currentThread().getName() + " is running: " + i); } } } |
方法3:实现java.util.concurrent的Callable和Future接口创建线程
和Runnable接口不一样,Callable接口提供了一个call()方法作为线程执行体,call()方法比run()方法功能要强大:call()方法可以
- 有返回值,
- 可以声明抛出异常
1 2 3 | public interface Callable<V> { V call() throws Exception; } |
以下几点要注意:
- Java5提供了Future接口来接收Callable接口中call()方法的返回值。
- Callable接口是 Java5 新增的接口,不是Runnable接口的子接口,所以Callable对象不能直接作为Thread对象的target。
- 针对这个问题,引入了RunnableFuture接口,RunnableFuture接口实现了Runnable接口和Future接口,可以作为Thread对象的target 。
- 同时,Java5提供了一个RunnableFuture接口的实现类:FutureTask 。FutureTask可以作为Thread对象的target。
介绍了相关概念之后,使用Callable和Future创建线程的步骤如下:
(1)定义一个类实现Callable接口,并重写(语法强制重写)call()方法,该call()方法将作为线程执行体,并且有返回值
(2)创建Callable实现类的实例,使用FutureTask类来包装Callable对象
(3)使用FutureTask对象作为Thread对象的target创建并启动线程
(4)调用FutureTask对象的get()方法来获得子线程执行结束后的返回值
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 | import java.util.concurrent.Callable; import java.util.concurrent.ExecutionException; import java.util.concurrent.FutureTask; public class CreateThreadTest { public static void main(String[] args) { CallableTest callableTest = new CallableTest(); FutureTask<Integer> futureTask = new FutureTask<>(callableTest); new Thread(futureTask).start(); try { System.out.println( "子线程的返回值: " + futureTask.get()); } catch (InterruptedException e) { e.printStackTrace(); } catch (ExecutionException e) { e.printStackTrace(); } } } class CallableTest implements Callable{ @Override public Integer call() throws Exception { int sum = 0 ; for ( int i = 1 ; i < 101 ; i++) { sum += i; } System.out.println(Thread.currentThread().getName() + " is running: " + sum); return sum; } } |
创建线程的三种方式的对比
1.实现Runnable/Callable接口相比继承Thread类的优势
(1)适合多个线程进行资源共享
(2)可以避免java中单继承的限制
(3)增加程序的健壮性,代码和数据独立
(4)线程池只能放入Runable或Callable接口实现类,不能直接放入继承Thread的类
2.Callable和Runnable的区别
(1) Callable重写的是call()方法,Runnable重写的方法是run()方法
(2) call()方法执行后可以有返回值,run()方法没有返回值
(3) call()方法可以抛出异常,run()方法不可以
(4) 运行Callable任务可以拿到一个Future对象,表示异步计算的结果 。通过Future对象可以了解任务执行情况,可取消任务的执行,还可获取执行结果
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
· 浏览器原生「磁吸」效果!Anchor Positioning 锚点定位神器解析
· DeepSeek 开源周回顾「GitHub 热点速览」
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
· AI与.NET技术实操系列(二):开始使用ML.NET
· 单线程的Redis速度为什么快?