Java 中 synchronized 与 AtomicInteger 的区别

简介: 在Java多线程编程中,`synchronized`和`AtomicInteger`均用于实现线程安全,但原理与适用场景不同。`synchronized`是基于对象锁的同步机制,适用于复杂逻辑和多变量同步,如银行转账;而`AtomicInteger`采用CAS算法,适合单一变量的原子操作,例如计数器更新。二者各有优劣,应根据具体需求选择使用。

在 Java 的多线程编程中,synchronizedAtomicInteger 都是用来实现线程安全的,但它们有着本质的区别。

一、实现原理不同

synchronized 是 Java 内置的同步机制,它通过对象的锁来实现对共享资源的独占式访问。当一个线程获取到对象的锁后,其他线程必须等待,直到锁被释放。它主要依赖于 JVM 的对象头中的锁标志位来进行实现。这种同步机制相对重量级,特别是在竞争激烈的场景下,线程频繁地获取和释放锁,会导致性能开销较大。

AtomicInteger 是基于 CAS(Compare-And-Swap)算法实现的原子类。CAS 是一种无锁算法,它通过硬件指令级别的原子操作来确保变量的更新是一个不可分割的整体,从而使多个线程能够并发地安全访问共享变量,而不需要依赖传统的锁机制。这使得 AtomicInteger 在性能上通常优于 synchronized。

二、使用场景不同

  1. 单一变量的原子操作AtomicInteger 适用于对单个变量进行原子操作的场景,比如对一个计数器进行增减操作。例如,维护一个在线人数的计数器,多个线程同时在线或离线时,可以使用 AtomicInteger 来安全地更新人数。
AtomicInteger onlineCount = new AtomicInteger(0);
// 线程在线
onlineCount.incrementAndGet();
// 线程离线
onlineCount.decrementAndGet();
  1. 复杂同步逻辑synchronized 更适合用于更复杂的同步场景,比如需要对多个变量进行同步操作,或者需要对一段代码进行同步控制。例如,在一个银行转账的场景中,涉及到两个账户的余额更新,此时可以使用 synchronized 来确保整个转账过程的原子性。
public class BankAccount {
    private double balance;
    public synchronized void transfer(BankAccount target, double amount) {
        if (this.balance >= amount) {
            this.balance -= amount;
            target.balance += amount;
        }
    }
}

三、功能特性不同

  1. 可重入性synchronized 是可重入的,这意味着一个线程在获取到锁后,可以再次获取相同的锁,而不会发生死锁。这在一定程度上提高了代码的可读性和可维护性。
public class ReentrantExample {
    private Object lock = new Object();
    public void outerMethod() {
        synchronized (lock) {
            innerMethod();
        }
    }
    public void innerMethod() {
        synchronized (lock) {
            // 执行一些操作
        }
    }
}
  1. 公平性synchronized 不是公平锁。公平锁是指线程获取锁的顺序是按照线程申请锁的顺序来获取的,而非公平锁则可能会让后申请的线程先获取到锁。AtomicInteger 并不涉及公平性的问题,它主要是提供一种高效的原子操作方式。
  2. 锁的粒度控制synchronized 在一定程度上可以灵活控制锁的粒度。可以通过合理地定义同步代码块或者同步方法来控制锁的范围,从而在保证线程安全的同时,提高程序的并发性能。
public class LockGranularityExample {
    private final Object lock1 = new Object();
    private final Object lock2 = new Object();
    public void method1() {
        synchronized (lock1) {
            // 操作共享资源 1
        }
    }
    public void method2() {
        synchronized (lock2) {
            // 操作共享资源 2
        }
    }
}

AtomicInteger 的锁粒度是固定的,它只针对单个变量进行原子操作,无法像 synchronized 那样灵活地控制锁的粒度。

在选择使用 synchronized 还是 AtomicInteger 时,需要根据具体的业务场景和性能需求来综合考虑。如果只是对单个变量进行简单的原子操作,AtomicInteger 通常是一个更好的选择;如果涉及到复杂的同步逻辑或者需要对多个变量进行同步,那么 synchronized 可能会更合适。

目录
相关文章
|
16天前
|
安全 Java API
Java SE 与 Java EE 区别解析及应用场景对比
在Java编程世界中,Java SE(Java Standard Edition)和Java EE(Java Enterprise Edition)是两个重要的平台版本,它们各自有着独特的定位和应用场景。理解它们之间的差异,对于开发者选择合适的技术栈进行项目开发至关重要。
70 1
|
2月前
|
Java 测试技术
Java浮点类型详解:使用与区别
Java中的浮点类型主要包括float和double,它们在内存占用、精度范围和使用场景上有显著差异。float占用4字节,提供约6-7位有效数字;double占用8字节,提供约15-16位有效数字。float适合内存敏感或精度要求不高的场景,而double精度更高,是Java默认的浮点类型,推荐在大多数情况下使用。两者都存在精度限制,不能用于需要精确计算的金融领域。比较浮点数时应使用误差范围或BigDecimal类。科学计算和工程计算通常使用double,而金融计算应使用BigDecimal。
872 102
|
3月前
|
存储 缓存 人工智能
Java int和Integer的区别
本文介绍了Java中int与Integer的区别及==与equals的比较机制。Integer是int的包装类,支持null值。使用==比较时,int直接比较数值,而Integer比较对象地址;在-128至127范围内的Integer值可缓存,超出该范围或使用new创建时则返回不同对象。equals方法则始终比较实际数值。
116 0
|
4月前
|
算法 Java 数据库连接
Java 与 C++ 区别深入剖析及应用实例详解
本文深入剖析了Java和C++两种编程语言的区别,从编译与执行机制、面向对象特性、数据类型与变量、内存管理、异常处理等方面进行对比,并结合游戏开发、企业级应用开发、操作系统与嵌入式开发等实际场景分析其特点。Java以跨平台性强、自动内存管理著称,适合企业级应用;C++则因高性能和对硬件的直接访问能力,在游戏引擎和嵌入式系统中占据优势。开发者可根据项目需求选择合适语言,提升开发效率与软件质量。附面试资料链接:[点此获取](https://pan.quark.cn/s/4459235fee85)。
312 0
|
4月前
|
存储 Java C语言
Java List 复制:浅拷贝与深拷贝方法及区别
我是小假 期待与你的下一次相遇 ~
336 1
|
5月前
|
Java
Java 中 Exception 和 Error 的区别
在 Java 中,`Exception` 和 `Error` 都是 `Throwable` 的子类,用于表示程序运行时的异常情况。`Exception` 表示可被捕获和处理的异常,分为受检异常(Checked)和非受检异常(Unchecked),通常用于程序级别的错误处理。而 `Error` 表示严重的系统级问题,如内存不足或 JVM 错误,一般不建议捕获和处理。编写程序时应重点关注 `Exception` 的处理,确保程序稳定性。
116 0
|
6月前
|
Java 编译器 程序员
java中重载和多态的区别
本文详细解析了面向对象编程中多态与重载的概念及其关系。多态是OOP的核心,分为编译时多态(静态多态)和运行时多态(动态多态)。编译时多态主要通过方法重载和运算符重载实现,如Java中的同名方法因参数不同而区分;运行时多态则依赖继承和方法重写,通过父类引用调用子类方法实现。重载是多态的一种形式,专注于方法签名的多样性,提升代码可读性。两者结合增强了程序灵活性与扩展性,帮助开发者更好地实现代码复用。
212 0
|
7月前
|
存储 缓存 人工智能
【原理】【Java并发】【synchronized】适合中学者体质的synchronized原理
本文深入解析了Java中`synchronized`关键字的底层原理,从代码块与方法修饰的区别到锁升级机制,内容详尽。通过`monitorenter`和`monitorexit`指令,阐述了`synchronized`实现原子性、有序性和可见性的原理。同时,详细分析了锁升级流程:无锁 → 偏向锁 → 轻量级锁 → 重量级锁,结合对象头`MarkWord`的变化,揭示JVM优化锁性能的策略。此外,还探讨了Monitor的内部结构及线程竞争锁的过程,并介绍了锁消除与锁粗化等优化手段。最后,结合实际案例,帮助读者全面理解`synchronized`在并发编程中的作用与细节。
463 8
【原理】【Java并发】【synchronized】适合中学者体质的synchronized原理
|
7月前
|
缓存 安全 Java
【Java并发】【synchronized】适合初学者体质入门的synchronized
欢迎来到我的Java线程同步入门指南!我不是外包员工,梦想是写高端CRUD。2025年我正在沉淀中,博客更新速度加快,欢迎点赞、收藏、关注。 本文介绍Java中的`synchronized`关键字,适合初学者。`synchronized`用于确保多个线程访问共享资源时不会发生冲突,避免竞态条件、保证内存可见性、防止原子性破坏及协调多线程有序访问。
177 8
【Java并发】【synchronized】适合初学者体质入门的synchronized
|
9月前
|
Java 程序员 调度
Java 高级面试技巧:yield() 与 sleep() 方法的使用场景和区别
本文详细解析了 Java 中 `Thread` 类的 `yield()` 和 `sleep()` 方法,解释了它们的作用、区别及为什么是静态方法。`yield()` 让当前线程释放 CPU 时间片,给其他同等优先级线程运行机会,但不保证暂停;`sleep()` 则让线程进入休眠状态,指定时间后继续执行。两者都是静态方法,因为它们影响线程调度机制而非单一线程行为。这些知识点在面试中常被提及,掌握它们有助于更好地应对多线程编程问题。
347 9