首页
壁纸
统计
友链
Search
1
Vue2详细笔记
37 阅读
2
Nestjs概述-中文
19 阅读
3
ExpressAPI
17 阅读
4
答辩
14 阅读
5
JavaScript企业数据处理实用指南
13 阅读
Nodejs
Vue
Java
Msql
登录
Search
Wasnl
累计撰写
36
篇文章
累计收到
1
条评论
首页
栏目
Nodejs
Vue
Java
Msql
页面
壁纸
统计
友链
搜索到
16
篇与
的结果
2026-04-09
Java 多线程与线程池完整笔记
Java 多线程与线程池完整笔记一、线程池基础1.1 什么是线程池线程池是一种多线程处理模式,它在系统启动时预先创建一定数量的线程放入池中。这些线程在没有任务时处于等待状态,当有任务提交时,池中某个空闲线程会被唤醒执行任务。任务执行完成后,线程并不会被销毁,而是返回池中继续等待下一个任务。┌─────────────────────────────────────────────────┐ │ 线程池架构 │ ├─────────────────────────────────────────────────┤ │ ┌─────┐ ┌─────┐ ┌─────┐ ┌─────┐ │ │ │线程1│ │线程2│ │线程3│ │线程N│ ← 常驻线程 │ │ └──┬──┘ └──┬──┘ └──┬──┘ └──┬──┘ │ │ │ │ │ │ │ │ └─────────┴────┬────┴─────────┘ │ │ ▼ │ │ ┌───────────────┐ │ │ │ 任务队列 │ ← 缓冲待处理任务 │ │ └───────────────┘ │ └─────────────────────────────────────────────────┘1.2 为什么需要线程池优势说明资源复用避免频繁创建和销毁线程的开销,线程对象可重复使用响应速度任务到达时无需等待线程创建,立即执行可管理性统一分配、调优和监控线程,防止资源耗尽流量控制通过队列和最大线程数限制并发量,保护系统二、ThreadPoolExecutor 核心参数详解2.1 七个核心参数public ThreadPoolExecutor( int corePoolSize, // 1. 核心线程数 int maximumPoolSize, // 2. 最大线程数 long keepAliveTime, // 3. 空闲线程存活时间 TimeUnit unit, // 4. 时间单位 BlockingQueue<Runnable> workQueue, // 5. 工作队列 ThreadFactory threadFactory, // 6. 线程工厂 RejectedExecutionHandler handler // 7. 拒绝策略 )参数含义比喻corePoolSize核心线程数,即使空闲也保留在池中的线程数量餐厅正式员工,永不辞退maximumPoolSize最大线程数 = 核心线程 + 临时线程正式员工 + 临时工的上限keepAliveTime临时线程空闲超过该时间则被回收临时工多久没活干就辞退unitkeepAliveTime 的时间单位秒、毫秒等workQueue存放待执行任务的阻塞队列餐厅等位区threadFactory创建线程的工厂,可自定义线程名、优先级等员工招聘标准handler队列满且线程达最大值时的拒绝策略客满时的处理方式2.2 常用工作队列类型队列类型特点适用场景SynchronousQueue不存储任务,直接交给线程执行CachedThreadPoolLinkedBlockingQueue无界队列(默认 Integer.MAX_VALUE)FixedThreadPoolArrayBlockingQueue有界队列,需指定容量需限制队列长度的场景PriorityBlockingQueue优先级队列任务有优先级要求三、线程池执行流程(五步核心逻辑)// 伪代码表示执行逻辑 public void execute(Runnable task) { // 步骤1:当前线程数 < corePoolSize if (workerCount < corePoolSize) { addWorker(task, true); // 新建核心线程执行 return; } // 步骤2:核心线程已满,尝试放入队列 if (workQueue.offer(task)) { // 放入成功,等待空闲线程取走执行 return; } // 步骤3:队列已满,尝试创建临时线程 if (workerCount < maximumPoolSize) { addWorker(task, false); // 新建临时线程执行 return; } // 步骤4:线程数已达最大值,执行拒绝策略 reject(task); }流程图:提交任务 │ ▼ 当前线程数 < corePoolSize? ──Yes──▶ 创建核心线程执行 │ No │ ▼ 任务能否入队? ──Yes──▶ 入队等待执行 │ No(队列已满) │ ▼ 当前线程数 < maximumPoolSize? ──Yes──▶ 创建临时线程执行 │ No │ ▼ 执行拒绝策略四、四种拒绝策略策略类行为比喻使用建议AbortPolicy抛出 RejectedExecutionException直接赶走客人并吵架默认策略,开发常用,快速失败CallerRunsPolicy由提交任务的线程自己执行老板亲自服务客人可减缓提交速度,重要任务兜底DiscardPolicy静默丢弃新任务悄悄赶走客人不重要的任务,如日志上报DiscardOldestPolicy丢弃队列中最老的任务赶走排队最久的人优先处理新任务// 自定义拒绝策略示例 new ThreadPoolExecutor.AbortPolicy() // 抛异常 new ThreadPoolExecutor.CallerRunsPolicy() // 调用者执行 new ThreadPoolExecutor.DiscardPolicy() // 直接丢弃 new ThreadPoolExecutor.DiscardOldestPolicy() // 丢弃最旧的五、Executors 四种快捷线程池⚠️ 开发规范:强制禁止使用 Executors 创建线程池,必须通过 ThreadPoolExecutor 构造函数显式指定参数。原因:FixedThreadPool 和 SingleThreadPool 使用无界队列,可能 OOMCachedThreadPool 和 ScheduledThreadPool 允许创建无限数量线程,可能 OOM工厂方法核心特点适用场景风险newFixedThreadPool(n)固定线程数,无界队列重负载服务器队列无限堆积newCachedThreadPool()0核心,无限临时线程,同步队列大量短期异步任务线程数爆炸newSingleThreadExecutor()单线程,无界队列任务需顺序执行队列无限堆积newScheduledThreadPool(n)支持定时/周期任务定时调度线程数爆炸// ❌ 错误示范 ExecutorService pool = Executors.newFixedThreadPool(10); // ✅ 正确示范 ExecutorService pool = new ThreadPoolExecutor( 10, 20, 60L, TimeUnit.SECONDS, new ArrayBlockingQueue<>(1000), new ThreadFactoryBuilder().setNameFormat("my-pool-%d").build(), new ThreadPoolExecutor.CallerRunsPolicy() );六、线程数设置公式6.1 CPU 密集型任务定义:大量时间用于计算,很少等待 I/O线程数 = CPU 核心数 + 1加 1 是为了在某个线程因缺页中断等原因暂停时,有备用线程顶上,最大化 CPU 利用率。6.2 I/O 密集型任务定义:大量时间等待 I/O 操作(网络、磁盘、数据库)线程数 = CPU 核心数 × 2更精确的公式:线程数 = CPU 核心数 × (1 + 平均等待时间 / 平均计算时间)6.3 混合型任务将任务拆分为 CPU 密集和 I/O 密集两部分,分别用不同线程池处理。七、线程池生命周期与优雅关闭7.1 五种状态状态说明接收新任务执行队列任务RUNNING正常运行✅✅SHUTDOWN关闭中(调用 shutdown())❌✅STOP强制停止(调用 shutdownNow())❌❌TIDYING过渡状态,所有任务终止❌❌TERMINATED完全终止❌❌7.2 优雅关闭代码模板public void shutdownGracefully(ExecutorService pool) { pool.shutdown(); // ① 停止接收新任务 try { // ② 等待现有任务完成,最多等待 60 秒 if (!pool.awaitTermination(60, TimeUnit.SECONDS)) { pool.shutdownNow(); // ③ 超时则强制关闭 // ④ 再等待 5 秒,确认是否终止 if (!pool.awaitTermination(5, TimeUnit.SECONDS)) { System.err.println("线程池未能正常终止"); } } } catch (InterruptedException e) { pool.shutdownNow(); // ⑤ 当前线程被中断,也强制关闭 Thread.currentThread().interrupt(); } }八、ScheduledThreadPoolExecutor 定时任务线程池8.1 概述ScheduledThreadPoolExecutor 继承自 ThreadPoolExecutor,专门用于执行定时和周期性任务,是传统 Timer 的替代品。对比项TimerScheduledThreadPoolExecutor线程模型单线程多线程池异常处理任务抛异常则 Timer 线程终止单任务异常不影响其他任务任务调度基于绝对时间基于相对时间,更精准适用场景简单定时生产级定时调度8.2 三种核心 APIScheduledExecutorService scheduler = Executors.newScheduledThreadPool(5); // 1. 延迟执行一次 ScheduledFuture<?> future = scheduler.schedule( () -> System.out.println("延迟执行"), 5, TimeUnit.SECONDS ); // 2. 固定频率执行(不管上次是否完成) scheduler.scheduleAtFixedRate( () -> System.out.println("固定频率"), 1, // 首次延迟 1 秒 3, // 每 3 秒执行一次 TimeUnit.SECONDS ); // 3. 固定延迟执行(等上次完成后再延迟) scheduler.scheduleWithFixedDelay( () -> System.out.println("固定延迟"), 1, // 首次延迟 1 秒 3, // 上次完成后等待 3 秒 TimeUnit.SECONDS );8.3 scheduleAtFixedRate vs scheduleWithFixedDelay方法调度方式效果scheduleAtFixedRate固定频率(周期固定)不管任务执行多久,严格按周期触发,可能任务堆叠scheduleWithFixedDelay固定延迟(间隔固定)等上次执行完,再等固定时间才触发,周期 = 执行时间 + 延迟scheduleAtFixedRate (周期 = 3秒) 任务执行2秒 → 等待1秒 → 下一轮 任务执行4秒 → 立即触发下一轮(无等待) scheduleWithFixedDelay (延迟 = 3秒) 任务执行2秒 → 等待3秒 → 下一轮(间隔5秒) 任务执行4秒 → 等待3秒 → 下一轮(间隔7秒)九、线程池性能监控与调优9.1 关键监控指标指标含义告警阈值排查方向活跃线程数当前正在工作的线程数> 最大线程数 80%线程数不足或任务耗时过长队列长度等待执行的任务数> 队列容量 70%消费速度跟不上生产速度任务完成数已完成任务累计增长停滞线程池可能卡死或阻塞拒绝任务数被拒绝的任务累计> 0(正常应为 0)线程数和队列容量需调大线程池大小当前池中线程总数长期接近 maximumPoolSize调大 maximumPoolSize任务等待时间任务从提交到执行的时间> 业务可接受阈值处理能力不足9.2 监控实现方案@Component public class ThreadPoolMonitor { private final ThreadPoolExecutor executor; @Scheduled(fixedDelay = 30000) // 每30秒采集一次 public void monitor() { int activeCount = executor.getActiveCount(); long completedCount = executor.getCompletedTaskCount(); int queueSize = executor.getQueue().size(); int poolSize = executor.getPoolSize(); long rejectCount = // 自定义记录 log.info("线程池监控 - 活跃:{}, 队列:{}, 池大小:{}, 已完成:{}, 拒绝:{}", activeCount, queueSize, poolSize, completedCount, rejectCount); // 告警判断 if (queueSize > queueCapacity * 0.7) { alert("队列使用率超过70%"); } } }9.3 调优策略现象可能原因解决方案队列持续增长消费速度 < 生产速度增加线程数,或优化任务执行效率活跃线程数持续满任务耗时过长拆分任务、异步化、增加最大线程数拒绝次数频繁线程池容量不足增大队列容量和最大线程数CPU 使用率低但队列满线程阻塞于 I/O增加线程数(I/O 密集型公式)OOM 内存溢出无界队列堆积改用有界队列 + 合理拒绝策略十、最佳实践总结10.1 编码规范// ✅ 正确:显式创建 ThreadPoolExecutor ThreadPoolExecutor executor = new ThreadPoolExecutor( 10, // 核心线程数 20, // 最大线程数 60L, TimeUnit.SECONDS, // 空闲超时 new ArrayBlockingQueue<>(1000), // 有界队列 new NamedThreadFactory("biz-pool"), // 自定义线程工厂 new ThreadPoolExecutor.CallerRunsPolicy() // 拒绝策略 ); // ❌ 错误:使用 Executors 快捷方法 ExecutorService pool = Executors.newFixedThreadPool(10);10.2 必须设置的内容[ ] 有界队列(避免 OOM)[ ] 合适的拒绝策略(根据业务选型)[ ] 有意义的线程名称(便于排查问题)[ ] 监控指标采集(JMX / Micrometer)[ ] 优雅关闭逻辑(应用停止时调用)10.3 参数选择参考业务类型corePoolSizemaximumPoolSize队列类型队列容量计算密集型 APICPU 核数 + 1CPU 核数 + 1ArrayBlockingQueue100~500I/O 密集型 APICPU 核数 × 2CPU 核数 × 4ArrayBlockingQueue500~2000后台批量处理1~51~10LinkedBlockingQueue10000+即时响应任务10~5050~200SynchronousQueue0本笔记涵盖了 Java 多线程线程池的核心知识点,建议结合实战加深理解。
2026年04月09日
4 阅读
0 评论
1 点赞
2026-04-09
Java多线程与线程池
# Java多线程与线程池目录线程池基础再深入ThreadPoolExecutor 源码级剖析阻塞队列深度对比与选型拒绝策略详解与自定义实战线程工厂定制与线程命名规范线程池动态调整与运行时管理任务提交与结果处理进阶线程池钩子方法与扩展点ScheduledThreadPoolExecutor 深度解析线程池监控与告警体系搭建常见陷阱与最佳实践生产级线程池配置模板一、线程池基础再深入1.1 线程池的设计思想 —— 池化技术池化技术的核心是空间换时间,预先创建资源对象,使用时直接获取,用完后归还,避免频繁创建销毁的开销。除了线程池,常见的池化技术还有:数据库连接池(HikariCP、Druid)对象池(Apache Commons Pool)内存池(Netty 的 ByteBuf)1.2 线程池解决了什么问题?(细化版)问题线程池解决方案线程创建开销大线程对象复用,避免反复 new Thread().start() 的系统调用线程数量失控通过 maximumPoolSize 和队列容量双重限制并发上限任务堆积导致 OOM使用有界队列 + 拒绝策略,防止无限制接受任务线程生命周期管理复杂统一提供 shutdown() / shutdownNow() 接口任务执行情况难以追踪提供 beforeExecute / afterExecute 钩子和统计指标1.3 线程池的替代方案比较方案优点缺点new Thread(runnable).start()简单直接无复用,资源浪费,难以管理Timer支持定时单线程,异常即停,不推荐ThreadPoolExecutor生产级标准参数较多,需理解原理ForkJoinPool工作窃取,适合递归分治任务不适合普通任务CompletableFuture + 自定义线程池异步编程更优雅学习曲线稍高二、ThreadPoolExecutor 源码级剖析2.1 核心数据结构 —— ctl 原子变量ThreadPoolExecutor 使用一个 AtomicInteger 类型的 ctl 变量,同时存储两个信息:高 3 位:线程池运行状态(RUNNING、SHUTDOWN、STOP、TIDYING、TERMINATED)低 29 位:当前工作线程数量(workerCount)private final AtomicInteger ctl = new AtomicInteger(ctlOf(RUNNING, 0)); private static final int COUNT_BITS = Integer.SIZE - 3; // 29 private static final int CAPACITY = (1 << COUNT_BITS) - 1; // 约 5.3 亿 // 状态常量(高3位) private static final int RUNNING = -1 << COUNT_BITS; // 111... private static final int SHUTDOWN = 0 << COUNT_BITS; // 000... private static final int STOP = 1 << COUNT_BITS; // 001... private static final int TIDYING = 2 << COUNT_BITS; // 010... private static final int TERMINATED = 3 << COUNT_BITS; // 011... // 拆包方法 private static int runStateOf(int c) { return c & ~CAPACITY; } private static int workerCountOf(int c) { return c & CAPACITY; } private static int ctlOf(int rs, int wc) { return rs | wc; }状态转换图: RUNNING │ shutdown() ▼ SHUTDOWN ──队列为空──▶ TIDYING ── terminated()──▶ TERMINATED │ │ shutdownNow() ▼ STOP ──────────────▶ TIDYING ── terminated()──▶ TERMINATED2.2 内部类 Worker —— 工作线程的封装private final class Worker extends AbstractQueuedSynchronizer implements Runnable { final Thread thread; // 实际执行任务的线程 Runnable firstTask; // 构造时可能携带的首个任务 volatile long completedTasks; // 完成任务计数 Worker(Runnable firstTask) { setState(-1); // 初始禁止中断,直到线程启动 this.firstTask = firstTask; this.thread = getThreadFactory().newThread(this); } public void run() { runWorker(this); // 核心循环 } }Worker 继承 AQS 实现了简单的非重入互斥锁,目的是:在执行任务期间,防止其他线程中断正在运行任务的 Worker(interruptIdleWorkers 只中断空闲的)锁状态表示 Worker 是否正在执行任务(state >= 1 表示忙碌)2.3 runWorker —— 线程池运行核心循环final void runWorker(Worker w) { Thread wt = Thread.currentThread(); Runnable task = w.firstTask; w.firstTask = null; w.unlock(); // 允许中断 boolean completedAbruptly = true; try { while (task != null || (task = getTask()) != null) { w.lock(); // 1. 检查线程池状态,必要时中断自己 if ((runStateAtLeast(ctl.get(), STOP) || (Thread.interrupted() && runStateAtLeast(ctl.get(), STOP))) && !wt.isInterrupted()) wt.interrupt(); try { beforeExecute(wt, task); // 钩子:任务执行前 Throwable thrown = null; try { task.run(); // 执行任务 } catch (RuntimeException x) { thrown = x; throw x; } catch (Error x) { thrown = x; throw x; } catch (Throwable x) { thrown = x; throw new Error(x); } finally { afterExecute(task, thrown); // 钩子:任务执行后 } } finally { task = null; w.completedTasks++; w.unlock(); } } completedAbruptly = false; } finally { processWorkerExit(w, completedAbruptly); } }2.4 getTask —— 从队列获取任务(包含线程销毁逻辑)private Runnable getTask() { boolean timedOut = false; for (;;) { int c = ctl.get(); int rs = runStateOf(c); // 检查是否需要减少 Worker if (rs >= SHUTDOWN && (rs >= STOP || workQueue.isEmpty())) { decrementWorkerCount(); return null; } int wc = workerCountOf(c); // 是否允许核心线程超时 或 当前线程数超过核心数 boolean timed = allowCoreThreadTimeOut || wc > corePoolSize; if ((wc > maximumPoolSize || (timed && timedOut)) && (wc > 1 || workQueue.isEmpty())) { if (compareAndDecrementWorkerCount(c)) return null; continue; } try { // 根据 timed 决定是 poll(带超时)还是 take(阻塞) Runnable r = timed ? workQueue.poll(keepAliveTime, TimeUnit.NANOSECONDS) : workQueue.take(); if (r != null) return r; timedOut = true; } catch (InterruptedException retry) { timedOut = false; } } }关键点:非核心线程通过 poll 超时返回 null 后会被回收核心线程默认 take 无限阻塞,除非设置 allowCoreThreadTimeOut(true)三、阻塞队列深度对比与选型3.1 常用阻塞队列底层结构与特性队列底层结构锁机制容量适用场景ArrayBlockingQueue数组单锁(ReentrantLock)必须指定固定容量、内存占用可控LinkedBlockingQueue单向链表两把锁(putLock/takeLock)默认 Integer.MAX_VALUE吞吐量高、无需预知容量SynchronousQueue无存储公平模式队列/非公平栈0直接交付,CachedThreadPoolPriorityBlockingQueue二叉堆数组单锁无界任务有优先级排序DelayQueue优先队列单锁无界定时任务存储3.2 ArrayBlockingQueue vs LinkedBlockingQueue 性能差异// 压测对比结论(仅供参考) // ArrayBlockingQueue:入队出队共用一把锁,并发竞争激烈时性能下降 // LinkedBlockingQueue:入队和出队各用一把锁,吞吐量更高,但内存占用更大 // 选择建议: // - 内存敏感、容量可预估 → ArrayBlockingQueue // - 追求高吞吐、内存较充裕 → LinkedBlockingQueue(需指定容量)3.3 SynchronousQueue 的两种模式// 公平模式:TransferQueue(FIFO) new SynchronousQueue<>(true); // 非公平模式:TransferStack(LIFO)——默认 new SynchronousQueue<>();SynchronousQueue 内部没有容量,每个插入操作必须等待另一个线程的移除操作。CachedThreadPool 使用它,使得任务立即交给线程执行,不积压。四、拒绝策略详解与自定义实战4.1 JDK 内置四种策略源码分析// AbortPolicy public void rejectedExecution(Runnable r, ThreadPoolExecutor e) { throw new RejectedExecutionException("Task " + r.toString() + " rejected from " + e.toString()); } // CallerRunsPolicy public void rejectedExecution(Runnable r, ThreadPoolExecutor e) { if (!e.isShutdown()) { r.run(); // 直接在调用者线程执行 } } // DiscardPolicy public void rejectedExecution(Runnable r, ThreadPoolExecutor e) { // do nothing } // DiscardOldestPolicy public void rejectedExecution(Runnable r, ThreadPoolExecutor e) { if (!e.isShutdown()) { e.getQueue().poll(); // 丢弃队头 e.execute(r); // 重新尝试提交 } }4.2 自定义拒绝策略实战(带日志与监控)public class MonitoredCallerRunsPolicy implements RejectedExecutionHandler { private static final Logger log = LoggerFactory.getLogger(MonitoredCallerRunsPolicy.class); private final MeterRegistry meterRegistry; private final Counter rejectCounter; public MonitoredCallerRunsPolicy(MeterRegistry meterRegistry) { this.meterRegistry = meterRegistry; this.rejectCounter = Counter.builder("thread.pool.reject") .description("线程池拒绝任务计数") .register(meterRegistry); } @Override public void rejectedExecution(Runnable r, ThreadPoolExecutor e) { rejectCounter.increment(); log.warn("线程池 {} 任务被拒绝,当前线程数: {}/{}, 队列大小: {}/{},降级为调用者执行", e.toString(), e.getPoolSize(), e.getMaximumPoolSize(), e.getQueue().size(), e.getQueue().remainingCapacity() + e.getQueue().size()); if (!e.isShutdown()) { // 在调用者线程中执行,并记录耗时(可选) long start = System.nanoTime(); try { r.run(); } finally { log.debug("降级任务执行耗时: {}ms", TimeUnit.NANOSECONDS.toMillis(System.nanoTime() - start)); } } } }4.3 其他常见自定义策略// 1. 阻塞等待策略(当队列满时,让提交线程阻塞一小段时间再尝试) public class BlockingWaitPolicy implements RejectedExecutionHandler { private final long maxWaitMs; public BlockingWaitPolicy(long maxWaitMs) { this.maxWaitMs = maxWaitMs; } @Override public void rejectedExecution(Runnable r, ThreadPoolExecutor e) { if (!e.isShutdown()) { try { if (!e.getQueue().offer(r, maxWaitMs, TimeUnit.MILLISECONDS)) { throw new RejectedExecutionException("等待超时后仍无法提交"); } } catch (InterruptedException ex) { Thread.currentThread().interrupt(); throw new RejectedExecutionException("等待期间被中断", ex); } } } } // 2. 持久化到数据库/消息队列的策略 public class PersistencePolicy implements RejectedExecutionHandler { @Override public void rejectedExecution(Runnable r, ThreadPoolExecutor e) { // 将任务序列化存储到 DB 或 MQ,后续补偿 TaskStorage.save((Serializable) r); log.info("任务被拒绝,已持久化待补偿"); } }五、线程工厂定制与线程命名规范5.1 自定义 ThreadFactory 模板public class NamedThreadFactory implements ThreadFactory { private final AtomicInteger threadNumber = new AtomicInteger(1); private final String namePrefix; private final boolean daemon; private final int priority; private final Thread.UncaughtExceptionHandler exceptionHandler; public NamedThreadFactory(String poolName) { this(poolName, false, Thread.NORM_PRIORITY); } public NamedThreadFactory(String poolName, boolean daemon, int priority) { this.namePrefix = poolName + "-thread-"; this.daemon = daemon; this.priority = priority; this.exceptionHandler = (t, e) -> log.error("线程 {} 发生未捕获异常", t.getName(), e); } @Override public Thread newThread(Runnable r) { Thread t = new Thread(r, namePrefix + threadNumber.getAndIncrement()); t.setDaemon(daemon); t.setPriority(priority); t.setUncaughtExceptionHandler(exceptionHandler); return t; } }5.2 使用示例ThreadPoolExecutor executor = new ThreadPoolExecutor( 10, 20, 60, TimeUnit.SECONDS, new ArrayBlockingQueue<>(1000), new NamedThreadFactory("order-processor", false, Thread.NORM_PRIORITY), new ThreadPoolExecutor.CallerRunsPolicy() );5.3 守护线程与用户线程的区别类型行为适用场景用户线程(默认)JVM 等待所有用户线程结束后才退出执行业务逻辑的线程池守护线程JVM 退出时立即终止,不管是否执行完后台辅助任务(如日志刷盘)六、线程池动态调整与运行时管理6.1 可动态调整的参数// 以下方法均可在运行时调用 executor.setCorePoolSize(newCoreSize); executor.setMaximumPoolSize(newMaxSize); executor.setKeepAliveTime(newTime, TimeUnit.SECONDS); executor.setRejectedExecutionHandler(newHandler); executor.setThreadFactory(newFactory); // 仅影响后续新建线程 executor.allowCoreThreadTimeOut(true); // 允许核心线程超时回收6.2 动态调整实战:基于负载的自动扩缩容@Component public class AdaptiveThreadPoolManager { private final ThreadPoolExecutor executor; private final int minCoreSize; private final int maxCoreSize; @Scheduled(fixedDelay = 10000) public void adjustPoolSize() { int queueSize = executor.getQueue().size(); int activeCount = executor.getActiveCount(); int currentCore = executor.getCorePoolSize(); // 队列堆积严重且活跃线程接近核心数 → 增加核心线程 if (queueSize > 500 && activeCount >= currentCore * 0.9) { int newCore = Math.min(currentCore + 2, maxCoreSize); executor.setCorePoolSize(newCore); log.info("队列堆积,增加核心线程至 {}", newCore); } // 队列持续为空且核心线程数大于最小值 → 减少核心线程 if (queueSize == 0 && activeCount < currentCore * 0.3 && currentCore > minCoreSize) { int newCore = Math.max(currentCore - 1, minCoreSize); executor.setCorePoolSize(newCore); log.info("队列空闲,减少核心线程至 {}", newCore); } } }6.3 预热核心线程// 提前创建所有核心线程,避免懒加载导致初期响应慢 executor.prestartAllCoreThreads(); // 或者预热指定数量 executor.prestartCoreThread();七、任务提交与结果处理进阶7.1 execute vs submit 的区别方法返回值异常处理execute(Runnable)void异常抛出到 UncaughtExceptionHandlersubmit(Runnable)Future<?>异常封装在 Future.get() 中submit(Callable<T>)Future<T>异常封装在 Future.get() 中// execute 的异常不会被吞掉,会传播到线程的 UncaughtExceptionHandler executor.execute(() -> { throw new RuntimeException("execute异常"); }); // submit 的异常会被封装在 Future 中,必须调用 get() 才会抛出 Future<?> future = executor.submit(() -> { throw new RuntimeException("submit异常"); }); try { future.get(); // 这里会抛出 ExecutionException } catch (ExecutionException e) { e.getCause(); // 原始异常 }7.2 CompletionService —— 先完成先获取当需要处理一批任务并按照完成顺序获取结果时,使用 CompletionService 比手动遍历 Future 列表更高效。ExecutorService executor = Executors.newFixedThreadPool(10); CompletionService<String> completionService = new ExecutorCompletionService<>(executor); // 提交多个任务 List<Future<String>> futures = new ArrayList<>(); for (int i = 0; i < 100; i++) { final int taskId = i; futures.add(completionService.submit(() -> { Thread.sleep(ThreadLocalRandom.current().nextInt(1000)); return "Task-" + taskId + " result"; })); } // 按完成顺序获取结果 for (int i = 0; i < 100; i++) { Future<String> completed = completionService.take(); // 阻塞等待下一个完成的任务 String result = completed.get(); System.out.println(result); }7.3 invokeAll 与 invokeAnyList<Callable<String>> tasks = Arrays.asList( () -> { Thread.sleep(1000); return "A"; }, () -> { Thread.sleep(500); return "B"; } ); // invokeAll:等待所有任务完成,返回所有 Future List<Future<String>> allFutures = executor.invokeAll(tasks); // invokeAny:只要有一个任务成功完成就返回其结果,其他任务取消 String firstResult = executor.invokeAny(tasks); // 返回 "B"八、线程池钩子方法与扩展点8.1 三个可重写的钩子方法public class MonitoredThreadPool extends ThreadPoolExecutor { public MonitoredThreadPool(...) { super(...); } @Override protected void beforeExecute(Thread t, Runnable r) { super.beforeExecute(t, r); // 可以在此记录任务开始时间,存入 ThreadLocal TaskContext.start(t.getName(), r); } @Override protected void afterExecute(Runnable r, Throwable t) { super.afterExecute(r, t); try { if (t != null) { log.error("任务执行异常", t); metrics.recordError(); } // 记录任务执行耗时 long duration = TaskContext.getDuration(); metrics.recordTaskDuration(duration); } finally { TaskContext.clear(); } } @Override protected void terminated() { super.terminated(); log.info("线程池已完全终止"); metrics.unregister(); } }8.2 使用 ThreadLocal 传递上下文(需注意清理)public class TaskContext { private static final ThreadLocal<Long> START_TIME = new ThreadLocal<>(); private static final ThreadLocal<String> TASK_NAME = new ThreadLocal<>(); public static void start(String threadName, Runnable task) { START_TIME.set(System.nanoTime()); TASK_NAME.set(task.getClass().getSimpleName()); } public static long getDuration() { Long start = START_TIME.get(); return start == null ? -1 : TimeUnit.NANOSECONDS.toMillis(System.nanoTime() - start); } public static void clear() { START_TIME.remove(); // ⚠️ 必须清理,否则会导致内存泄漏 TASK_NAME.remove(); } }为什么必须清理 ThreadLocal?线程池中的线程是复用的,如果不在 afterExecute 中清理,ThreadLocal 中的数据会残留到下一个任务中,导致数据错乱或内存泄漏(Value 对象一直被强引用)。九、ScheduledThreadPoolExecutor 深度解析9.1 架构与继承关系ThreadPoolExecutor ▲ │ ScheduledThreadPoolExecutor │ └── 内部类 ScheduledFutureTask (实现 RunnableScheduledFuture) └── 内部类 DelayedWorkQueue (基于堆的延迟队列)9.2 DelayedWorkQueue 实现原理数据结构:二叉小顶堆,按照任务的下次执行时间排序线程等待机制:Leader-Follower 模式,只有一个 Leader 线程等待堆顶任务的到期时间,其他线程无限期等待,减少不必要的唤醒// 简化的 Leader-Follower 逻辑 public RunnableScheduledFuture<?> take() throws InterruptedException { lock.lockInterruptibly(); try { for (;;) { RunnableScheduledFuture<?> first = queue[0]; if (first == null) { available.await(); // 无任务,等待 } else { long delay = first.getDelay(NANOSECONDS); if (delay <= 0) { return finishPoll(first); // 任务到期,取出 } first = null; if (leader != null) { available.await(); // 已有 Leader,当前线程等待 } else { Thread thisThread = Thread.currentThread(); leader = thisThread; try { available.awaitNanos(delay); // 作为 Leader 等待 } finally { if (leader == thisThread) leader = null; } } } } } finally { if (leader == null && queue[0] != null) available.signal(); // 唤醒一个 Follower lock.unlock(); } }9.3 ScheduledFutureTask 关键字段private class ScheduledFutureTask<V> extends FutureTask<V> implements RunnableScheduledFuture<V> { private long time; // 下次执行时间(纳秒) private final long period; // 周期:正数 fixedRate,负数 fixedDelay,0 表示单次 private final long sequenceNumber; // 全局序号,用于任务排序 }9.4 scheduleAtFixedRate 与 scheduleWithFixedDelay 源码差异// 在 run 方法中重新设置下次执行时间 public void run() { boolean periodic = isPeriodic(); if (!canRunInCurrentRunState(periodic)) cancel(false); else if (!periodic) super.run(); else if (super.runAndReset()) { setNextRunTime(); // 关键差异点 reExecutePeriodic(outerTask); } } private void setNextRunTime() { long p = period; if (p > 0) // scheduleAtFixedRate time += p; // 基于上次计划时间累加,保持固定频率 else // scheduleWithFixedDelay time = triggerTime(-p); // 基于当前完成时间 + delay }后果对比示例:// 假设任务执行需要 4 秒,周期/延迟设为 3 秒 // scheduleAtFixedRate (period=3 > 0) // 任务1:0秒开始,4秒结束 → 下次执行时间 = 0+3=3秒(已过),立即执行任务2 // 任务2:3秒(实际4秒)开始... // scheduleWithFixedDelay (period=-3) // 任务1:0秒开始,4秒结束 → 下次执行时间 = 当前时间+3秒 = 7秒十、线程池监控与告警体系搭建10.1 基于 Micrometer 的完整监控实现@Component public class ThreadPoolMetricsBinder { private final MeterRegistry meterRegistry; private final Map<String, ThreadPoolExecutor> pools = new ConcurrentHashMap<>(); public void register(String poolName, ThreadPoolExecutor executor) { pools.put(poolName, executor); // 核心指标 Gauge Gauge.builder("thread.pool.core.size", executor, ThreadPoolExecutor::getCorePoolSize) .tag("pool", poolName).register(meterRegistry); Gauge.builder("thread.pool.max.size", executor, ThreadPoolExecutor::getMaximumPoolSize) .tag("pool", poolName).register(meterRegistry); Gauge.builder("thread.pool.active", executor, ThreadPoolExecutor::getActiveCount) .tag("pool", poolName).register(meterRegistry); Gauge.builder("thread.pool.pool.size", executor, ThreadPoolExecutor::getPoolSize) .tag("pool", poolName).register(meterRegistry); Gauge.builder("thread.pool.largest.size", executor, ThreadPoolExecutor::getLargestPoolSize) .tag("pool", poolName).register(meterRegistry); Gauge.builder("thread.pool.queue.size", executor, e -> e.getQueue().size()) .tag("pool", poolName).register(meterRegistry); Gauge.builder("thread.pool.queue.remaining", executor, e -> e.getQueue().remainingCapacity()) .tag("pool", poolName).register(meterRegistry); Gauge.builder("thread.pool.completed", executor, e -> e.getCompletedTaskCount()) .tag("pool", poolName).register(meterRegistry); // 派生计算指标(队列使用率) Gauge.builder("thread.pool.queue.usage", executor, e -> { int size = e.getQueue().size(); int capacity = size + e.getQueue().remainingCapacity(); return capacity > 0 ? (double) size / capacity : 0; }).tag("pool", poolName).register(meterRegistry); } // 定时打印监控日志(便于开发调试) @Scheduled(fixedDelay = 30000) public void logPoolStats() { pools.forEach((name, pool) -> { log.info("线程池[{}] 状态 - 核心:{}, 当前:{}, 活跃:{}, 队列:{}/{}, 完成:{}, 拒绝:{}", name, pool.getCorePoolSize(), pool.getPoolSize(), pool.getActiveCount(), pool.getQueue().size(), pool.getQueue().size() + pool.getQueue().remainingCapacity(), pool.getCompletedTaskCount(), getRejectedCount(pool) // 需自行记录 ); }); } }10.2 拒绝计数与异常计数记录public class InstrumentedThreadPool extends ThreadPoolExecutor { private final Counter rejectedCounter; private final Counter exceptionCounter; public InstrumentedThreadPool(..., MeterRegistry registry, String poolName) { super(...); this.rejectedCounter = Counter.builder("thread.pool.rejected") .tag("pool", poolName).register(registry); this.exceptionCounter = Counter.builder("thread.pool.exception") .tag("pool", poolName).register(registry); } @Override protected void afterExecute(Runnable r, Throwable t) { super.afterExecute(r, t); if (t != null) { exceptionCounter.increment(); } } // 自定义拒绝处理器,在其内部调用 rejectedCounter.increment() }10.3 告警规则配置(Prometheus + AlertManager 示例)groups: - name: thread_pool_alerts interval: 30s rules: - alert: ThreadPoolQueueHighUsage expr: thread_pool_queue_usage > 0.8 for: 5m annotations: summary: "线程池 {{ $labels.pool }} 队列使用率超过 80%" - alert: ThreadPoolRejectOccurred expr: increase(thread_pool_rejected_total[1m]) > 0 annotations: summary: "线程池 {{ $labels.pool }} 出现任务拒绝" - alert: ThreadPoolActiveThreadHigh expr: thread_pool_active / thread_pool_max_size > 0.9 for: 10m annotations: summary: "线程池 {{ $labels.pool }} 活跃线程数持续接近上限"十一、常见陷阱与最佳实践11.1 陷阱一:ThreadLocal 内存泄漏// ❌ 错误:在任务中设置 ThreadLocal 但未清理 executor.submit(() -> { ThreadLocal<HeavyObject> tl = new ThreadLocal<>(); tl.set(new HeavyObject()); // 任务结束后 tl 仍被线程持有 // 未调用 remove() }); // ✅ 正确:使用 try-finally 确保清理 executor.submit(() -> { ThreadLocal<HeavyObject> tl = new ThreadLocal<>(); try { tl.set(new HeavyObject()); // 业务逻辑 } finally { tl.remove(); } });11.2 陷阱二:线程池被多个任务共享时的并发问题// 线程池本身是线程安全的,但要注意: // 如果任务间共享可变状态,必须加锁 class SharedCounter { private int count = 0; public void increment() { count++; } // 非原子操作 } // ❌ 多个任务同时修改共享对象 SharedCounter counter = new SharedCounter(); for (int i = 0; i < 1000; i++) { executor.submit(counter::increment); } // 结果可能小于 1000 // ✅ 使用原子类或同步 class SafeCounter { private AtomicInteger count = new AtomicInteger(0); public void increment() { count.incrementAndGet(); } }11.3 陷阱三:使用无界队列导致 OOM// ❌ 致命错误 ExecutorService pool = Executors.newFixedThreadPool(10); // 队列是 LinkedBlockingQueue,默认 Integer.MAX_VALUE // 如果任务提交速度远大于处理速度,队列会无限增长直至 OOM // ✅ 使用有界队列 new ThreadPoolExecutor(10, 10, 0L, TimeUnit.MILLISECONDS, new ArrayBlockingQueue<>(1000), // 指定容量 ...);11.4 陷阱四:异常被吞没// submit 的异常如果不调用 get() 会被静默吞掉 Future<?> future = executor.submit(() -> { throw new RuntimeException(); }); // 这里不会打印任何异常日志 // 解决方案: // 1. 始终调用 get() 处理异常 // 2. 使用 execute 代替 submit // 3. 在 afterExecute 中捕获并记录异常(如前面所示)11.5 陷阱五:线程池关闭后仍有任务提交executor.shutdown(); executor.execute(task); // 抛出 RejectedExecutionException11.6 最佳实践清单实践项具体做法禁止 Executors 快捷方法显式使用 ThreadPoolExecutor 构造函数有界队列 + 合理拒绝策略根据业务选择 CallerRunsPolicy 或自定义自定义线程命名使用 NamedThreadFactory监控指标全覆盖集成 Micrometer,暴露关键指标优雅关闭使用两阶段关闭模板异常处理在 afterExecute 中统一记录未捕获异常上下文传递使用 TransmittableThreadLocal 或手动清理参数动态化提供配置中心调整核心参数的能力十二、生产级线程池配置模板12.1 通用业务线程池配置@Configuration public class ThreadPoolConfig { @Bean("commonExecutor") public ThreadPoolExecutor commonExecutor() { int cpuCores = Runtime.getRuntime().availableProcessors(); ThreadPoolExecutor executor = new ThreadPoolExecutor( cpuCores * 2, // corePoolSize cpuCores * 4, // maximumPoolSize 60, TimeUnit.SECONDS, // keepAlive new ArrayBlockingQueue<>(2000), // 有界队列 new NamedThreadFactory("common-pool", false, Thread.NORM_PRIORITY), new ThreadPoolExecutor.CallerRunsPolicy() ); executor.allowCoreThreadTimeOut(true); // 空闲时回收核心线程 return executor; } }12.2 高吞吐 I/O 密集型线程池@Bean("ioIntensiveExecutor") public ThreadPoolExecutor ioIntensiveExecutor() { // 根据 I/O 等待时间调整 int coreSize = Runtime.getRuntime().availableProcessors() * 2; int maxSize = coreSize * 4; return new ThreadPoolExecutor( coreSize, maxSize, 120, TimeUnit.SECONDS, new LinkedBlockingQueue<>(5000), // 容量较大 new NamedThreadFactory("io-pool", false, Thread.NORM_PRIORITY), new BlockingWaitPolicy(100) // 自定义阻塞等待拒绝策略 ); }12.3 定时任务专用线程池@Bean("scheduledExecutor") public ScheduledExecutorService scheduledExecutor() { return new ScheduledThreadPoolExecutor( Runtime.getRuntime().availableProcessors(), new NamedThreadFactory("scheduled-pool", false, Thread.NORM_PRIORITY), new ThreadPoolExecutor.CallerRunsPolicy() ); }12.4 带有完整监控的线程池构建器public class MonitoredThreadPoolBuilder { public static ThreadPoolExecutor build(String poolName, int core, int max, int queueCapacity, MeterRegistry registry) { ThreadPoolExecutor executor = new InstrumentedThreadPool( core, max, 60L, TimeUnit.SECONDS, new ArrayBlockingQueue<>(queueCapacity), new NamedThreadFactory(poolName), new MonitoredCallerRunsPolicy(registry), registry, poolName ); // 注册到监控绑定器 return executor; } }附录:快速参考卡线程池参数速查表参数推荐值范围备注corePoolSizeCPU核数 ~ CPU核数×2根据任务类型调整maximumPoolSizecorePoolSize ~ corePoolSize×4不宜过大,避免上下文切换开销keepAliveTime30s ~ 120s临时线程回收时间队列容量500 ~ 10000根据任务提交速率和处理速率计算拒绝策略选择指南场景推荐策略关键业务,不可丢失CallerRunsPolicy 或持久化策略可丢弃的非核心任务DiscardPolicy新任务优先级高于旧任务DiscardOldestPolicy快速失败,开发调试AbortPolicy(默认)线程数计算公式N = CPU 核心数 U = 目标 CPU 利用率 (0 < U <= 1) W/C = 等待时间与计算时间的比值 最优线程数 = N * U * (1 + W/C)以上内容涵盖了从基础概念到源码原理、从监控告警到生产实践的完整知识体系,可作为学习和工作的全面参考资料。
2026年04月09日
9 阅读
0 评论
0 点赞
2026-04-03
SpringBoot
Spring Boot 2.7 入门到精通笔记(苍穹外卖实战版)一、Spring Boot 概述1.1 什么是 Spring BootSpring Boot 是 Spring 框架的“脚手架”,核心设计理念是 约定优于配置。与传统 Spring 的区别:对比项传统 SpringSpring Boot配置方式繁琐的 XML 配置文件注解 + 自动配置依赖管理手动管理版本、处理冲突起步依赖(Starter)自动引入部署方式打包成 WAR 部署到 Tomcat内嵌 Tomcat,直接 java -jar开发效率需要大量样板代码只需关心业务逻辑1.2 Spring Boot 核心特性自动配置:根据类路径中的依赖自动配置 Bean起步依赖:通过 Starter 机制简化依赖管理内嵌容器:默认集成 Tomcat/Jetty/Undertow生产就绪:Actuator 提供监控、健康检查等能力1.3 Spring Boot 2.7 新特性(重点)Spring Boot 2.7 是一个重要版本,2.5 已停止维护[reference:0]。自动配置变更:自动配置注册文件从 META-INF/spring.factories 迁移到 META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports新增 @AutoConfiguration 注解,用于标识自动配置类[reference:1]支持 after/before 属性进行配置排序其他新特性:支持 GraphQL(spring-boot-starter-graphql)支持 Podman 容器引擎Web Server SSL 增强[reference:2]1.4 苍穹外卖项目概述苍穹外卖是一个企业级 O2O 外卖平台,分为管理端和用户端(微信小程序):角色功能管理端员工管理、菜品/套餐管理、订单管理、数据统计用户端登录授权、浏览点餐、购物车结算、订单追踪技术栈:Spring Boot 2.7 + Spring MVC + MyBatis-Plus + MySQL + Redis + OSS(阿里云)+ AOP + Swagger(Knife4j)+ WebSocket[reference:3]二、快速开始2.1 项目结构(苍穹外卖)苍穹外卖采用 Maven 多模块结构,各模块职责明确,便于维护和扩展:sky-take-out(父工程) ├── pom.xml # 统一依赖版本管理 ├── sky-common/ # 公共模块:工具类、常量、异常类 │ ├── constant/ # 常量定义(如JwtClaimsConstant) │ ├── context/ # ThreadLocal上下文 │ ├── exception/ # 自定义异常 │ ├── properties/ # 配置属性类 │ └── utils/ # 工具类(JWT、阿里OSS、微信支付等) ├── sky-pojo/ # 实体模块:Entity、DTO、VO │ ├── entity/ # 数据库实体类(与表一一对应) │ ├── dto/ # 数据传输对象(层间传递) │ └── vo/ # 视图对象(返回给前端) └── sky-server/ # 后端服务模块(核心业务代码) ├── config/ # 配置类(Redis、Swagger、WebMvc等) ├── controller/ # 控制器层 ├── service/ # 业务逻辑层 ├── mapper/ # 数据访问层 ├── interceptor/ # 拦截器(如JWT认证拦截器) ├── aspect/ # AOP切面(公共字段填充) └── SkyApplication.java # 启动类POJO 分类说明:Entity:实体类,与数据库表一一对应DTO(Data Transfer Object):数据传输对象,用于层间传递数据VO(View Object):视图对象,专门返回给前端展示[reference:4]2.2 父工程 POM 配置(版本管理)<!-- sky-take-out/pom.xml --> <parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>2.7.0</version> <relativePath/> </parent> <groupId>com.sky</groupId> <artifactId>sky-take-out</artifactId> <version>1.0-SNAPSHOT</version> <packaging>pom</packaging> <properties> <java.version>1.8</java.version> <mybatis-plus.version>3.5.2</mybatis-plus.version> </properties> <dependencies> <!-- 统一管理的依赖在这里声明版本 --> </dependencies>2.3 子模块依赖配置<!-- sky-server/pom.xml --> <dependencies> <!-- Spring Boot Web Starter --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> <!-- MyBatis-Plus Starter --> <dependency> <groupId>com.baomidou</groupId> <artifactId>mybatis-plus-boot-starter</artifactId> <version>3.5.2</version> </dependency> <!-- MySQL 驱动 --> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <scope>runtime</scope> </dependency> <!-- Redis --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-redis</artifactId> </dependency> <!-- 引用公共模块 --> <dependency> <groupId>com.sky</groupId> <artifactId>sky-common</artifactId> <version>1.0-SNAPSHOT</version> </dependency> </dependencies>2.4 启动类package com.sky; import lombok.extern.slf4j.Slf4j; import org.springframework.boot.SpringApplication; import org.springframework.boot.autoconfigure.SpringBootApplication; import org.springframework.transaction.annotation.EnableTransactionManagement; @Slf4j @SpringBootApplication @EnableTransactionManagement // 开启事务管理 public class SkyApplication { public static void main(String[] args) { SpringApplication.run(SkyApplication.class, args); log.info("苍穹外卖后端服务启动成功!"); } }@SpringBootApplication 复合注解解析:注解作用@SpringBootConfiguration标识为配置类(@Configuration 的派生)@ComponentScan开启组件扫描,扫描启动类所在包及子包@EnableAutoConfiguration开启自动配置(核心)自动配置原理(5 步必背):@SpringBootApplication → @EnableAutoConfiguration@Import(AutoConfigurationImportSelector.class) 导入选择器加载 META-INF/spring.factories(或 2.7 新路径 AutoConfiguration.imports)读取自动配置类的全限定类名通过 @ConditionalOnXxx 条件注解判断是否加载三、配置文件3.1 application.yml(苍穹外卖示例)server: port: 8080 spring: datasource: url: jdbc:mysql://localhost:3306/sky_take_out?serverTimezone=Asia/Shanghai&useUnicode=true&characterEncoding=utf-8 username: root password: 123456 driver-class-name: com.mysql.cj.jdbc.Driver redis: host: localhost port: 6379 database: 0 main: allow-circular-references: true # 允许循环依赖 mybatis-plus: configuration: log-impl: org.apache.ibatis.logging.stdout.StdOutImpl # 打印 SQL 日志 global-config: db-config: id-type: auto # 主键自增 mapper-locations: classpath:mapper/*.xml sky: jwt: admin-secret-key: "itcast" admin-ttl: 7200000 admin-token-name: "token"3.2 多环境配置(必背)创建多个配置文件,通过 spring.profiles.active 激活:文件环境application.yml默认配置(公共)application-dev.yml开发环境application-prod.yml生产环境# application.yml(主配置) spring: profiles: active: dev # 激活开发环境 # application-dev.yml server: port: 8081 spring: config: activate: on-profile: dev # application-prod.yml server: port: 80 spring: config: activate: on-profile: prod激活方式:# 命令行激活 java -jar sky-server.jar --spring.profiles.active=prod # IDEA VM options -Dspring.profiles.active=prod四、分层架构(苍穹外卖实战)4.1 分层架构图解(外卖公司比喻)层级角色苍穹外卖示例Controller客服经理(接电话)接收 HTTP 请求,返回 JSON 响应Service业务主管(管流程)实现业务逻辑,调用 MapperMapper仓库管理员(管库存)执行 SQL,操作数据库Entity货物清单与数据库表一一对应的实体4.2 Entity 层(实体类)package com.sky.entity; import lombok.AllArgsConstructor; import lombok.Builder; import lombok.Data; import lombok.NoArgsConstructor; import java.io.Serializable; import java.time.LocalDateTime; @Data @Builder @NoArgsConstructor @AllArgsConstructor public class Employee implements Serializable { private static final long serialVersionUID = 1L; private Long id; private String username; private String name; private String password; private String phone; private String sex; private String idNumber; private Integer status; private LocalDateTime createTime; private LocalDateTime updateTime; private Long createUser; private Long updateUser; }4.3 Mapper 层(数据访问层)package com.sky.mapper; import com.sky.entity.Employee; import org.apache.ibatis.annotations.Mapper; import org.apache.ibatis.annotations.Select; @Mapper public interface EmployeeMapper { @Select("select * from employee where username = #{username}") Employee getByUsername(String username); // 分页查询员工 Page<Employee> pageQuery(EmployeePageQueryDTO pageQueryDTO); // 更新员工信息 void update(Employee employee); }MyBatis-Plus 增强版(推荐) :// 继承 BaseMapper,自动拥有 CRUD 方法 @Mapper public interface DishMapper extends BaseMapper<Dish> { // 无需写任何基础 CRUD,BaseMapper 已提供: // insert(), deleteById(), updateById(), selectById(), selectList() 等 } // Service 层还可以继承 IService @Service public class DishService extends ServiceImpl<DishMapper, Dish> { // 自动拥有 saveBatch(), updateBatchById() 等批量方法 }4.4 Service 层(业务逻辑层)package com.sky.service.impl; import com.sky.context.BaseContext; import com.sky.dto.EmployeeDTO; import com.sky.entity.Employee; import com.sky.mapper.EmployeeMapper; import com.sky.service.EmployeeService; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.BeanUtils; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; import org.springframework.util.DigestUtils; @Slf4j @Service public class EmployeeServiceImpl implements EmployeeService { @Autowired private EmployeeMapper employeeMapper; @Override @Transactional // 开启事务 public void save(EmployeeDTO employeeDTO) { Employee employee = new Employee(); // 拷贝属性(DTO → Entity) BeanUtils.copyProperties(employeeDTO, employee); // 设置默认密码(MD5 加密) employee.setPassword(DigestUtils.md5DigestAsHex("123456".getBytes())); employee.setStatus(1); employee.setCreateTime(LocalDateTime.now()); employee.setUpdateTime(LocalDateTime.now()); // 从 ThreadLocal 获取当前操作用户 ID employee.setCreateUser(BaseContext.getCurrentId()); employee.setUpdateUser(BaseContext.getCurrentId()); employeeMapper.insert(employee); } }4.5 Controller 层(控制层)package com.sky.controller.admin; import com.sky.dto.EmployeeDTO; import com.sky.dto.EmployeeLoginDTO; import com.sky.dto.EmployeePageQueryDTO; import com.sky.result.PageResult; import com.sky.result.Result; import com.sky.service.EmployeeService; import com.sky.vo.EmployeeLoginVO; import io.swagger.annotations.Api; import io.swagger.annotations.ApiOperation; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.web.bind.annotation.*; @RestController @RequestMapping("/admin/employee") @Api(tags = "员工管理接口") @Slf4j public class EmployeeController { @Autowired private EmployeeService employeeService; @PostMapping("/login") @ApiOperation("员工登录") public Result<EmployeeLoginVO> login(@RequestBody EmployeeLoginDTO loginDTO) { log.info("员工登录:{}", loginDTO.getUsername()); EmployeeLoginVO employeeLoginVO = employeeService.login(loginDTO); return Result.success(employeeLoginVO); } @PostMapping @ApiOperation("新增员工") public Result save(@RequestBody EmployeeDTO employeeDTO) { log.info("新增员工:{}", employeeDTO); employeeService.save(employeeDTO); return Result.success(); } @GetMapping("/page") @ApiOperation("员工分页查询") public Result<PageResult> page(EmployeePageQueryDTO pageQueryDTO) { log.info("员工分页查询:{}", pageQueryDTO); PageResult pageResult = employeeService.pageQuery(pageQueryDTO); return Result.success(pageResult); } }4.6 分层调用链路总结HTTP 请求 ↓ Controller(接收参数,调用 Service) ↓ Service(业务逻辑,调用 Mapper) ↓ Mapper(执行 SQL,操作数据库) ↓ Entity(数据载体) ↓ 返回结果 → VO → JSON 响应⚠️ 常见错误:❌ Controller 直接调用 Mapper(越级)❌ Mapper 层写业务判断逻辑❌ Service 层写 SQL 拼接五、核心注解速查表5.1 组件注解注解用途层级@Component通用组件,交给 Spring 管理任意@ControllerWeb 控制器(配合 @ResponseBody 返回 JSON)表现层@RestController@Controller + @ResponseBody 的合并表现层@Service业务逻辑层组件业务层@Repository数据访问层组件,会转换持久层异常数据层@Repository 的特殊功能:将数据库操作抛出的异常自动转换为 Spring 统一的数据访问异常体系[reference:5]。5.2 请求映射注解注解作用@RequestMapping通用请求映射,可指定 method@GetMapping处理 GET 请求@PostMapping处理 POST 请求@PutMapping处理 PUT 请求@DeleteMapping处理 DELETE 请求5.3 参数绑定注解注解作用示例@PathVariable获取 URL 路径参数/user/{id} → @PathVariable Long id@RequestParam获取 URL 查询参数?name=张三 → @RequestParam String name@RequestBody获取请求体(JSON → 对象)@RequestBody EmployeeDTO dto@RequestHeader获取请求头@RequestHeader("token") String token5.4 依赖注入注解注解作用@Autowired按类型自动注入(Spring 原生)@Resource按名称自动注入(JSR-250)@Qualifier配合 @Autowired,指定 Bean 名称@Value注入配置文件中的值(如 ${server.port})@ConfigurationProperties批量注入配置属性5.5 配置与条件注解注解作用@Configuration标识配置类@Bean在配置类中声明 Bean@ConditionalOnClass类路径存在指定类时才生效@ConditionalOnMissingBean容器中没有指定 Bean 时才生效@ConditionalOnProperty配置文件存在指定属性时才生效六、全局异常处理6.1 苍穹外卖全局异常处理器package com.sky.handler; import com.sky.constant.MessageConstant; import com.sky.exception.BaseException; import com.sky.result.Result; import lombok.extern.slf4j.Slf4j; import org.springframework.web.bind.annotation.ExceptionHandler; import org.springframework.web.bind.annotation.RestControllerAdvice; @RestControllerAdvice @Slf4j public class GlobalExceptionHandler { // 捕获业务异常 @ExceptionHandler(BaseException.class) public Result exceptionHandler(BaseException ex) { log.error("业务异常:{}", ex.getMessage()); return Result.error(ex.getMessage()); } // 捕获 SQL 异常(如用户名重复) @ExceptionHandler(DuplicateKeyException.class) public Result handleDuplicateKeyException(DuplicateKeyException ex) { log.error("SQL 异常:{}", ex.getMessage()); return Result.error(MessageConstant.ALREADY_EXISTS); } // 兜底异常处理 @ExceptionHandler(Exception.class) public Result exceptionHandler(Exception ex) { log.error("系统异常:{}", ex.getMessage()); return Result.error(MessageConstant.UNKNOWN_ERROR); } }6.2 统一响应对象package com.sky.result; import lombok.Data; import java.io.Serializable; @Data public class Result<T> implements Serializable { private Integer code; // 1 成功,0 失败 private String msg; private T data; public static <T> Result<T> success() { Result<T> result = new Result<>(); result.code = 1; return result; } public static <T> Result<T> success(T data) { Result<T> result = new Result<>(); result.code = 1; result.data = data; return result; } public static <T> Result<T> error(String msg) { Result<T> result = new Result<>(); result.code = 0; result.msg = msg; return result; } }七、JWT 认证与拦截器7.1 JWT 工具类package com.sky.utils; import io.jsonwebtoken.*; import lombok.Data; import org.springframework.boot.context.properties.ConfigurationProperties; import org.springframework.stereotype.Component; import java.util.Date; import java.util.Map; @Data @Component @ConfigurationProperties(prefix = "sky.jwt") public class JwtUtil { private String adminSecretKey; private Long adminTtl; private String adminTokenName; // 生成 JWT 令牌 public String createJWT(Map<String, Object> claims, Long ttlMillis) { long nowMillis = System.currentTimeMillis(); Date now = new Date(nowMillis); JwtBuilder builder = Jwts.builder() .setClaims(claims) .setIssuedAt(now) .signWith(SignatureAlgorithm.HS256, adminSecretKey); if (ttlMillis > 0) { builder.setExpiration(new Date(nowMillis + ttlMillis)); } return builder.compact(); } // 解析 JWT 令牌 public Claims parseJWT(String token) { return Jwts.parser() .setSigningKey(adminSecretKey) .parseClaimsJws(token) .getBody(); } }7.2 JWT 认证拦截器package com.sky.interceptor; import com.sky.constant.JwtClaimsConstant; import com.sky.context.BaseContext; import com.sky.properties.JwtProperties; import com.sky.utils.JwtUtil; import io.jsonwebtoken.Claims; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Component; import org.springframework.web.method.HandlerMethod; import org.springframework.web.servlet.HandlerInterceptor; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; @Component @Slf4j public class JwtTokenAdminInterceptor implements HandlerInterceptor { @Autowired private JwtUtil jwtUtil; @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) { // 判断当前拦截的是否为 Controller 方法 if (!(handler instanceof HandlerMethod)) { return true; } // 从请求头中获取令牌 String token = request.getHeader(jwtUtil.getAdminTokenName()); // 解析令牌 try { Claims claims = jwtUtil.parseJWT(token); Long empId = Long.valueOf(claims.get(JwtClaimsConstant.EMP_ID).toString()); // 将员工 ID 存入 ThreadLocal(供后续业务使用) BaseContext.setCurrentId(empId); log.info("当前员工 ID:{}", empId); return true; } catch (Exception ex) { log.error("JWT 解析失败:{}", ex.getMessage()); response.setStatus(401); return false; } } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) { // 请求结束后清除 ThreadLocal,防止内存泄漏 BaseContext.removeCurrentId(); } }7.3 ThreadLocal 工具类(苍穹外卖)package com.sky.context; public class BaseContext { private static ThreadLocal<Long> threadLocal = new ThreadLocal<>(); public static void setCurrentId(Long id) { threadLocal.set(id); } public static Long getCurrentId() { return threadLocal.get(); } public static void removeCurrentId() { threadLocal.remove(); } }ThreadLocal 原理:每个线程拥有自己的变量副本,线程隔离。用完必须 remove(),否则在线程池环境下会导致内存泄漏[reference:6]。7.4 注册拦截器package com.sky.config; import com.sky.interceptor.JwtTokenAdminInterceptor; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.context.annotation.Configuration; import org.springframework.web.servlet.config.annotation.InterceptorRegistry; import org.springframework.web.servlet.config.annotation.WebMvcConfigurationSupport; @Configuration public class WebMvcConfig extends WebMvcConfigurationSupport { @Autowired private JwtTokenAdminInterceptor jwtTokenAdminInterceptor; @Override protected void addInterceptors(InterceptorRegistry registry) { registry.addInterceptor(jwtTokenAdminInterceptor) .addPathPatterns("/admin/**") // 拦截所有 /admin 开头的请求 .excludePathPatterns("/admin/employee/login") // 排除登录接口 .excludePathPatterns("/admin/employee/logout"); } }八、AOP 切面编程(公共字段自动填充)8.1 自定义注解package com.sky.annotation; import com.sky.enumeration.OperationType; import java.lang.annotation.*; @Target(ElementType.METHOD) @Retention(RetentionPolicy.RUNTIME) public @interface AutoFill { OperationType value(); // UPDATE 或 INSERT }8.2 切面类实现package com.sky.aspect; import com.sky.annotation.AutoFill; import com.sky.constant.AutoFillConstant; import com.sky.context.BaseContext; import com.sky.enumeration.OperationType; import lombok.extern.slf4j.Slf4j; import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.annotation.Pointcut; import org.aspectj.lang.reflect.MethodSignature; import org.springframework.stereotype.Component; import java.lang.reflect.Method; import java.time.LocalDateTime; @Aspect @Component @Slf4j public class AutoFillAspect { // 切点:标注了 @AutoFill 注解的方法 @Pointcut("@annotation(com.sky.annotation.AutoFill)") public void autoFillPointCut() {} @Before("autoFillPointCut()") public void autoFill(JoinPoint joinPoint) { // 获取当前被拦截的方法上的注解 MethodSignature signature = (MethodSignature) joinPoint.getSignature(); AutoFill autoFill = signature.getMethod().getAnnotation(AutoFill.class); OperationType operationType = autoFill.value(); // 获取方法的参数(实体对象) Object[] args = joinPoint.getArgs(); if (args == null || args.length == 0) return; Object entity = args[0]; // 准备要赋的值 LocalDateTime now = LocalDateTime.now(); Long currentId = BaseContext.getCurrentId(); // 根据操作类型设置不同字段 if (operationType == OperationType.INSERT) { // 插入操作:设置 createTime, updateTime, createUser, updateUser try { Method setCreateTime = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_CREATE_TIME, LocalDateTime.class); Method setUpdateTime = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_UPDATE_TIME, LocalDateTime.class); Method setCreateUser = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_CREATE_USER, Long.class); Method setUpdateUser = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_UPDATE_USER, Long.class); setCreateTime.invoke(entity, now); setUpdateTime.invoke(entity, now); setCreateUser.invoke(entity, currentId); setUpdateUser.invoke(entity, currentId); } catch (Exception e) { log.error("公共字段填充失败:{}", e.getMessage()); } } else if (operationType == OperationType.UPDATE) { // 更新操作:设置 updateTime, updateUser try { Method setUpdateTime = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_UPDATE_TIME, LocalDateTime.class); Method setUpdateUser = entity.getClass().getDeclaredMethod(AutoFillConstant.SET_UPDATE_USER, Long.class); setUpdateTime.invoke(entity, now); setUpdateUser.invoke(entity, currentId); } catch (Exception e) { log.error("公共字段填充失败:{}", e.getMessage()); } } } }8.3 使用示例@Mapper public interface EmployeeMapper { @AutoFill(OperationType.INSERT) void insert(Employee employee); @AutoFill(OperationType.UPDATE) void update(Employee employee); }九、缓存与 Redis9.1 Spring Cache 配置package com.sky.config; import org.springframework.cache.annotation.EnableCaching; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.data.redis.cache.RedisCacheConfiguration; import org.springframework.data.redis.serializer.RedisSerializationContext; import org.springframework.data.redis.serializer.StringRedisSerializer; import java.time.Duration; @Configuration @EnableCaching public class RedisConfig { @Bean public RedisCacheConfiguration cacheConfiguration() { return RedisCacheConfiguration.defaultCacheConfig() .entryTtl(Duration.ofMinutes(30)) // 缓存过期时间 30 分钟 .disableCachingNullValues() // 不缓存 null 值 .serializeKeysWith(RedisSerializationContext.SerializationPair .fromSerializer(new StringRedisSerializer())); } }9.2 缓存注解使用@Service @Slf4j public class DishServiceImpl implements DishService { @Autowired private DishMapper dishMapper; // 查询时缓存:key = "dish:categoryId:" + categoryId @Cacheable(value = "dish", key = "'dish:categoryId:' + #categoryId") public List<Dish> getByCategoryId(Long categoryId) { return dishMapper.getByCategoryId(categoryId); } // 更新时清空缓存 @CacheEvict(value = "dish", key = "'dish:categoryId:' + #dishDTO.categoryId") public void updateWithFlavor(DishDTO dishDTO) { // 更新业务逻辑 } // 批量清空缓存(key 模糊匹配) @CacheEvict(value = "dish", allEntries = true) public void deleteBatch(List<Long> ids) { // 删除业务逻辑 } }9.3 常用 Redis 操作(StringRedisTemplate)@Service public class ShopStatusService { @Autowired private StringRedisTemplate redisTemplate; private static final String SHOP_STATUS_KEY = "SHOP_STATUS"; public void setStatus(Integer status) { redisTemplate.opsForValue().set(SHOP_STATUS_KEY, String.valueOf(status)); } public Integer getStatus() { String status = redisTemplate.opsForValue().get(SHOP_STATUS_KEY); return status == null ? 0 : Integer.parseInt(status); } }十、Spring Task 定时任务10.1 开启定时任务@SpringBootApplication @EnableScheduling // 开启定时任务 public class SkyApplication { public static void main(String[] args) { SpringApplication.run(SkyApplication.class, args); } }10.2 订单状态定时处理package com.sky.task; import com.sky.service.OrderService; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.scheduling.annotation.Scheduled; import org.springframework.stereotype.Component; @Component @Slf4j public class OrderTask { @Autowired private OrderService orderService; // 每天凌晨 1 点处理超时未支付的订单 @Scheduled(cron = "0 0 1 * * ?") public void processTimeoutOrder() { log.info("定时任务:处理超时未支付订单"); orderService.processTimeoutOrders(); } // 每分钟执行一次,处理配送中的订单(送达后自动完成) @Scheduled(cron = "0 * * * * ?") public void processDeliveryOrder() { log.info("定时任务:处理配送中订单"); orderService.processDeliveryOrders(); } }10.3 Cron 表达式速查字段允许值特殊字符秒0-59, - * /分0-59, - * /时0-23, - * /日1-31, - * ? / L W月1-12, - * /周0-7(0 或 7 都代表周日), - * ? / L #年(可选)空或 1970-2099, - * /常用表达式:0 0 1 * * ? → 每天凌晨 1 点0 */5 * * * ? → 每 5 分钟一次0 0 12 * * MON-FRI → 工作日中午 12 点十一、WebSocket 实时通信11.1 WebSocket 配置package com.sky.config; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.web.socket.server.standard.ServerEndpointExporter; @Configuration public class WebSocketConfig { @Bean public ServerEndpointExporter serverEndpointExporter() { return new ServerEndpointExporter(); } }11.2 WebSocket 服务端package com.sky.websocket; import lombok.extern.slf4j.Slf4j; import org.springframework.stereotype.Component; import javax.websocket.*; import javax.websocket.server.PathParam; import javax.websocket.server.ServerEndpoint; import java.util.concurrent.ConcurrentHashMap; @Component @ServerEndpoint("/ws/{sid}") @Slf4j public class WebSocketServer { // 存储所有在线会话(线程安全) private static ConcurrentHashMap<String, Session> sessionMap = new ConcurrentHashMap<>(); @OnOpen public void onOpen(Session session, @PathParam("sid") String sid) { sessionMap.put(sid, session); log.info("WebSocket 连接建立:{}", sid); } @OnClose public void onClose(@PathParam("sid") String sid) { sessionMap.remove(sid); log.info("WebSocket 连接关闭:{}", sid); } @OnMessage public void onMessage(String message, Session session) { log.info("收到消息:{}", message); } @OnError public void onError(Session session, Throwable error) { log.error("WebSocket 错误:{}", error.getMessage()); } // 发送消息给指定用户 public void sendToClient(String sid, String message) { Session session = sessionMap.get(sid); if (session != null && session.isOpen()) { try { session.getBasicRemote().sendText(message); log.info("发送消息给 {}:{}", sid, message); } catch (Exception e) { log.error("发送失败:{}", e.getMessage()); } } } }11.3 来单提醒/催单场景@Service public class OrderServiceImpl implements OrderService { @Autowired private WebSocketServer webSocketServer; // 用户下单 → 来单提醒(推送给管理端) public void submit(OrdersSubmitDTO submitDTO) { // ... 下单业务逻辑 // 来单提醒 Map<String, Object> reminder = new HashMap<>(); reminder.put("type", 1); // 1=来单提醒 reminder.put("orderId", order.getId()); reminder.put("content", "您有一笔新订单"); webSocketServer.sendToClient("admin", JSON.toJSONString(reminder)); } // 用户催单 → 推送给管理端 public void reminder(Long orderId) { Map<String, Object> reminder = new HashMap<>(); reminder.put("type", 2); // 2=用户催单 reminder.put("orderId", orderId); reminder.put("content", "用户催单啦"); webSocketServer.sendToClient("admin", JSON.toJSONString(reminder)); } }十二、文件上传(阿里云 OSS)12.1 OSS 配置属性类package com.sky.properties; import lombok.Data; import org.springframework.boot.context.properties.ConfigurationProperties; import org.springframework.stereotype.Component; @Data @Component @ConfigurationProperties(prefix = "sky.alioss") public class AliOssProperties { private String endpoint; private String accessKeyId; private String accessKeySecret; private String bucketName; }12.2 OSS 工具类package com.sky.utils; import com.aliyun.oss.OSS; import com.aliyun.oss.OSSClientBuilder; import com.sky.properties.AliOssProperties; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Component; import org.springframework.web.multipart.MultipartFile; import java.io.IOException; import java.util.UUID; @Component @Slf4j public class AliOssUtil { @Autowired private AliOssProperties aliOssProperties; public String upload(MultipartFile file) { String endpoint = aliOssProperties.getEndpoint(); String accessKeyId = aliOssProperties.getAccessKeyId(); String accessKeySecret = aliOssProperties.getAccessKeySecret(); String bucketName = aliOssProperties.getBucketName(); // 生成唯一文件名(防止覆盖) String originalFilename = file.getOriginalFilename(); String extension = originalFilename.substring(originalFilename.lastIndexOf(".")); String objectName = UUID.randomUUID().toString() + extension; OSS ossClient = new OSSClientBuilder().build(endpoint, accessKeyId, accessKeySecret); try { ossClient.putObject(bucketName, objectName, file.getInputStream()); // 返回文件访问 URL return "https://" + bucketName + "." + endpoint + "/" + objectName; } catch (IOException e) { log.error("文件上传失败:{}", e.getMessage()); throw new RuntimeException(e); } finally { ossClient.shutdown(); } } }12.3 通用上传接口@RestController @RequestMapping("/admin/common") @Api(tags = "通用接口") @Slf4j public class CommonController { @Autowired private AliOssUtil aliOssUtil; @PostMapping("/upload") @ApiOperation("文件上传") public Result<String> upload(MultipartFile file) { log.info("文件上传:{}", file.getOriginalFilename()); String url = aliOssUtil.upload(file); return Result.success(url); } }十三、Knife4j + Swagger 接口文档13.1 添加依赖<dependency> <groupId>com.github.xiaoymin</groupId> <artifactId>knife4j-spring-boot-starter</artifactId> <version>3.0.3</version> </dependency>13.2 Swagger 配置类package com.sky.config; import com.github.xiaoymin.knife4j.spring.annotations.EnableKnife4j; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import springfox.documentation.builders.ApiInfoBuilder; import springfox.documentation.builders.PathSelectors; import springfox.documentation.builders.RequestHandlerSelectors; import springfox.documentation.service.ApiInfo; import springfox.documentation.spi.DocumentationType; import springfox.documentation.spring.web.plugins.Docket; import springfox.documentation.swagger2.annotations.EnableSwagger2; @Configuration @EnableSwagger2 @EnableKnife4j public class SwaggerConfig { @Bean public Docket adminApi() { return new Docket(DocumentationType.SWAGGER_2) .groupName("管理端接口") .apiInfo(apiInfo()) .select() .apis(RequestHandlerSelectors.basePackage("com.sky.controller.admin")) .paths(PathSelectors.any()) .build(); } @Bean public Docket userApi() { return new Docket(DocumentationType.SWAGGER_2) .groupName("用户端接口") .apiInfo(apiInfo()) .select() .apis(RequestHandlerSelectors.basePackage("com.sky.controller.user")) .paths(PathSelectors.any()) .build(); } private ApiInfo apiInfo() { return new ApiInfoBuilder() .title("苍穹外卖 API 文档") .version("1.0") .description("苍穹外卖后端接口文档") .build(); } }13.3 使用示例@Api(tags = "员工管理接口") @RestController @RequestMapping("/admin/employee") public class EmployeeController { @PostMapping("/login") @ApiOperation("员工登录") @ApiImplicitParams({ @ApiImplicitParam(name = "username", value = "用户名", required = true), @ApiImplicitParam(name = "password", value = "密码", required = true) }) public Result<EmployeeLoginVO> login(@RequestBody EmployeeLoginDTO loginDTO) { // ... } }访问地址:http://localhost:8080/doc.html十四、常用工具类速查14.1 BeanUtils(属性拷贝)// DTO → Entity Employee employee = new Employee(); BeanUtils.copyProperties(employeeDTO, employee);14.2 DigestUtils(MD5 加密)// 密码 MD5 加密 String encryptedPassword = DigestUtils.md5DigestAsHex("123456".getBytes());14.3 Jackson(JSON 序列化)// 对象 → JSON 字符串 String jsonStr = JSON.toJSONString(object); // JSON 字符串 → 对象 User user = JSON.parseObject(jsonStr, User.class);14.4 LocalDateTime(时间处理)// 获取当前时间 LocalDateTime now = LocalDateTime.now(); // 时间格式化 DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy-MM-dd HH:mm:ss"); String formatted = now.format(formatter);十五、项目部署与运行15.1 打包# 在项目根目录执行 mvn clean package # 跳过测试打包 mvn clean package -DskipTests15.2 运行# 运行 JAR 包 java -jar sky-server.jar # 指定环境运行 java -jar sky-server.jar --spring.profiles.active=prod # 后台运行 nohup java -jar sky-server.jar > app.log 2>&1 &15.3 Nginx 反向代理配置server { listen 80; server_name api.sky.com; location / { proxy_pass http://127.0.0.1:8080; proxy_set_header Host $host; proxy_set_header X-Real-IP $remote_addr; } }十六、快速记忆口诀启动流程SpringApplication.run → 初始化上下文 → 加载自动配置 → 启动内嵌容器 → 注册 Controller → 应用就绪自动配置@SpringBootApplication → @EnableAutoConfiguration → @Import → spring.factories → @ConditionalOnXxx分层架构Controller 接请求,Service 写逻辑,Mapper 查数据库,Entity 存数据JWT 认证登录签发 Token → 请求携带 Header → 拦截器校验 → ThreadLocal 存用户 ID → 业务使用 → 请求结束清除缓存注解@Cacheable 查缓存,@CachePut 更新缓存,@CacheEvict 清缓存定时任务@EnableScheduling 开启,@Scheduled(cron) 标记,Cron 表达式定时间全局异常@RestControllerAdvice 全局处理,@ExceptionHandler 分类捕获本笔记基于 Spring Boot 2.7 + 苍穹外卖实战项目整理,覆盖了从项目搭建到部署上线的完整流程,适合日常开发查阅和面试前背诵。建议结合源码实践加深理解。
2026年04月03日
12 阅读
0 评论
0 点赞
2026-04-03
多线程编程笔记
Java 多线程编程笔记一、多线程概述1.1 什么是多线程Multi‑threading:一个程序内同时运行多个线程,并发执行不同任务。线程:CPU 调度的最小单位。比进程更轻量(创建/切换开销小)。1.2 多线程的优势提高持续响应性(UI 不卡顿)充分利用多核 CPU改善用户体验提高系统吞吐量1.3 多线程面临的问题问题说明线程安全数据竞争、共享资源并发访问、原子性被破坏死锁线程相互等待资源,程序永久阻塞内存一致性缓存、指令重排、可见性问题调试困难并发 Bug 不可复现、时序依赖、执行路径复杂二、Java 多线程模型JVM 线程模型:线程映射(通常 1:1 映射到 OS 线程)线程调度:由操作系统调度器决定(抢占式 + 时间片轮转)Java 内存模型(JMM):定义变量的可见性、有序性、原子性规则三、线程创建方式(4 种)方式实现特点继承 Thread 类class MyThread extends Thread { run() }简单,但受单继承限制实现 Runnableclass MyRunnable implements Runnable { run() }避免单继承,可共享任务实例实现 Callableclass MyCallable implements Callable<T> { call() }有返回值、可抛异常线程池Executors.newFixedThreadPool(n)生产首选,资源复用示例(Lambda 简化):new Thread(() -> System.out.println("run")).start();3.1 Callable + Future 工作流程(“点菜模型”)你点菜 → 提交 Callable 任务厨师做菜 → 线程池异步执行服务员给你叫号器 → 返回 Future 对象你玩手机 → 主线程不阻塞想吃饭了,去取餐 → 调用 future.get()没做好就等(阻塞),做好了直接拿走Future<Integer> future = executor.submit(() -> 100); Integer result = future.get(); // 阻塞直到结果返回四、线程基础操作 & 常用方法4.1 核心方法清单方法作用状态变化start()启动线程NEW → RUNNABLEThread.sleep(millis)休眠RUNNABLE → TIMED_WAITINGjoin()等待该线程结束RUNNABLE → WAITINGjoin(timeout)等待超时RUNNABLE → TIMED_WAITINGyield()让出 CPU(仍在就绪队列)RUNNABLE(不改变状态)Thread.currentThread()获取当前线程-setName() / getName()设置/获取线程名-setDaemon(true)设为守护线程必须在 start() 前调用4.2 线程属性优先级:setPriority(1~10),仅建议,不保证顺序。守护线程:JVM 退出时自动终止(如 GC 线程)。4.3 异常处理(重点)普通 try-catch 抓不住子线程里的异常。使用 UncaughtExceptionHandler:Thread thread = new Thread(() -> { throw new RuntimeException("崩了"); }); thread.setUncaughtExceptionHandler((t, e) -> { System.out.println("线程 " + t.getName() + " 异常:" + e.getMessage()); }); thread.start();五、线程生命周期(完整 6 种状态)NEW → RUNNABLE → BLOCKED / WAITING / TIMED_WAITING → TERMINATED状态触发条件NEWnew Thread(),尚未 start()RUNNABLEstart() 后,等待 CPU 调度BLOCKED进入 synchronized 但锁被占用WAITINGwait()、join() 无超时、LockSupport.park()TIMED_WAITINGsleep(time)、wait(time)、join(time)TERMINATEDrun() 执行完毕或异常退出六、线程调度机制抢占式调度:高优先级可抢占低优先级(但不保证)。时间片轮转:同优先级线程轮流获得时间片。公平调度:JVM 尽量避免线程饥饿。七、线程监控工具工具用途jstack命令行生成线程堆栈快照(定位死锁)jconsole图形化监控(线程、内存、类加载)VisualVM功能最强大的可视化工具八、线程间协作(wait / notify / notifyAll)必须在 synchronized 块内调用,否则 IllegalMonitorStateException。wait():释放锁,进入 WAITING,等待 notify/notifyAll。notify():随机唤醒一个等待线程。notifyAll():唤醒所有等待线程。synchronized (lock) { while (条件不满足) lock.wait(); // 条件满足,执行操作 lock.notifyAll(); }九、同步机制详解(从简单到复杂)9.1 synchronized(内置锁,无脑首选)自动加锁/解锁,不会忘记释放。锁某个对象 → 最小粒度,性能最好。锁当前对象 this → 整个方法串行。锁 Class 对象 → 全局锁。// 同步代码块(推荐) private final Object lock = new Object(); public void deduct() { synchronized(lock) { /* 临界区 */ } } // 同步方法 public synchronized void add() { count++; } // 静态同步方法(类锁) public static synchronized void globalMethod() { }9.2 ReentrantLock(显式锁,更灵活)需要手动 lock() / unlock(),必须放在 finally 中。高级功能:tryLock(timeout):超时放弃,避免死锁。lockInterruptibly():可中断抢锁。new ReentrantLock(true):公平锁(排队不插队)。多 Condition:精准唤醒指定线程。private final Lock lock = new ReentrantLock(); public void update() { lock.lock(); try { // 业务逻辑 } finally { lock.unlock(); } }9.3 volatile(轻量级同步)只保证:可见性 + 禁止指令重排。不保证:原子性(不能用于 count++)。适用场景:布尔状态标记、单例双重检查中的 instance。private volatile boolean stop = false; public void run() { while (!stop) { /* 执行任务 */ } } public void stopTask() { stop = true; }9.4 原子类(无锁,CAS)AtomicInteger、AtomicLong、AtomicReference 等。高并发计数用 LongAdder(分段累加,性能更高)。private AtomicInteger count = new AtomicInteger(0); count.incrementAndGet(); // 原子+19.5 线程安全集合(必记两个)ConcurrentHashMap:分段锁 + CAS,高并发 Map。CopyOnWriteArrayList:读多写少场景,写时复制数组,不阻塞读。Map<String, Integer> map = new ConcurrentHashMap<>(); List<String> list = new CopyOnWriteArrayList<>();9.6 ThreadLocal(线程隔离)每个线程拥有自己的变量副本,不竞争。必须 remove(),否则内存泄漏(尤其在线程池中)。private ThreadLocal<String> userLocal = new ThreadLocal<>(); userLocal.set("张三"); String user = userLocal.get(); userLocal.remove(); // 用完必须清9.7 同步机制选择总结(背诵)场景推荐方案简单逻辑同步(库存、判断)synchronized需要超时、中断、公平锁ReentrantLock线程开关、状态标记volatile计数、累加、ID 生成AtomicInteger / LongAdder多线程操作 Map / ListConcurrentHashMap / CopyOnWriteArrayList单例中存线程私有数据ThreadLocal禁忌:❌ 不要用 volatile 做 i++❌ 不要在单例 Bean 中写普通成员变量(若多线程修改)❌ 不要用 HashMap / ArrayList 做并发写入❌ 不要只锁一半逻辑(如只锁修改,不锁判断)十、死锁10.1 死锁的四个必要条件(缺一不可)互斥:资源只能被一个线程占用。请求与保持:拿着自己的资源,还请求别人的资源。不可剥夺:资源不能被强行抢走。循环等待:线程间形成等待闭环。10.2 避免死锁的方法破坏任意一个条件(如规定锁的顺序、使用 tryLock 超时放弃)。十一、优雅停止线程废弃:stop()、suspend()、resume()(不安全)。推荐方式:中断机制:thread.interrupt() 设置中断标志,线程内检查 Thread.currentThread().isInterrupted() 自行停止。标志位:volatile boolean running,线程定期检查。// 标志位方式 public class MyTask implements Runnable { private volatile boolean running = true; public void stop() { running = false; } public void run() { while (running) { /* 任务 */ } // 清理资源 } }十二、线程间通信(5 种实战方式)方式说明常用场景共享变量(volatile / 原子类)一个线程改,另一个读状态标记、计数器wait()/notify()配合 synchronized生产者‑消费者Condition(配合 ReentrantLock)多路等待,精准唤醒复杂协调Semaphore控制同时访问的线程数限流、连接池CountDownLatch / CyclicBarrier线程协调工具主等子、互相等待12.1 Semaphore 示例Semaphore semaphore = new Semaphore(3); // 最多3个线程同时访问 semaphore.acquire(); // 获取许可(阻塞) semaphore.release(); // 释放许可12.2 CountDownLatch(倒计时门栓)CountDownLatch latch = new CountDownLatch(3); for (int i = 0; i < 3; i++) { new Thread(() -> { // 执行任务 latch.countDown(); }).start(); } latch.await(); // 主线程等待3个任务完成12.3 CyclicBarrier(循环栅栏)CyclicBarrier barrier = new CyclicBarrier(3, () -> System.out.println("三人齐了,出发")); for (int i = 0; i < 3; i++) { new Thread(() -> { barrier.await(); // 互相等待 }).start(); }十三、线程池(企业开发核心)13.1 为什么需要线程池资源复用(避免频繁创建/销毁线程)响应速度快(任务来了直接执行)可管理(控制并发数、监控)13.2 ThreadPoolExecutor 7 大核心参数(必背)new ThreadPoolExecutor( corePoolSize, // 核心线程数(常驻) maximumPoolSize, // 最大线程数(核心+临时) keepAliveTime, // 临时线程空闲存活时间 TimeUnit unit, // 时间单位 BlockingQueue<Runnable> workQueue, // 任务队列 ThreadFactory threadFactory, // 线程工厂(一般用默认) RejectedExecutionHandler handler // 拒绝策略 );13.3 执行流程(5 步,必背)提交任务 → 核心线程未满 → 新建核心线程执行核心线程满 → 任务加入阻塞队列队列满 → 新建临时线程执行总线程数 = 最大线程数 → 执行拒绝策略临时线程空闲超时 → 销毁(回到核心线程数)13.4 4 种拒绝策略(背)策略行为比喻AbortPolicy(默认)抛 RejectedExecutionException直接赶走客人,还吵架CallerRunsPolicy调用者线程自己执行任务让老板自己服务客人DiscardPolicy悄悄丢弃任务,不报错悄悄赶走客人,不吭声DiscardOldestPolicy丢弃队列中最老的任务,然后重提赶走排队最久的,让新人进13.5 Executors 提供的 4 种常用线程池(开发规范禁止使用,需手动创建)工厂方法特点适用场景newFixedThreadPool(n)固定线程数,无界队列重负载服务器newCachedThreadPool()动态线程数,空闲60s回收大量短期异步任务newSingleThreadExecutor()单线程,顺序执行任务需顺序处理newScheduledThreadPool(n)支持定时/延迟执行定时任务规范:不要用 Executors 创建,因为 FixedThreadPool 使用无界队列可能 OOM,CachedThreadPool 可能创建过多线程。必须使用 new ThreadPoolExecutor 自定义。13.6 线程数设置公式(背)CPU 密集型(计算为主):线程数 = CPU 核心数 + 1IO 密集型(网络、磁盘):线程数 = CPU 核心数 × 2 (或更多)十四、定时任务线程池(ScheduledThreadPoolExecutor)14.1 三种定时任务类型方法作用特点schedule(Runnable, delay, unit)延迟执行一次-scheduleAtFixedRate(...)固定频率执行不关心上次任务是否完成,可能叠加scheduleWithFixedDelay(...)固定延迟执行必须等上次完成,再等延迟时间ScheduledExecutorService scheduler = Executors.newScheduledThreadPool(2); scheduler.scheduleAtFixedRate(() -> System.out.println("ping"), 0, 2, TimeUnit.SECONDS);14.2 对比 Timer 的优势多线程,任务互不干扰(Timer 单线程,一个任务异常会影响全部)支持线程池复用十五、线程池优雅关闭(餐厅下班逻辑)挂出「停止接客」牌子 → shutdown()(不再接收新任务)让客人慢慢吃完 → awaitTermination(timeout, unit) 等待现有任务完成等了很久还没吃完 → 直接清场 → shutdownNow()(强制中断所有任务)executor.shutdown(); try { if (!executor.awaitTermination(60, TimeUnit.SECONDS)) { executor.shutdownNow(); } } catch (InterruptedException e) { executor.shutdownNow(); }十六、性能调优与监控16.1 关键指标活跃线程数:< 最大线程数的 80%队列长度:< 队列容量的 70%任务完成数:持续增长,增长停滞说明处理异常拒绝任务数:应为 0,频繁拒绝需优化配置16.2 监控工具JMX(ThreadPoolExecutor 暴露的 getActiveCount() 等)Micrometer(配合 Prometheus + Grafana)jstack 检查死锁十七、快速记忆口诀创建线程四种法:继承、实现、Callable、池。同步锁的选择:简单用 syn,灵活用 Re,开关用 volatile,计数用原子类。线程池七参数:核心最大时单队工拒。关闭优雅三步走:shutdown → 等待 → shutdownNow。死锁四条件:互斥请求不剥夺,循环等待一把锁。
2026年04月03日
8 阅读
0 评论
0 点赞
2026-04-03
Java 集合框架笔记(基于 Java 8)
Java 集合框架笔记(基于 Java 8)一、集合框架概述集合框架是一个统一的架构,用于存储和操作一组对象。根接口是 Iterable,核心接口包括 Collection(单列集合)和 Map(双列集合)。1.1 接口继承关系图Iterable<E> ↓ Collection<E> ├── List<E> │ ├── ArrayList<E> │ ├── LinkedList<E> │ ├── Vector<E> │ └── Stack<E> ├── Set<E> │ ├── HashSet<E> │ ├── LinkedHashSet<E> │ └── SortedSet<E> (接口) │ └── TreeSet<E> └── Queue<E> ├── PriorityQueue<E> └── Deque<E> └── ArrayDeque<E> Map<K,V> (独立体系) ├── HashMap<K,V> ├── LinkedHashMap<K,V> ├── Hashtable<K,V> └── SortedMap<K,V> (接口) └── TreeMap<K,V>1.2 核心接口简介接口描述特点Collection所有单列集合的根接口定义基本操作如添加、删除、大小等List有序集合,允许重复元素支持索引访问,元素顺序与插入顺序一致Set不允许重复元素无索引,通常基于哈希或排序实现Queue队列(先进先出)支持队首队尾操作,也有双端队列 DequeMap键值对映射键唯一,每个键映射到一个值二、Collection 接口所有集合类的根接口,定义了集合的基本操作。2.1 基本方法方法描述boolean add(E e)添加元素boolean remove(Object o)移除元素int size()返回元素个数boolean isEmpty()是否为空void clear()清空所有元素boolean contains(Object o)是否包含某元素Iterator<E> iterator()返回迭代器Object[] toArray()转换为数组三、List 接口有序集合,允许重复元素,支持通过索引操作。3.1 List 特有方法方法描述E get(int index)获取指定位置元素E set(int index, E element)替换指定位置元素void add(int index, E element)在指定位置插入元素E remove(int index)删除指定位置元素int indexOf(Object o)返回首次出现的索引int lastIndexOf(Object o)返回最后一次出现的索引List<E> subList(int from, int to)获取子列表3.2 ArrayList数据结构:动态数组 初始容量:10(默认构造函数) 扩容机制:新容量 = 旧容量 + 旧容量 >> 1(即 1.5 倍) 时间复杂度:get(index) / set(index, element):O(1)add(E e) 平均 O(1),最坏 O(n)(扩容)add(index, E e) / remove(index):O(n) 线程安全:否创建方式// 默认初始容量10 ArrayList<String> list1 = new ArrayList<>(); // 指定初始容量 ArrayList<String> list2 = new ArrayList<>(20); // 从其他集合创建(Java 9+ List.of 返回不可变List) List<String> source = List.of("Java", "Python", "C++"); ArrayList<String> list3 = new ArrayList<>(source); // 使用接口引用(推荐) List<String> list4 = new ArrayList<>();常用操作// 添加 list.add("Java"); // 末尾添加 list.add(1, "C++"); // 指定位置插入 list.addAll(List.of("Go", "Rust")); // 添加多个 // 访问与修改 String first = list.get(0); String last = list.get(list.size() - 1); list.set(1, "Python"); // 查找 int idx = list.indexOf("Java"); boolean has = list.contains("C++"); // 删除 list.remove(0); // 按索引 list.remove("Java"); // 按对象(删除第一个匹配) list.removeAll(List.of("Go", "Rust")); list.removeIf(s -> s.length() > 4); // Java 8+ list.clear(); // 缩容释放内存 list.trimToSize();线程安全解决方案Collections.synchronizedList(new ArrayList<>())CopyOnWriteArrayList(读多写少场景)手动 synchronized 同步注意事项遍历时(增强 for 或迭代器)不能直接调用 list.remove() 或 list.add(),应使用迭代器的 remove() 或 removeIf()。List.of() 返回的集合不可修改。3.3 LinkedList数据结构:双向链表 时间复杂度:get(index) / set(index, element):O(n)addFirst / addLast / removeFirst / removeLast:O(1)中间插入/删除:O(n)(需先定位)特点:实现 List 和 Deque 接口,可作列表、队列、双端队列、栈。创建与添加LinkedList<String> list = new LinkedList<>(); List<String> list2 = new LinkedList<>(); // 接口引用 list.add("Apple"); // 末尾 list.addFirst("Grape"); // 头部 list.addLast("Banana"); // 尾部 list.add(1, "Orange"); // 指定位置 list.offer("Cherry"); // 队列入队(末尾)访问操作String first = list.getFirst(); String last = list.getLast(); String element = list.get(2); // O(n) // 队列/双端队列查看 String head = list.peek(); String headFirst = list.peekFirst(); String headLast = list.peekLast(); int index = list.indexOf("Apple");删除操作list.removeFirst(); list.removeLast(); list.remove(1); list.remove("Apple"); // 队列出队 String polled = list.poll(); list.pollFirst(); list.pollLast();作为队列(FIFO)Queue<String> queue = new LinkedList<>(); queue.offer("客户1"); queue.offer("客户2"); String head = queue.peek(); // 查看队首 String customer = queue.poll(); // 出队作为栈(LIFO)Deque<String> stack = new LinkedList<>(); stack.push("方法A"); stack.push("方法B"); String top = stack.peek(); // 查看栈顶 String method = stack.pop(); // 出栈作为双端队列Deque<Integer> deque = new LinkedList<>(); deque.addFirst(2); deque.addLast(3); deque.addFirst(1); deque.addLast(4); // 结果 [1,2,3,4] deque.peekFirst(); deque.peekLast(); deque.removeFirst(); deque.removeLast();注意事项get(index) 在大列表上频繁使用会导致性能问题(O(n))。没有 trimToSize 方法(链表动态分配)。3.4 Vector 和 StackVector:早期动态数组,方法使用 synchronized 实现线程安全,性能较差,现已不推荐使用。Stack:继承 Vector,提供栈操作 push、pop、peek。同样不推荐,建议使用 Deque 实现(如 ArrayDeque)。四、Set 接口不允许重复元素(最多一个 null),无索引。底层依赖 equals() 和 hashCode() 判断重复。4.1 实现类对比实现类排序性能允许 null线程安全适用场景HashSet无序最高是否一般用途,高性能LinkedHashSet插入顺序中等是否需要保持插入顺序TreeSet自然排序/定制排序较低否否需要排序和范围查询4.2 使用示例Set<String> fruits = new HashSet<>(); fruits.add("苹果"); fruits.add("香蕉"); fruits.add("苹果"); // 重复,不会添加 System.out.println(fruits.size()); // 24.3 最佳实践默认使用 HashSet,需要有序用 LinkedHashSet,需要排序用 TreeSet。线程安全包装:Collections.synchronizedSet(new HashSet<>()) 或 ConcurrentHashMap.newKeySet()。自定义对象存入 Set 时必须重写 equals() 和 hashCode();存入 TreeSet 还需实现 Comparable 或提供 Comparator。合理设置 HashSet 初始容量(避免频繁扩容),负载因子默认 0.75。4.4 注意事项HashSet 迭代顺序不确定,不要依赖。TreeSet 不允许 null,会抛出 NullPointerException。修改已添加到 Set 中的对象可能导致数据丢失或重复。五、Queue 接口FIFO 队列,提供队首队尾操作。Deque 子接口支持双端操作。5.1 主要方法方法描述失败时offer(E e)入队返回 falsepoll()出队并移除返回 nullpeek()查看队首不移除返回 nulladd(E e)入队抛出异常remove()出队并移除抛出异常element()查看队首抛出异常推荐使用:offer、poll、peek(避免异常)。5.2 实现类对比实现类数据结构是否支持 null特点LinkedList双向链表是同时实现 List 和 Deque,插入删除效率高ArrayDeque动态数组否性能优于 LinkedList,一般队列首选PriorityQueue堆(数组)否按优先级排序(自然排序或 Comparator)5.3 使用示例Queue<String> queue = new ArrayDeque<>(); queue.offer("A"); queue.offer("B"); String head = queue.peek(); // A String out = queue.poll(); // A // 优先级队列 Queue<Integer> pq = new PriorityQueue<>(); pq.offer(5); pq.offer(1); pq.offer(3); System.out.println(pq.poll()); // 15.4 线程安全队列ArrayBlockingQueue(有界阻塞队列)LinkedBlockingQueue(可选有界)PriorityBlockingQueue(无界优先级队列)六、Map 接口键值对映射,键唯一,值可重复。不继承 Collection。6.1 常用方法方法描述V put(K key, V value)添加键值对,返回旧值V get(Object key)根据键获取值V remove(Object key)删除键值对boolean containsKey(Object key)是否包含键boolean containsValue(Object value)是否包含值int size()元素个数Set<K> keySet()返回所有键的 Set 视图Collection<V> values()返回所有值的 Collection 视图Set<Map.Entry<K,V>> entrySet()返回所有键值对的 Set 视图6.2 实现类对比实现类排序允许 null线程安全底层结构适用场景HashMap无序键/值均可否哈希表通用,最高性能LinkedHashMap插入顺序/访问顺序键/值均可否哈希表+双向链表需要保持顺序TreeMap自然排序/定制排序键不可为 null否红黑树需要排序和范围查询Hashtable无序键/值均不可是(全表锁)哈希表已淘汰,不推荐6.3 使用示例Map<String, Integer> map = new HashMap<>(); map.put("apple", 10); map.put("banana", 20); int value = map.get("apple"); // 10 map.remove("banana"); // 遍历 for (Map.Entry<String, Integer> entry : map.entrySet()) { System.out.println(entry.getKey() + "=" + entry.getValue()); }6.4 最佳实践声明时使用接口类型:Map<K,V> map = new HashMap<>()。作为键的类必须重写 equals() 和 hashCode(),最好使用不可变对象。合理设置 HashMap 初始容量,避免频繁 rehash。多线程环境使用 ConcurrentHashMap(不要使用 Hashtable)。七、迭代器(Iterator)7.1 Iterator 接口统一遍历集合的方式,支持安全删除。Iterator<String> it = list.iterator(); while (it.hasNext()) { String s = it.next(); if (condition) it.remove(); // 安全删除 }方法:boolean hasNext()E next()void remove()(可选)7.2 ListIterator 接口Iterator 的子接口,专门用于 List,支持双向遍历和修改。额外方法:boolean hasPrevious()E previous()int nextIndex() / int previousIndex()void set(E e) 修改当前元素void add(E e) 在当前位置插入7.3 增强 for 循环(for-each)语法糖,编译器自动转换为 Iterator 遍历。int[] numbers = {1,2,3,4,5}; for (int num : numbers) { System.out.println(num); } List<String> list = ...; for (String s : list) { // 不能修改 list 结构(添加/删除) }限制:无法获取当前索引遍历时不能修改集合结构(否则抛出 ConcurrentModificationException)不能修改基本类型数组的元素值(但可以修改引用类型对象的属性)八、并发集合(java.util.concurrent)8.1 ConcurrentHashMapJava 7 及之前:分段锁(Segment),默认 16 个段。Java 8+:Node 数组 + CAS + synchronized 对链表/红黑树头节点加锁,锁粒度更细。支持并发读(无锁)和并发写(部分加锁)。不允许 null 键/值。8.2 CopyOnWriteArrayList写时复制策略:修改时复制整个数组,修改后替换原数组。读操作无锁,性能极高。适用于读多写极少的场景(如配置列表、监听器列表)。缺点:写操作内存开销大,数据一致性弱(可能读到旧数据)。8.3 BlockingQueue 阻塞队列实现类特点ArrayBlockingQueue有界,基于数组,FIFO,支持公平/非公平锁LinkedBlockingQueue可选有界(默认 Integer.MAX_VALUE),基于链表PriorityBlockingQueue无界,按优先级排序核心阻塞方法:put(E e):队列满时阻塞take():队列空时阻塞典型应用:生产者-消费者模式。九、选择指南9.1 根据数据结构选择需求推荐实现需要索引访问List(ArrayList / LinkedList)不允许重复元素Set(HashSet / LinkedHashSet / TreeSet)键值对存储Map(HashMap / LinkedHashMap / TreeMap)FIFO 队列Queue(ArrayDeque 首选)频繁随机访问ArrayList频繁插入/删除(中间)LinkedList需要排序TreeSet / TreeMap保持插入顺序LinkedHashSet / LinkedHashMap线程安全(单元素操作)Vector / Hashtable(不推荐),推荐 ConcurrentHashMap、CopyOnWriteArrayList 等9.2 时间复杂度速查操作ArrayListLinkedListHashSetTreeSet添加(末尾)O(1)*O(1)O(1)**O(log n)插入(中间)O(n)O(n)--删除(中间)O(n)O(n)O(1)**O(log n)按值查找O(n)O(n)O(1)**O(log n)按索引查找O(1)O(n)--扩容时 O(n) ** 假设哈希函数良好十、常见问题与注意事项ConcurrentModificationException:在遍历集合时直接修改结构(增删),应使用迭代器的 remove() 或 removeIf()。HashSet / HashMap 中自定义对象:必须正确重写 equals() 和 hashCode(),否则无法去重或查找。TreeSet / TreeMap 中元素:必须实现 Comparable 或传入 Comparator,否则抛出 ClassCastException。ArrayList 删除元素后不会自动缩容:可调用 trimToSize() 释放多余内存。LinkedList 的 get(index):效率低(O(n)),避免在循环中频繁调用。Arrays.asList() 返回的 List:是固定大小的视图,不支持 add / remove,会抛出 UnsupportedOperationException。优先使用接口类型声明:List<String> list = new ArrayList<>();,便于更换实现。
2026年04月03日
10 阅读
0 评论
0 点赞
1
2
3
4