线程池笔记

简介: 线程池笔记

线程的状态

 

小结:

虚拟机中线程的六种状态:

新建状态(NEW) → 创建线程对象

就绪状态(RUNNABLE)→ start方法

阻塞状态(BLOCKED)→无法获得锁对象

等待状态(WAITING) → wait方法

计时等待(TIMED_WAITING) → sleep方法

结束状态(TERMINATED)→全部代码允许完毕

线程池-基本原理

之前写代码的弊端:

1.用到线程的时候就创建

2.用完之后线程消失

1.创建一个池子

2.当有任务需要执行时,才会创建线程对象,当任务执行完毕,线程对象归还给池子

线程池-Executors默认线程池

1.创建一个池子,池子中是空的 →创建Executors中的静态方法

2.有任务需要执行时,创建线程对象。任务执行完毕,线程uix归还给池子。 → submit方法 池子会自动地帮我们创建对象,任务执行完毕,也会自动把线程对象归还池子

3.所有的任务全部执行完毕,关闭连接池 → shutdown方法

package test14;
 
import java.util.concurrent.Executor;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
 
public class MyThreadPoolDemo {
    public static void main(String[] args) throws InterruptedException {
        ExecutorService executorService = Executors.newCachedThreadPool();
        executorService.submit(()->{
            System.out.println(Thread.currentThread().getName() + "在执行了");
        });
 
        Thread.sleep(2000);
 
        executorService.submit(()->{
            System.out.println(Thread.currentThread().getName() + "在执行了");
        });
 
        executorService.shutdown();
    }
}

线程池-Executors创建指定上限的线程池

可以利用断点查看程序运行时的各个参数

线程池里面有2个

package test14;
 
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.ThreadPoolExecutor;
 
public class MyThreadPoolDemo2 {
    public static void main(String[] args) {
        //参数nThreads不是初始值而是最大值
        ExecutorService executorService = Executors.newFixedThreadPool(10);
        ThreadPoolExecutor pool = (ThreadPoolExecutor) executorService;
        System.out.println(pool.getPoolSize());//0
                executorService.submit(()->{
            System.out.println(Thread.currentThread().getName() + "在执行了");
        });
 
        executorService.submit(()->{
            System.out.println(Thread.currentThread().getName() + "在执行了");
        });
        System.out.println(pool.getPoolSize());//2  打印当前线程数
 
        //executorService.shutdown(); //关闭线程池
    }
}

 

ThreadPooIExecutor

故事中的核心元素

核心元素一:正式员工数量 → 核心线程数量

核心元素二:餐厅最大员工数 → 线程池中最大线程的数量

核心元素三:临时员工空闲多长时间被辞退(值)→ 空闲时间(值)

核心元素四:临时员工空闲多长时间被辞退(单位)→空闲时间(单位)

核心元素五:排队的客户 → 阻塞队列

核心元素六:从哪里招人 → 创建线程的方式

核心元素七:当排队人数过多,超出顾客请下次再来(拒绝服务)→ 要执行的任务过多时的解决方案

package test14;
 
import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.Executors;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
 
public class MyThreadPoolDemo3 {
    public static void main(String[] args) {
/*      参数一:核心线程数量
        参数二:最大线程数
        参数三:空闲线程最大存活时间
        参数四:时间单位
        参数五:任务队列
        参数六:创建线程工厂
        参数七:任务的拒绝策略    */
        ThreadPoolExecutor pool = new ThreadPoolExecutor(2,5,2, TimeUnit.SECONDS,new ArrayBlockingQueue<>(10), Executors.defaultThreadFactory(),new ThreadPoolExecutor.AbortPolicy());
        pool.submit(new MyRunnable());
        pool.submit(new MyRunnable());
 
        pool.shutdown();
    }
}
 
//*************************分隔符
 
package test14;
 
public class MyRunnable implements Runnable{
    @Override
    public void run() {
        System.out.println(Thread.currentThread().getName() + "在执行了");
    }
}

创建线程池对象

参数一: 核心线程数量 → 不能小于0

参数二:最大线程数 → 不能小于等于0,最大数量>=核心线程数量

参数三:空闲线程最大存活时间 → 不能小于0

参数四:时间单位 → 时间单位

参数五:任务队列→不能为null

参数六:创建线程工厂 → 不能为null

参数七: → 不能为null

参数详解

package test14;
 
import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.Executors;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
 
public class MyThreadPoolDemo4 {
    public static void main(String[] args) {
/*      参数一:核心线程数量
        参数二:最大线程数
        参数三:空闲线程最大存活时间
        参数四:时间单位
        参数五:任务队列 --- 让任务再队列中等着,等有线程空闲了,再从这个队列中获取任务并执行
        参数六:创建线程工厂 --- 按照默认的方式创建线程对象
        参数七:任务的拒绝策略 --- 什么时候拒绝任务?  当提交的任务 > 池子中最大线程数量
        如何拒绝?   */
        ThreadPoolExecutor pool = new ThreadPoolExecutor(2,5,2, TimeUnit.SECONDS,new ArrayBlockingQueue<>(10), Executors.defaultThreadFactory(),new ThreadPoolExecutor.AbortPolicy());
        for (int i = 0 ; i < 15 ; i ++) {
            pool.submit(new MyRunnable());
        }
 
 
        pool.shutdown();
    }
}

非默认任务拒绝策略

ThreadPoolExecutor.AbortPolicy 丢弃任务并抛出RejectedExecutionException异常。是默认的策略

ThreadPoolExecutor.DiscardPolicy:丢弃任务,但是不抛出异常,这是不推荐的做法。

ThreadPoolExecutor.DiscardOldestPolicy:抛弃队列中等待最久的任务,然后把当前任务加入队列中

ThreadPoolExecutor.CallerRunsPolicy:调用任务的run()方法绕过线程池直接执行

package test14;
 
import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.Executors;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
 
public class MyThreadPoolDemo5 {
    public static void main(String[] args) {
/*      参数一:核心线程数量
        参数二:最大线程数
        参数三:空闲线程最大存活时间
        参数四:时间单位
        参数五:任务队列 --- 让任务再队列中等着,等有线程空闲了,再从这个队列中获取任务并执行
        参数六:创建线程工厂 --- 按照默认的方式创建线程对象
        参数七:任务的拒绝策略 --- 什么时候拒绝任务?  当提交的任务 > 池子中最大线程数量
        如何拒绝?   */
        ThreadPoolExecutor pool = new ThreadPoolExecutor(
                1,
                2,
                2,
                TimeUnit.SECONDS,
                new ArrayBlockingQueue<>(1),
                Executors.defaultThreadFactory(),
                new ThreadPoolExecutor.DiscardPolicy());
        for (int i = 1 ; i < 5 ; i ++) {
            int y = i;
            pool.submit(()->{
 
                System.out.println(Thread.currentThread().
                        getName() + "---" + y);
            });
        }
 
 
        pool.shutdown();
    }
}
package test14;
 
import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.Executors;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;
 
public class MyThreadPoolDemo6 {
    public static void main(String[] args) {
/*      参数一:核心线程数量
        参数二:最大线程数
        参数三:空闲线程最大存活时间
        参数四:时间单位
        参数五:任务队列 --- 让任务再队列中等着,等有线程空闲了,再从这个队列中获取任务并执行
        参数六:创建线程工厂 --- 按照默认的方式创建线程对象
        参数七:任务的拒绝策略 --- 什么时候拒绝任务?  当提交的任务 > 池子中最大线程数量
        如何拒绝?   */
        ThreadPoolExecutor pool = new ThreadPoolExecutor(
                1,
                2,
                2,
                TimeUnit.SECONDS,
                new ArrayBlockingQueue<>(1),
                Executors.defaultThreadFactory(),
                new ThreadPoolExecutor.DiscardOldestPolicy());
        for (int i = 1 ; i < 10 ; i ++) {
            int y = i;
            pool.submit(()->{
 
                System.out.println(Thread.currentThread().
                        getName() + "---" + y);
            });
        }
 
 
        pool.shutdown();
    }
}


目录
相关文章
|
机器学习/深度学习 缓存 Java
Python 线程,进程,多线程,多进程以及并行执行for循环笔记
Python 线程,进程,多线程,多进程以及并行执行for循环笔记
685 0
Python 线程,进程,多线程,多进程以及并行执行for循环笔记
|
3月前
|
设计模式 缓存 Java
谷粒商城笔记+踩坑(14)——异步和线程池
初始化线程的4种方式、线程池详解、异步编排 CompletableFuture
谷粒商城笔记+踩坑(14)——异步和线程池
|
7月前
|
Java 测试技术 开发工具
Android 笔记:AndroidTrain , Lint , build(1),只需一篇文章吃透Android多线程技术
Android 笔记:AndroidTrain , Lint , build(1),只需一篇文章吃透Android多线程技术
|
7月前
|
存储 缓存 调度
FFmpeg开发笔记(十九)FFmpeg开启两个线程分别解码音视频
《FFmpeg开发实战》第10章示例playsync.c在处理音频流和视频流交错的文件时能实现同步播放,但对于分开存储的格式,会出现先播放全部声音再快速播放视频的问题。为解决此问题,需改造程序,增加音频处理线程和队列,以及相关锁,先将音视频帧读入缓存,再按时间戳播放。改造包括声明新变量、初始化线程和锁、修改数据包处理方式等。代码修改后在playsync2.c中,编译运行成功,控制台显示日志,SDL窗口播放视频并同步音频,证明改造有效。
116 0
FFmpeg开发笔记(十九)FFmpeg开启两个线程分别解码音视频
|
Java 程序员 Windows
[笔记]Windows核心编程《十一》Windows线程池
[笔记]Windows核心编程《十一》Windows线程池
36694 4
[笔记]Windows核心编程《十一》Windows线程池
|
7月前
|
NoSQL Java 应用服务中间件
线程池笔记
线程池笔记
45 0
|
12月前
|
Java 容器
膜拜!清华大佬手撸多线程并发源码笔记Github上线3天星标35k+
你为什么要学习多线程?是因为理想吗?是因为热爱吗? 哦~原来是为了面试打基础、做准备啊!没错,这真的很现实!
膜拜!清华大佬手撸多线程并发源码笔记Github上线3天星标35k+
|
缓存 编译器 调度
[笔记]Windows核心编程《七》用户模式下的线程同步
[笔记]Windows核心编程《七》用户模式下的线程同步
|
缓存 安全 网络协议
[笔记]C++并发编程实战 《三》线程间共享数据(四)
[笔记]C++并发编程实战 《三》线程间共享数据(四)
|
存储 安全 算法
[笔记]C++并发编程实战 《三》线程间共享数据(三)
[笔记]C++并发编程实战 《三》线程间共享数据(三)
下一篇
无影云桌面