Java中synchronized 关键字

synchronized 是 Java 中用于实现线程同步的关键字,它提供了一种简单有效的方式来控制多线程对共享资源的访问,确保线程安全。

基本概念

synchronized 关键字可以实现:

  1. 互斥性:同一时刻只有一个线程可以执行被同步的代码

  2. 可见性:线程对共享变量的修改对其他线程立即可见

使用方式

synchronized 有三种主要使用方式:

1. 同步实例方法

java 复制代码
public synchronized void method() {
    // 同步代码
}
  • 锁对象是当前实例对象(this)

  • 同一实例的同步方法在同一时刻只能被一个线程访问

2. 同步静态方法

java 复制代码
public static synchronized void staticMethod() {
    // 同步代码
}
  • 锁对象是当前类的 Class 对象(如 MyClass.class)

  • 所有实例的同步静态方法在同一时刻只能被一个线程访问

3. 同步代码块

java 复制代码
public void method() {
    // 非同步代码
    
    synchronized(lockObject) {
        // 同步代码
    }
    
    // 非同步代码
}
  • 可以灵活指定锁对象

  • 缩小同步范围,提高并发性能

实现原理

synchronized 是基于 JVM 内置锁(Monitor)实现的:

  1. 进入同步代码:线程尝试获取对象的 Monitor

    • 成功则持有 Monitor,计数器 +1

    • 失败则进入阻塞队列等待

  2. 退出同步代码:释放 Monitor,计数器 -1

    • 计数器为 0 时完全释放锁

    • 唤醒等待队列中的线程

锁的升级过程(JDK 1.6+优化)

为了减少锁操作的开销,JVM 实现了锁升级机制:

  1. 无锁状态:对象刚创建时

  2. 偏向锁:适用于只有一个线程访问的场景

    • 记录线程 ID,减少同步开销
  3. 轻量级锁(自旋锁):适用于短时间等待的场景

    • 线程通过 CAS 自旋尝试获取锁
  4. 重量级锁:竞争激烈时升级

    • 线程进入阻塞状态,由操作系统管理

注意事项

  1. 锁对象选择

    • 不要使用 String 常量等可能被共享的对象作为锁

    • 建议使用专门创建的私有 final 对象作为锁

  2. 死锁风险

    • 避免嵌套获取多个锁

    • 如果需要获取多个锁,应保持一致的获取顺序

  3. 性能考虑

    • 尽量减小同步代码块的范围

    • 避免在同步代码块中执行耗时操作

与 Lock 接口的比较

特性 synchronized Lock (如 ReentrantLock)
实现方式 JVM 内置 Java 代码实现
锁获取方式 自动获取释放 需要手动 lock/unlock
尝试非阻塞获取锁 不支持 tryLock() 支持
公平锁 非公平 可配置公平/非公平
条件变量 单一 可创建多个 Condition
性能 JDK 优化后接近 更灵活但稍复杂

synchronized 是 Java 中最基本的线程同步机制,虽然功能不如 Lock 接口丰富,但在大多数情况下已经足够,并且随着 JVM 的优化,其性能已经非常优秀。

相关推荐
不会C语言的男孩16 小时前
C++ Primer 第2章:变量和基本类型
开发语言·c++
小江的记录本16 小时前
【JVM虚拟机】堆内存分代模型:年轻代(Eden+Survivor)、老年代、元空间Metaspace(附《思维导图》+《面试高频考点清单》)
java·前端·jvm·后端·python·spring·面试
在繁华处16 小时前
Java从零到熟练(三):流程控制
java·开发语言·python
唐青枫17 小时前
Java Optional 实战指南:优雅处理空值与链式转换
java
一起学开源17 小时前
一文读懂 ReAct 范式:让 AI Agent 真正学会“思考+行动“
java·javascript·react.js·ecmascript·react·alibaba·智能体开发
云泽80817 小时前
C++ 可调用对象通关指南:深度解析 Lambda 表达式、function 包装器与 bind 绑定器
开发语言·c++·算法
逍遥德18 小时前
MQTT教程详解-04.SpringBoot集成MQTT(告别手动控制)
java·spring boot·物联网·中间件·iot·iotdb
语戚18 小时前
力扣 3161. 块放置查询:线段树解法(Java 实现)
java·算法·leetcode·面试·线段树·力扣·
我命由我1234519 小时前
Android 开发问题:MlKitException: An internal error occurred during initialization.
android·java·java-ee·android jetpack·android-studio·androidx·android runtime
星恒随风19 小时前
Python 基础语法详解(一):从表达式、变量到数据类型
开发语言·笔记·python·学习