我该如何创建一个Java线程?

在JDK8中,不使用线程池的情况下,一般使用Thread类来创建线程,使用Thread类有三种方式来创建线程。

继承Thread类:通过该方式需要集成hread类,Java是单继承的,这种方式会占用继承名额,不够灵活。

实现Runnable接口:该方法没有返回值,可以通过引用变量的方式返回线程中的计算结果

实现Callable接口:该方法可以返回值,不过需要使用FutureTask对Callable实现类再封装一层,Callable实现类不能直接作为Thread构造方法的参数(代码中通过FutureTask.get()方法获取返回结果,这是一个异步阻塞方法,其方法内部有一个for死循环,直到执行任务的线程修改了FutureTask的state,这个这里不再细述)。

下面是示例代码:

java 复制代码
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask;

// 通过实现Runnable接口来创建线程
class Worker01Thread implements Runnable{

    @Override
    public void run() {
        System.out.println("worker01 start work");
    }

}


// 通过继承Thread类来创建线程,其实和Runnable接口挺像的
class Worker02Thread extends Thread{
    @Override
    public void run() {
        System.out.println("worker02 start work");
    }
}

// 通过实现Callable接口来创建线程,其call()方法是有返回值的
class Worker03Thread implements Callable<String>{
    @Override
    public String call() throws Exception {
        return "worker03 start work";
    }
}

public class CreateThreadDemo01 {

    public static void main(String[] args) throws ExecutionException, InterruptedException {

        Thread t1 = new Thread(new Worker01Thread());
        t1.start();

        Thread t2 = new Thread(new Worker02Thread());
        t2.start();

        // 这里需要将Callable接口的实现类封装成FutureTask,再创建线程
        FutureTask<String> futureTask = new FutureTask<>(new Worker03Thread());
        Thread t3 = new Thread(futureTask);
        t3.start();
        // futureTask.get(),可以获取线程的返回值,如果线程被阻塞住,futureTask.get()方法也会一直阻塞在这里
        System.out.println(futureTask.get());

    }
}

在JDK21中,可以创建虚拟线程,虚拟线程只能通过Runnable接口或者线程池的方式创建,下面是Runnable接口创建虚拟线程的示例:

java 复制代码
class VirtualWorkerThread implements Runnable {
    @Override
    public void run() {
        System.out.println("VirtualWorkerThread start work");
    }
}

public class CreateVirtualThreadDemo01 {
    public static void main(String[] args) throws InterruptedException {
        Thread t1 = Thread.ofVirtual().start(new VirtualWorkerThread());
        t1.join(); // 不加这一步,主线程提前结束,虚拟线程还没输出结果
    }
}
相关推荐
无名指的等待7121 小时前
Redisson的Lock和TryLock的区别
java·开发语言·数据库
new_daimond1 小时前
Apache Shiro 技术详解
java·apache
yuriy.wang1 小时前
Spring IOC源码篇六 核心方法obtainFreshBeanFactory.parseCustomElement
java·后端·spring
.鸣1 小时前
idea学习日记10: 字符串相关类的底层原理
java·学习
在未来等你1 小时前
Kafka面试精讲 Day 24:Spring Kafka开发实战
java·spring boot·面试·kafka·消息队列·spring kafka·@kafkalistener
龙茶清欢2 小时前
1、Lombok入门与环境配置:理解Lombok作用、配置IDE与构建工具
java·spring boot·spring cloud
龙茶清欢2 小时前
2、Nginx 与 Spring Cloud Gateway 详细对比:定位、场景与分工
java·运维·spring boot·nginx·spring cloud·gateway
Eoch772 小时前
HashMap夺命十连问,你能撑到第几轮?
java·后端
云动雨颤2 小时前
程序出错瞎找?教你写“会说话”的错误日志,秒定位原因
java·运维·php
魔芋红茶2 小时前
RuoYi 学习笔记 3:二次开发
java·笔记·学习