Java创建线程的三种方式
一、继承Thread类创建
通过继承Thread并且重写其run(),run方法中即线程执行任务。创建后的子类通过调用 start() 方法即可执行线程方法。
通过继承Thread实现的线程类,多个线程间无法共享线程类的实例变量。(需要创建不同Thread对象,自然不共享)
例子:
/** * 通过继承Thread实现线程 */ public class ThreadTest extends Thread{ private int i = 0 ; @Override public void run() { for(;i<50;i++){ System.out.println(Thread.currentThread().getName() + " is running " + i ); } } public static void main(String[] args) { for(int j=0;j<50;j++){if(j=20){ new ThreadTest().start() ; new ThreadTest().start() ; } } } }
二、 通过Runnable接口创建线程类
该方法需要先 定义一个类实现Runnable接口,并重写该接口的 run() 方法,此run方法是线程执行体。
接着创建 Runnable实现类的对象,作为创建Thread对象的参数target,此Thread对象才是真正的线程对象。通过实现Runnable接口的线程类,是互相共享资源的。
/** * 通过实现Runnable接口实现的线程类 */ public class RunnableTest implements Runnable { private int i ; @Override public void run() { for(;i<50;i++){ System.out.println(Thread.currentThread().getName() + " -- " + i); } } public static void main(String[] args) { for(int i=0;i<100;i++){ System.out.println(Thread.currentThread().getName() + " -- " + i); if(i==20){ RunnableTest runnableTest = new RunnableTest() ; new Thread(runnableTest,"线程1").start() ; new Thread(runnableTest,"线程2").start() ; } } } }
三、 使用Callable和Future创建线程
从继承Thread类和实现Runnable接口可以看出,上述两种方法都不能有返回值,且不能声明抛出异常。而Callable接口则实现了此两点,Callable接口如同Runable接口的升级版,其提供的call()方法将作为线程的执行体,同时允许有返回值。
但是Callable对象不能直接作为Thread对象的target,因为Callable接口是 Java 5 新增的接口,不是Runnable接口的子接口。
对于这个问题的解决方案,就引入 Future接口,此接口可以接受call() 的返回值,Future接口是Runnable接口的子接口,可以作为Thread对象的target 。并且, Future 接口提供了一个实现类:FutureTask 。
FutureTask实现了Future接口、Runnable接口 ,可以作为 Thread对象的target。
关系如下:
import java.util.concurrent.Callable; import java.util.concurrent.FutureTask; public class CallableTest { public static void main(String[] args) { CallableTest callableTest = new CallableTest() ; //因为Callable接口是函数式接口,可以使用Lambda表达式 FutureTask task = new FutureTask((Callable)()->{ int i = 0 ; for(;i<100;i++){ System.out.println(Thread.currentThread().getName() + "的循环变量i的值 :" + i); } return i; }); for(int i=0;i<100;i++){ System.out.println(Thread.currentThread().getName()+" 的循环变量i : + i"); if(i==20){ new Thread(task,"有返回值的线程").start(); } } try{ System.out.println("子线程返回值 : " + task.get()); }catch (Exception e){ e.printStackTrace(); } } }
参考:https://www.php.cn/java-article-410783.html
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· Manus的开源复刻OpenManus初探
· AI 智能体引爆开源社区「GitHub 热点速览」
· 三行代码完成国际化适配,妙~啊~
· .NET Core 中如何实现缓存的预热?