在Java中,对一个对象加锁,锁的是对象的监视器(Monitor)。在Java中,每个对象都与一个监视器相关联,当它被用作同步锁时,实质上是利用了这个监视器来控制对共享资源的访问。对象的监视器通过内部的一个锁(称为 intrinsic lock 或 monitor lock)来实现线程之间的同步。当一个线程试图访问一个同步的代码块或方法时,它必须首先获得该对象的锁。
当线程获得锁之后,其他试图获得这个锁的线程将被阻塞,直到持有锁的线程释放锁。这种机制确保了同一时间只有一个线程可以进入由锁保护的代码区域,从而保护了共享资源不会因并发访问而出现数据一致性问题。
一、对象的监视器(Monitor)
对象监视器是同步机制的基础,实现了线程的互斥和通信。当一个线程想要执行同步代码块或方法时:
- 如果对象锁没有被其他线程持有,当前线程锁住该对象并进入代码块继续执行;
- 如果对象锁被其他线程持有,当前线程进入等待状态,直到锁被释放;
二、Synchronized关键字
Java 中实现锁的主要机制是通过 synchronized
关键字。synchronized
可以用来修饰一个方法或者一个代码块。
方法同步
当声明一个方法为 synchronized
的时候,线程访问这个方法就需要获得对象的内部锁。这对于对象的实例方法而言,锁就是对象实例本身;对于静态方法而言,锁就是这个类的Class对象。
代码块同步
synchronized
同步代码块可以对任意对象进行加锁,加锁的对象称作同步监视器。同步代码块的语法是:
synchronized(锁对象) {
// 需要同步的代码
}
三、锁的状态
Java 对象锁具有不同的状态:无锁状态、偏向锁状态、轻量级锁状态和重量级锁状态,它们随着竞争情况逐步升级。
偏向锁
这是锁的第一级状态,假设线程间是协作关系而没有竞争,当一个线程访问同步块并获取锁时,其所在的对象就进入了偏向模式。当这个线程再次请求时,就无需再进行任何同步操作。
轻量级锁
当锁对象被不同线程所访问,偏向锁就会升级为轻量级锁。轻量级锁通过CAS操作和自旋来提高性能,当锁竞争不激烈时,使用轻量级锁可以避免线程的阻塞和唤醒,从而提高效率。
重量级锁
当线程竞争激烈且自旋失败时,轻量级锁会升级为重量级锁。重量级锁会导致其他申请锁的线程进入阻塞状态,等待操作系统的调度。
四、锁优化
为了减少锁带来的性能开销,Java虚拟机(JVM)通过几种机制对锁进行了优化:
锁粗化 – 当检测到一系列的连锁操作都对同一个对象加锁,JVM 会将锁的范围扩展(粗化)到整个操作序列,避免多次加锁解锁。
锁消除 – JVM 通过逃逸分析技术判断,无需同步的代码块的锁可以被消除。
自旋锁 – 而不是立即挂起线程,JVM 会让线程执行几个空循环(称为“自旋”),以期待锁很快被释放。
适应性自旋锁 – JVM 会根据锁对象以往的历史数据来决定自旋的次数。
五、锁的底层实现
JVM 在底层实现锁时,主要涉及到两种机制:
对象头 Mark Word – 每个 Java 对象头都包含锁相关的信息,包括锁状态标志、锁的持有者线程 ID 等。
监视器对象 Monitor – 在 JVM 的 C++ 实现中,每个对象关联一个 Monitor 对象,它包含两个基本的数据结构:WAItSet 和 EntryList,分别用于存放调用了 wait 方法而进入等待状态的线程,以及尝试获取锁但未成功的线程。
在进行对象加锁时,Java 虚拟机需要进行一系列的操作来确保线程安全。不断优化的锁机制是Java多线程编程性能优化的关键所在。
相关问答FAQs:
1. 为什么要对Java对象加锁?
在多线程环境下,当多个线程对同一个对象进行并发操作时,可能会导致不可预期的结果。这是因为并发操作可能会引发线程安全问题,如数据竞争、死锁等。因此,加锁是一种常用的机制,用于保护并发操作中的共享资源,以确保线程安全。
2. 加锁时锁住的是哪些内容?
在Java中,对一个对象加锁时,实际上锁住的是该对象的监视器锁(也称为内部锁或互斥锁)。每个对象都有一个相关联的监视器锁,该锁是用于控制线程对该对象的访问的。当一个线程获得该锁时,其他线程将被阻塞,直到该线程释放锁。
3. 在Java中如何对对象加锁?
在Java中,可以使用synchronized关键字或Lock接口来对对象进行加锁。synchronized关键字可以应用于方法、代码块或静态方法,而Lock接口则提供了更灵活的加锁机制,如可重入锁、读写锁等。
当使用synchronized关键字对方法或代码块进行加锁时,锁住的是当前对象。例如,对于一个方法声明为synchronized,该方法在执行期间会获得当前对象(this)的锁定。
当使用Lock接口进行加锁时,需要手动调用lock()方法来获得锁,并在执行完任务后调用unlock()方法释放锁。Lock接口的实现类提供了更高级的加锁特性,如可定时锁、公平锁等,以满足不同场景的需求。