16_线程池

本文详细介绍了线程的状态转换图、Java中的线程池(Executors工具类与三种常见线程池)、多线程的Callable接口应用、以及线程安全的单例设计模式(包括双重认证)。通过实例演示展示了如何在代码层面实现这些概念。
摘要由CSDN通过智能技术生成

完整的线程状态转换图

理论层面

代码层面

在这里插入图片描述

线程池

提高效率

3种线程池

Executors: 线程工具类, 负责产生线程池

ExecutorServices: 代表线程池对象,负责接受线程池

  • ExecutorService newCachedThreadPool()
    • 特点:
      • 会根据需要创建新线程,也可以自动删除,60s处于空闲状态的线程
      • 线程数量可变立刻执行提交的异步任务(异步任务:在子线程中执行的任务)
  • ExecutorService newFixedThreadPool(int nThreads)
    • 特点:
      • 线程数量固定
      • 维护一个无界队列(暂存已提交的来不及执行的任务)
      • 按照任务的提交顺序,将任务执行完毕
  • ExecutorService newSingleThreadExecutor()
    • 特点:
      • 单个线程
      • 维护了一个无界队列(暂存已提交的来不及执行的任务)
      • 按照任务的提交顺序,将任务执行完毕

线程池的使用

向线程池传入

Future<?> submit(Runnable task)

Future<T> submit(Callable<T> task)

说明
Future 表示异步计算的结果。它提供了检查计算是否完成的方法,以等待计算的完成,并获取计算的结果。

eg:


public class Demo {
    public static void main(String[] args) {
        /*
        线程池的使用:
        Future<?> submit(Runnable task)
        Future<T> submit(Callable<T> task)
         */

        // 1. 创建线程池对象
        // Executors: 线程工具类, 负责产生线程池
        ExecutorService pool = Executors.newCachedThreadPool();

        // 2. 向线程池中提交任务
        pool.submit(new MyRunnableTask());
        pool.submit(new MyRunnableTask());  
        
    }
}

class MyRunnableTask implements Runnable{

    @Override
    public void run() {
        System.out.println(Thread.currentThread().getName() + "hello world");
    }
}

关闭线程池

shutdown()      // 启动一次顺序关闭,执行以前提交的任务,但不接受新任务。

shutdownNow()     // 试图停止所有正在执行的活动任务,暂停处理正在等待的任务,并返回等待执行的任务列表。

eg:


public class Demo {
    public static void main(String[] args) {
        /*
        线程池的使用:
        Future<?> submit(Runnable task)
        Future<T> submit(Callable<T> task)
         */

        // 1. 创建线程池对象
        // Executors: 线程工具类, 负责产生线程池
        ExecutorService pool = Executors.newFixedThreadPool(2);

        // 2. 向线程池中提交任务
        pool.submit(new MyRunnableTask());
        pool.submit(new MyRunnableTask());
        pool.submit(new MyRunnableTask());
        pool.submit(new MyRunnableTask());

        // 3. 关闭线程池
        pool.shutdown();

        pool.shutdownNow();

    }
}

class MyRunnableTask implements Runnable{

    @Override
    public void run() {
        System.out.println(Thread.currentThread().getName() + "hello world");
    }
}

向线程池中提交callable类型的任务

Future 用来存储返回值的结果(Callable是带返回值的)

  • V(表示返回值数据的类型) get() ----> 如有必要,等待计算完成,然后获取其结果。
    • 可以得到这个get()也是一个阻塞的方法

eg:


public class Demo {
    public static void main(String[] args) throws ExecutionException, InterruptedException {
        // 创建线程池
        ExecutorService pool = Executors.newFixedThreadPool(2);

        // submit(Callable task)
        Future<String> future = pool.submit(new MyCallableTask());

        // 拿到返回值结果
        String s = future.get();
        System.out.println(s);
    }
}

class MyCallableTask implements Callable<String>{

    @Override
    public String call() throws Exception {

        for (int i = 0; i < 10; i++) {
            System.out.println(i);
        }
        return "aaaa";
    }
}


多线程的实现方式三:实现Callable接口

不借助线程池,需要借助FutureTask

构造方法


FutureTask(Callable<V> callable)  
//  创建一个 FutureTask,一旦运行就执行给定的 Callable。

eg:


public class Demo {
    public static void main(String[] args) throws ExecutionException, InterruptedException {
        // 1. 创建FutureTask对象
        /*
        FutureTask(Callable<V> callable)
        创建一个 FutureTask,一旦运行就执行给定的 Callable。
         */
        FutureTask<String> futureTask = new FutureTask<>(new MyCallable());

        // 2. 让任务运行在线程中
        Thread t = new Thread(futureTask);
        t.start();

        // 3. 获取任务执行的结果
        String s = futureTask.get();
        System.out.println(s);
    }
}

class MyCallable implements Callable<String>{

    @Override
    public String call() throws Exception {
        System.out.println("bbb");
        return "aaa";
    }
}


单例设计模式(线程安全)

  1. 构造方法私有
  2. 提供一个全局的自身的成员变量
  3. 提供一个静态的方法获取实例

同步方法


public class Singleton {

    //2. 提供一个全局的自身的成员变量
    private static Singleton instance;
    // 1. 构造方法私有

    private Singleton() {
    }

    //3. 提供一个静态的方法获取实例
    public  static synchronized Singleton getInstance() {
        if (instance == null) {
            instance = new Singleton();
        }
        return instance;
    }
}

双重认证

public class Singleton {

    //2. 提供一个全局的自身的成员变量
    private static Singleton instance;
    // 1. 构造方法私有

    private Singleton() {
    }

    //3. 提供一个静态的方法获取实例
    public  static Singleton getInstance() {

        // double check
        if (instance == null) {
            // 假设A线程抢到了CPU执行权 A执行
            // A持有锁对象
            // 切换B线程 B执行
            // B没有锁 进入不了sync
            synchronized (Singleton.class) {
                // A线程进来
                // B进来了
                // 第二次校验
                if (instance == null) {
                    instance = new Singleton();

                }
            }
        }
        return instance;
    }
}


  • 12
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

coo1heisenberg

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值