Java基础:线程的三种创建方式

一、继承Thread类

  1. 定义一个类继承线程类Thread
  2. 重写run()方法
  3. 创建线程对象
  4. 调用线程对象的start()方法创建线程

Thread类的常用API

  • setName(String name):给线程取名字
  • getName():获取线程的名字
  • public static Thread currentThread():获取当前线程对象,这个代码在哪个线程中就获取哪个线程对象
  • public static void sleep(long time):让当前线程休眠多少毫秒再继续执行

优点:

  • 编码简单

缺点:

  • 线程类已经继承Thread就无法继承其他类了,功能无法通过继承拓展(单继承的局限性)

二、实现Runable接口

  1. 定义一个线程任务类实现Runable接口
  2. 重写run()方法
  3. 创建线程任务对象
  4. 把线程任务对象包装成线程对象
  5. 调用线程对象的start()方法启动线程
java 复制代码
class MyRunable implements Runnable{

    @Override
    public void run() {
        System.out.println("hello runable");
    }
}
java 复制代码
public static void main(String[] args) {
    MyRunable target = new MyRunable();
    Thread t = new Thread(target);
    t.start();
}

优点:

  • 避免了Java单继承的局限性
  • 同一个任务对象可以被包装成多个线程对象
  • 适合多个线程去共享同一个资源
  • 实现解耦,线程任务代码可以被多个线程共享,线程任务代码和线程独立
  • 线程池只能放入实现Runable和Callable接口的线程任务,不能直接放入继承Thread类的线程对象

缺点

  • 编码复杂(其实还好啦)

匿名内部类的写法:

java 复制代码
Runnable target = new Runnable() {
    @Override
    public void run() {
        System.out.println("匿名内部类写法");
    }
};
Thread t = new Thread(target);
t.start();

//简化
new Thread(new Runnable() {
    @Override
    public void run() {
        System.out.println("匿名内部类写法");
    }
}).start();

//再简化
new Thread(() -> System.out.println("匿名内部类写法")).start();

三、实现Callable接口

  1. 定义一个线程任务类实现Callable接口,申明线程执行的结果类型
  2. 重写线程任务类的call方法,这个方法可以直接返回执行结果
  3. 创建一个Callable的线程任务对象
  4. 把线程任务对象包装成一个未来任务对象
  5. 把未来任务对象包装成线程对象
  6. 调用start()方法启动线程
java 复制代码
//1.定义一个线程任务类实现Callable接口,申明线程执行的结果类型
class MyCallable implements Callable<String>{

    //2.重写线程任务类的call方法,这个方法可以直接返回执行结果
    @Override
    public String call() throws Exception {
        System.out.println("hello callable");
        return "success";
    }
}
java 复制代码
//3.创建一个Callable的线程任务对象
Callable<String> call = new MyCallable();
//4.把线程任务对象包装成一个未来任务对象
//-- 未来任务对象其实就是一个Runable对象,这样就可以被包装成线程对象
//-- 未来任务对象可以在线程执行完毕后得到线程执行结果
FutureTask<String> future = new FutureTask<>(call);

//5.把未来任务对象包装成线程对象
Thread t = new Thread(future);
t.start();

//获取线程执行结果,如果线程还没执行完,让出CPU等线程执行完再来去结果
try {
    String res = future.get();
    System.out.println(res);
} catch (InterruptedException e) {
    e.printStackTrace();
} catch (ExecutionException e) {
    e.printStackTrace();
}

优点:

  • 拥有所有Runable的优点
  • 能直接得到线程执行的结果

缺点:

  • 编码复杂
相关推荐
yours_Gabriel4 分钟前
java基础:面向对象(二)
java·开发语言·笔记·学习
Enaium7 分钟前
Rust入门实战 编写Minecraft启动器#3解析资源配置
java·开发语言·rust
虫小宝18 分钟前
在Spring Boot中实现多线程任务调度
java·spring boot·spring
虫小宝1 小时前
如何在Java中实现PDF生成
java·开发语言·pdf
菜鸡且互啄692 小时前
在线教育平台,easyexcel使用案例
java·开发语言
八月林城2 小时前
JAVA导出数据库字典到Excel
java·数据库·excel
浅念同学4 小时前
算法-常见数据结构设计
java·数据结构·算法
杰哥在此6 小时前
Java面试题:讨论持续集成/持续部署的重要性,并描述如何在项目中实施CI/CD流程
java·开发语言·python·面试·编程
咖啡煮码7 小时前
深入剖析Tomcat(十五、十六) 关闭钩子,保证Tomcat的正常关闭
java·tomcat
C.C7 小时前
java IO流(1)
java·开发语言