Java 关键字:synchronized详解

基本使用

在这里插入图片描述

Java中的synchronized关键字用于在多线程环境下确保数据同步。它可以用来修饰方法和代码块
当一个线程访问一个对象的synchronized方法或代码块时,其他线程将无法访问该对象的其他synchronized方法或代码块。这样可以确保在同一时间只有一个线程能够执行该代码块或方法,避免了多线程环境下的数据不一致问题,例如:

public class SynchronizedExample {
    private int count = 0;
    public synchronized void increment() {
        count++;
    }
}

在上面的代码中,increment()方法是一个synchronized方法。当多个线程访问这个方法时,只有一个线程能够执行该方法的代码,其他线程将被阻塞。
synchronized关键字也可以用来修饰代码块,如:

public void increment() {
    synchronized(this) {
        count++;
    }
}

在上面的代码中,synchronized关键字修饰的是一个代码块,并且锁对象是当前对象(this)
注意:synchronized关键字会导致线程上下文切换和资源竞争,所以在使用时要注意性能问题

源码解析

底层实现是通过 Java 虚拟机(JVM)的对象头和监视器锁机制实现的

具体来说,当一个线程访问一个对象的 synchronized 方法或代码块时,它会试图获取该对象的监视器锁。如果该锁未被其他线程占用,该线程将获得该锁并执行代码;如果该锁被其他线程占用,该线程将进入阻塞状态,等待获取该锁

synchronized 是Java中用于实现同步的关键字,它在底层通过监视器锁(Monitor)来实现。下面是synchronized的源码解析:

在Java中,每个对象都有一个与之关联的监视器锁,也称为内置锁或对象锁。当线程进入一个synchronized方法或代码块时,它会尝试获取该对象的监视器锁。如果锁没有被其他线程占用,则该线程获得锁并开始执行代码;如果锁已经被其他线程占用,则该线程将被阻塞,直到锁被释放。

在Java虚拟机中,每个对象头中都包含一部分用于实现synchronized的相关信息。这些信息包括:

  • mark word:用于存储对象的标记信息,包括锁的状态。
  • Klass pointer:指向对象的类元数据,包括synchronized的相关信息。
  • monitor:与对象关联的监视器,它记录了当前占用锁的线程、等待锁的线程队列等。

当一个线程尝试获取一个对象的锁时,虚拟机会检查对象头中的标记信息。如果对象的锁状态为无锁状态,即未被其他线程占用,则该线程可以获取锁,并将标记信息设置为锁定状态。如果对象的锁状态为已锁定,并且当前线程是锁的所有者,则该线程可以继续执行代码。如果对象的锁状态为已锁定,并且当前线程不是锁的所有者,则该线程将被放入等待队列中,进入阻塞状态。

当持有锁的线程执行完synchronized方法或代码块后,它会释放锁,即将对象头中的锁状态置为无锁状态,并唤醒等待队列中的一个线程,使其获取锁并继续执行。

需要注意的是,synchronized关键字可以修饰方法和代码块。在方法上修饰的synchronized表示对整个方法进行同步,而在代码块上修饰的synchronized表示对该代码块进行同步,使用的锁对象通常是方法所属对象或指定的对象。

总结起来,通过监视器锁的机制,Java的synchronized能够保证同一时刻只有一个线程访问同步代码块或方法,避免了多线程的数据竞争和并发问题。

这里给出一份简化的 synchronized 关键字的源码:

public void synchronized method() {
    // 加锁
    Monitor.enter(this);
    try {
        // 同步代码块
    } finally {
        // 释放锁
        Monitor.exit(this);
    }
}

在这份代码中,方法通过调用 Monitor.enter 方法获取当前对象的监视器锁,并在 finally 块中调用 Monitor.exit 方法释放该锁。因此,在 synchronized 方法内部的代码可以保证在任意时刻只有一个线程可以访问

常见面试题

  • synchronized 方法和 synchronized 块的区别是什么?
    作用范围:synchronized 方法将整个方法体作为同步区块,而 synchronized 块可以将任意代码块作为同步区块
    锁的对象:synchronized 方法锁定的是整个对象,而 synchronized 块锁定的是在括号内指定的对象
    可控性:synchronized 方法的同步粒度比较大,不够灵活;而 synchronized 块可以更灵活地控制同步代码块的大小
    综上所述,在确定同步粒度时,通常使用 synchronized 块比使用 synchronized 方法更灵活,但是如果整个方法都需要同步,使用 synchronized 方法会更加简单易懂
  • 什么情况下可以使用 synchronized 关键字?
    synchronized 关键字可以用于在多线程环境下保证方法或代码块的原子性。具体来说,如果一个线程正在执行同步方法或代码块,则其他线程将无法访问该方法或代码块
    常见情况包括:
    当多个线程访问共享资源时,可以使用 synchronized 关键字保证线程的安全
    在访问共享变量时,需要对其进行同步控制
  • 在线程通信中,可以使用 synchronized 关键字保证线程之间的同步通信
    synchronized 关键字的性能开销如何?
    synchronized 关键字的使用会带来一些性能开销,因为它需要在多个线程之间进行同步。当线程访问同步代码块时,它必须获得锁,这会增加额外的开销。如果同步代码块执行时间过长,其他线程将一直等待,进而降低程序的性能。
    因此,应该尽量避免在高并发情况下使用 synchronized,或者使用其他的并发控制机制,如 java.util.concurrent 包中的锁和原子操作类等。
  • synchronized 关键字如何实现可重入?
    “可重入” 指的是同一线程可以多次获取同一个锁。例如,当线程 A 进入一个同步块时,如果它再次试图进入该块,则可以再次获取锁,而不会发生死锁
    在 Java 中,synchronized 关键字可以实现可重入,原因如下:
    synchronized 关键字使用对象监视器锁来实现同步。
    对象监视器锁是基于线程的,并且每个线程有一个独立的计数器,用于跟踪它在当前对象上获取的锁的数量。
    当线程试图获取锁时,如果它已经拥有该锁,则计数器将递增。
    当线程退出同步块时,计数器将递减。
    只有当计数器为零时,该线程才会释放锁。
    因此,如果一个线程在同一对象上多次进入同步块,它将多次获得该锁,并在退出该块时多次释放该锁。因此,synchronized 关键字是可重入的。
  • synchronized 关键字与 lock 机制的比较?
    synchronized 关键字和 Lock 机制都是用来保证线程同步的方法。但是它们有一些明显的差异:

灵活性:Lock 机制比 synchronized 关键字更灵活,因为它提供了更多的锁定操作,例如可以实现公平锁和非公平锁,还可以实现读写锁。
可中断性:Lock 机制可以中断一个线程的等待,而 synchronized 关键字不能。
可重入性:synchronized 关键字是自动可重入的,而 Lock 机制必须手动实现。
性能:如果比较的是相同的锁定操作,synchronized 关键字通常比 Lock 机制更快,因为它是内置的。
总体而言,在简单的同步情况下,synchronized 关键字更方便,但是在需要更多灵活性的情况下,Lock 机制可能是一个更好的选择。

好书推荐

深入理解 Java 高并发编程

修炼高并发内功,面试求职常备。计算机、系统、软件多层次讲透CPU并发、内核并发、Java并发、线程池、JVM原理。

在这里插入图片描述
购书链接:点击传送门

  • 内容简介

《深入理解Java高并发编程》致力于介绍Java高并发编程方面的知识。由于多线程处理涉及的知识内容十分丰富,因此介绍时必须从Java层面的讲解一直深入到底层的知识讲解。为了帮助读者轻松阅读本书并掌握其中知识,本书做了大量基础知识的铺垫。在第1篇基础知识储备中,主要介绍计算机原理、并发基础、常见语言的线程实现、Java并发入门、JUC之Java线程池、JUC之同步结构、Java NIO详解等内容。在第2篇深入Java并发原理中,详细介绍了JUC包中所有使用的原子类的原理与源码实现;非常关键且容易出错的volatile关键字的原理,从Java、JVM、C、汇编、CPU层面对其进行详细讲解;synchronized在JVM中获取锁和释放锁的流程;JUC包的核心结构——AQS的原理与源码实现,通过逐方法、逐行的解释,帮助读者彻底掌握AQS中提供的获取锁、释放锁、条件变量等操作的实现与原理。最后,详细介绍了JVM中JNI的实现原理,将Java Thread对象中的所有方法在JVM层面的实现流程进行了详细描述,以帮助读者在使用这些方法时,知道底层发生了什么,以及发生异常时如何从容解决问题。

  • 作者简介

黄俊,专注于研究Java语言, Hotspot, Linux内核,C语言与汇编,架构设计,多线程并发处理,专注于研究高效学习方式。曾就职于美团、阿里,前新东方业务架构师。

在这里插入图片描述

  • 105
    点赞
  • 159
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 91
    评论
### 回答1: `synchronized` 是 Java 语言中用于实现线程同步的关键字。它可以被用来修饰代码块或者方法。 当一个线程执行到被 `synchronized` 修饰的代码块或者方法时,它会尝试获取该代码块或者方法所对应的监视器锁(也称为对象锁)。如果该锁没有被其他线程持有,则该线程获取锁,进入临界区执行代码。如果锁已经被其他线程持有,该线程会被阻塞,直到它能够获取锁为止。 以下是 `synchronized` 的几种用法: 1. 修饰代码块: ``` synchronized (obj) { // 这里的代码需要线程同步 } ``` `obj` 可以是任意一个对象。当线程进入 synchronized 块时,它会尝试获取 `obj` 对象锁。 2. 修饰方法: ``` public synchronized void method() { // 这里的代码需要线程同步 } ``` 当线程调用该方法时,它会尝试获取该方法所对应对象的锁。如果该对象的锁已经被其他线程持有,则该线程会被阻塞,直到它能够获取锁为止。 3. 修饰静态方法: ``` public static synchronized void staticMethod() { // 这里的代码需要线程同步 } ``` 当线程调用该静态方法时,它会尝试获取该类的 Class 对象的锁。因为 Class 对象是唯一的,所以多个线程之间需要获取该类的 Class 对象的锁才能执行该静态方法。 4. 修饰类: ``` public class MyClass { public void method() { synchronized (MyClass.class) { // 这里的代码需要线程同步 } } } ``` 当线程进入 synchronized 块时,它会尝试获取该类的 Class 对象的锁。因为 Class 对象是唯一的,所以多个线程之间需要获取该类的 Class 对象的锁才能执行该代码块。 需要注意的是,`synchronized` 关键字虽然可以保证线程安全,但是它的性能比较低。因为当一个线程持有了一个对象的锁时,其他线程就无法同时访问该对象,这样就可能会导致线程之间的竞争和阻塞,影响程序的执行效率。 ### 回答2: synchronizedJava中用于实现线程同步的关键字。当多个线程访问共享资源时,如果不对共享资源进行适当的同步处理,可能会产生线程安全问题,例如数据不一致、死锁等。synchronized关键字可以帮助我们解决这些问题。 synchronized可以用在方法、代码块或静态方法中。当一个线程访问synchronized方法时,其他线程必须等待该线程执行完毕才能继续执行。synchronized方法会锁住整个对象实例,保证了同一时刻只能有一个线程执行该方法。 synchronized也可以用于代码块,即synchronized关键字后面跟着一对大括号。它可以指定一个对象作为锁,当线程进入synchronized代码块时,会尝试获取该对象的锁。如果锁已经被其他线程占用,则当前线程就会被阻塞,直到获取锁为止。这样可以保证在同一时刻只有一个线程执行临界区的代码,从而保证了线程安全。 对于静态方法,synchronized关键字会锁住整个类,同一时刻只有一个线程可以执行静态方法。这是因为静态方法是属于类而不是对象,所以它的锁就是类级别的锁。 除了上述用法,synchronized还可以用于修饰代码块中的对象引用。这种方式可以实现对不同对象的锁定,不同的对象之间互不影响。 需要注意的是,synchronized虽然可以保证线程安全,但也可能造成性能问题。由于synchronized会引起线程的阻塞和唤醒,所以在多线程访问频繁的情况下,使用synchronized可能会降低程序的性能。因此,可以考虑使用其他的并发控制方式,比如Lock接口和并发集合类等。 总而言之,synchronized关键字Java中用于实现线程安全的重要机制。通过锁定对象实例、类或对象引用,synchronized确保了同一时刻只有一个线程可以执行代码,从而解决了线程安全问题。 ### 回答3: synchronizedJava中的一个关键字,用于实现线程安全。当一个方法或者代码块被synchronized修饰时,同一时间只能有一个线程访问该方法或者代码块,其他线程需要等待,直到当前线程执行完毕。以下是synchronized关键字的详细解释: 1. 对象锁:synchronized可以用于方法和代码块两种方式。当修饰方法时,锁定的是当前对象实例,只有获取到当前对象实例的线程才能访问该方法。当修饰代码块时,需要指定一个对象作为锁,只有获取到该对象锁的线程才能执行代码块中的内容。 2. 类锁:使用synchronized修饰一个static方法或者使用synchronized修饰一个代码块,并且锁定的对象是该类的Class对象。只有获取到该类的Class对象的线程才能访问该方法或者代码块。类锁是一个全局锁,影响该类的所有实例对象。 3. 内部锁:当使用synchronized修饰一个类的实例方法或者代码块时,锁定的是该实例对象。每个实例对象都有自己的锁,即内部锁,只有获取到该实例对象的线程才能执行该方法或者代码块。 4. 对象锁的可重入性:在一个线程执行一个 synchronized 方法/代码块的过程中,当遇到同一个对象的其他 synchronized 方法/代码块时,该线程可以直接进入,即可重入。保证了同一个线程在访问某个对象的 synchronized 方法/代码块时,不会导致死锁。 5. 锁的释放:当一个线程执行完一个 synchronized 方法/代码块时,会释放该方法/代码块所获取的锁,让其他线程有机会获取到锁并执行。 通过使用synchronized关键字,可以有效保证多线程程序的线程安全性,避免了多个线程同时访问和修改共享数据带来的冲突和错误。但是在使用synchronized时需要注意避免过多的同步操作,以免影响程序性能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 91
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小尘要自信

不要打赏,我不配。

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值