Java并发编程质量关键在于规避共享状态、职责清晰和精准同步;ConcurrentHashMap优于手动锁HashMap;避免synchronized块中I/O;慎用ThreadLocal,优先框架上下文传递。
Java并发编程质量不取决于用了多少高级工具,而在于是否规避了共享状态、是否让线程职责清晰、是否在关键路径上做了必要同步且不多余。
ConcurrentHashMap 替代 HashMap + synchronized
手动加锁保护 HashMap 是常见但危险的做法:容易漏锁、死锁,且读操作也被阻塞。而 ConcurrentHashMap 的分段锁(JDK 8+ 改为 CAS + synchronized 细粒度节点锁)天然支持高并发读写。
put()、computeIfAbsent(),无需额外同步ConcurrentModificationException,但可能看不到最新写入 —— 这是设计取舍,不是 bugsize():它返回估算值;如需精确计数,改用 mappingCount()
synchronized 块里做 I/O 或远程调用同步块本质是串行化临界区,一旦里面包含耗时操作(如数据库查询、HTTP 请求),会严重拖慢其他线程,甚至引发线程池饥饿。
synchronized(this):暴露
private final Object lock = new Object();
CompletableFuture 替代裸 Thread 或 ExecutorService.submit(Runnable)
手动管理线程生命周期和结果传递极易出错:忘记 join()、异常丢失、回调嵌套地狱、线程复用混乱。
supplyAsync() + thenApply() 可自然链式编排,异常由 exceptionally() 或 handle() 捕获ForkJoinPool.commonPool(),但 IO 密集型任务建议显式传入专用线程池,如 Executors.newCachedThreadPool()
get():会阻塞当前线程;优先用 thenAccept() 等非阻塞方式消费结果ThreadLocal 传参,尤其在线程池中ThreadLocal 在线程复用场景下极易造成内存泄漏或上下文污染 —— 上一个请求设的值,下一个请求没清理就直接读到了。
threadLocal.remove(),不能只靠 set(null)
RequestContextHolder、SLF4J 的 MDC 已内置清理机制ThreadLocal 时,应重写 initialValue() 而非构造时赋值,避免子线程继承父线程值并发问题往往在压测或上线后才暴露,而修复成本远高于设计阶段的约束。最值得花时间的,不是学新 API,而是厘清哪些变量真需要共享、哪些状态本该属于单次请求生命周期。