当前位置: 首页 > news >正文

Java-线程池

理解池的概念

​     字符串常量池    String s1 = "abc";   String  s2="abc";   s1==s2;

​     数据库连接池    Connection  每次链接创建,每次用完销毁     创建对象是要花费时间的

Integer -128---+127也有缓存池

线程池的概念

线程池是一种多线程处理形式,通过预先创建并管理一组线程,避免频繁创建和销毁线程的开销,提高系统性能和资源利用率。适用于需要处理大量短期任务的场景。

线程池的核心参数

  • corePoolSize:核心线程数,线程池长期维持的线程数量。
  • maximumPoolSize:最大线程数,任务队列满后允许创建的最大线程数。
  • keepAliveTime:非核心线程闲置时的存活时间。
  • workQueue:任务队列,用于保存待执行任务(如 ArrayBlockingQueueLinkedBlockingQueue)。
  • threadFactory:线程工厂,用于定制线程创建(如命名线程)。
  • RejectedExecutionHandler:拒绝策略,处理任务队列满时的后续任务(如直接丢弃或抛出异常)。

线程池的工作流程

  1. 提交任务时,若当前线程数小于 corePoolSize,立即创建新线程执行任务。
  2. 若线程数已达 corePoolSize,任务被放入工作队列。
  3. 若队列已满且线程数未达 maximumPoolSize,创建非核心线程执行任务。
  4. 若队列和线程数均达上限,触发拒绝策略。

线程池中的队列

线程池有以下工作队列:

ArrayBlockingQueue

LinkedBlockingQueue

线程池的拒绝策略

1.AbortPolicy 直接抛出异常

2.CallerRunsPolicy 拒绝后,由提交任务的线程执行此任务(如main线程)

3.DiscardOldestPolicy 丢弃队列中等待时间最长的那一个

4.DiscardPolicy 丢弃最后来的无法执行的任务

向线程池提交任务的两种方法

execute 与 submit 的区别

void execute 适用于不需要关注返回值的场景

submit 方法适用于需要关注返回值的场景。

关闭线程池

shutdownNow 立刻关闭,即使还有未执行完的任务

shutdown 等待所有任务执行完了再关闭

java中创建线程方式:

1.继承Thread类

2.实现Runnable接口

3.实现Callable接口

4.线程池

Java 提供的线程池实现

  • FixedThreadPool:固定线程数,队列无界。
    ExecutorService fixedPool = Executors.newFixedThreadPool(5);
    
    
  • CachedThreadPool:线程数无上限,空闲线程回收。
    ExecutorService cachedPool = Executors.newCachedThreadPool();
    
    
  • SingleThreadExecutor:单线程,确保任务顺序执行。
    ExecutorService singleThreadPool = Executors.newSingleThreadExecutor();
    
    
  • ScheduledThreadPool:支持定时或周期性任务。
    ScheduledExecutorService scheduledPool = Executors.newScheduledThreadPool(3);
    
    

自定义线程池示例

推荐使用 ThreadPoolExecutor 构造函数灵活配置参数:

ThreadPoolExecutor customPool = new ThreadPoolExecutor(2,                      // corePoolSize4,                      // maximumPoolSize60, TimeUnit.SECONDS,   // keepAliveTimenew ArrayBlockingQueue<>(100), // workQueuenew ThreadPoolExecutor.CallerRunsPolicy() // 拒绝策略
);

线程ThreadLocal

作用: 为每个线程提供一个变量副本

使用:

 //创建ThreadLocal对象,为每个线程自动的提供一个变量副本static ThreadLocal<Integer> threadLocal = new ThreadLocal<Integer>(){@Overrideprotected Integer initialValue() {return 1; //初始化变量}};public static void main(String[] args) {/*这种方式在两个线程中对num进行操作,这个num是同一个*///线程1new Thread(new Runnable() {@Overridepublic void run() {threadLocal.set(10);threadLocal.set(threadLocal.get()+5);System.out.println(Thread.currentThread().getName()+":"+threadLocal.get());}}).start();//线程2new Thread(new Runnable() {@Overridepublic void run() {threadLocal.set(20);threadLocal.set(threadLocal.get()+10);System.out.println(Thread.currentThread().getName()+":"+threadLocal.get());}}).start();}

池的关闭

  • shutdown():平滑关闭,不再接受新任务,等待已提交任务完成。
  • shutdownNow():强制关闭,尝试中断所有线程并返回未执行任务列表。

ThreadLocal 内存泄漏问题

内存溢出: 内存不够用了

内存泄漏: 一些对象已经不再使用,但是虚拟机又不能回收的对象( 例如: 数据库连接对象,IO流对象,Socket 提供close)

如果使用ThreadLocal不当,会造成内存泄漏问题。

对象与引用关系

Object obj = new Object(); 强引用

obj= null; 没有引用

软引用: 被SoftReference对象管理的引用, 内存充足时,不回收该对象,一旦内存不足时,

就会回收软引用管理的对象.

 Object o1 = new Object();
SoftReference<Object> softReference = new SoftReference<Object>(o1);

弱引用: 被WeakReference对象管理的引用, 只要进行垃圾回收,就会被回收掉

ThreadLocal被弱引用管理的, 下次垃圾回收到来时,ThreadLocal会被回收掉, 造成ThreadLocalMap中的不存在了,

但是value还被外界引用, 所以ThreadLocalMap就不能被回收, 造成了内存泄漏.

所以, 正确的使用ThreadLocal方式是在用完之后, 主动删除ThreadLocalMap中的数据.

注意事项

  • 避免使用无界队列(如 LinkedBlockingQueue 未指定容量),可能导致内存溢出。
  • 根据任务类型(CPU密集型或IO密集型)合理设置线程数。
  • 监控线程池状态(如通过 getActiveCount())。
http://www.dtcms.com/a/276129.html

相关文章:

  • 【机器学习实战笔记 16】集成学习:LightGBM算法
  • AV1高层语法
  • PostgreSQL HOT (Heap Only Tuple) 更新机制详解
  • Swin Transformer核心思路讲解(个人总结)
  • 文件上传漏洞2-常规厂商检测限制绕过原理讲解
  • 强化学习、PPO和GRPO的通俗讲解
  • C语言第一章数据类型和变量(下)
  • Java 大视界:基于 Java 的大数据可视化在智慧城市能源消耗动态监测与优化决策中的应用(2025 实战全景)
  • 视频分析应用的搭建
  • 【Linux-云原生-笔记】Apache相关
  • NE综合实验2:RIP与OSPF动态路由优化配置、FTP/TELNET服务部署及精细化访问控制
  • Java反射与注解
  • 树形动态规划详解
  • 大数据时代UI前端的智能化服务升级:基于用户情境的主动服务设计
  • 【PycharmPyqt designer桌面程序设计】
  • 【学习新知识】用 Clang 提取函数体 + 构建代码知识库 + AI 问答系统
  • GD32 CAN1和TIMER0同时开启问题
  • 《通信原理》学习笔记——第一章
  • 细谈kotlin中缀表达式
  • H2在springboot的单元测试中的应用
  • skywalking镜像应用springboot的例子
  • try-catch-finally可能输出的答案?
  • Docker-镜像构建原因
  • C语言基础教程--从入门到精通
  • Spring Boot整合MyBatis+MySQL+Redis单表CRUD教程
  • STM32中的RTC(实时时钟)详解
  • R 语言绘制 10 种精美火山图:转录组差异基因可视化
  • JavaScript 常见10种设计模式
  • 码头智能哨兵:AI入侵检测系统如何终结废钢盗窃困局
  • Redis专题总结