java多线程的锁都是基于对象的,java中的每一个对象都可以作为一个锁。
类锁其实也是对象锁。
1.synchronized关键字
(1)synchronized在实例方法上,锁为当前实例
private synchronized void testMethod(String flag) throws InterruptedException {
synchronized (lock) {
for (int i = 0; i < 10; i++) {
lock.notify();
lock.wait();
System.out.println(flag + ":进入同步方法--" + i);
}
}
}
(2)关键字在静态方法上,锁为当前Class对象
//关键字在静态方法上,锁为当前Class对象
public static synchronized void staticLock(){
//
}
(3)锁在代码块上,锁为括号里面的对象
public void run() {
synchronized (lock) {
for (int i = 0; i < 5; i++) {
System.out.println("b" + ":进入同步方法--" + i);
lock.notify();
lock.wait();
}
lock.notify();
}
}
2.临界区
表示一块代码区域,它同一时刻只能由一个代码执行。
3.使用synchronized的流程细节区分锁种类
根据多个线程使用synchronized竞争同步资源的流程细节可以将锁分为以下几种状态:
- 无锁状态:不锁住资源,多个线程同时对资源进行修改,但是只有一个能够修改成功,其他线程会重试,直到改成功。
- 偏向锁状态:统一段代码被统一鲜橙多次访问,会自动获取锁。
- 轻量级锁状态:当锁是偏向锁的时候,被另外的线程访问,该偏向锁就会升级为轻量级锁。其他线程会以自旋的形式尝试获取锁,不会阻塞,从而提高性能。
- 重量级锁状态:当锁是轻量级锁时,又有一个线程尝试获取锁时,轻量级锁升级为重量级锁。
4.synchroized实现同步的原理
(1) 对象头
Hotspot虚拟机的对象头主要包括:Mark Word(标记字段,包括)和Klass Pointer(类型指针)
Mark Word:默认存储对象的hashCode和锁信息。
Klass Pointer:对象指向它的类元数据的指针,虚拟机通过这个指针确定这个对象是哪个类的实例。
(2)Monitor
可以理解为一种同步工具或同步机制。
5.锁的升级流程
(1)每一个线程在准备获取共享资源时,先检查Mark word放的是不是自己的ThreadId;如果是,表示当前线程处于“偏向锁”状态。
(2)如果Mark Word中不是自己的TreadId,用CAS执行切换Mark Word中的ThreadId 为新的线程Id,如果之前的线程不存在了,则锁为偏向锁;如果之前的线程存在,则暂停之前的线程,设置偏向锁标识为0,设置锁标志位为00,升级为轻量级锁,按照轻量级锁的方式进行竞争。
(3)
(4)
6.各种锁的优缺点对比