synchronized是java提供的原子性内置锁,这种内置而且使用者看不到的锁也叫做监视器锁,使用synchronized关键字后会在编译器之后在同步代码块前后加上monitorenter和monitorexit指令,他依赖操作系统底层的互斥锁实现,他的主要作用是实现原子性操作和解决共享变量的内存可见性问题。
执行monitorenter指令会尝试获取对象所,如果对象没有锁或者已经获得了对象锁,锁的计数器+1.此时其他竞争锁的线程会进入等待队列当中。
执行monitorexit指令会把锁的计数器-1.当计数器为0的时候,锁会释放,处于等待队列中的其他线程再继续竞争锁。
synchronized是排他锁,当一个线程获取了锁之后,其他线程必须等待该线程释放之后才能获得锁,由于java中线程与操作系统原生线程是一一对应的,线程阻塞或者唤醒从用户态到内核态这种转换十分消耗性能。
从内存语义来说,加锁过程会清除内存中的共享变量,再从主存中读取,而释放锁的过程是把工作内存中的共享变量写回主存。
synchronized实际上有两个队列WaitSet和entryList
1.如果有多个线程进入同步代码块时候,首先进入entryList
2.若有一个线程获取到monitor锁之后,就赋值给当前线程,计数器+1
3.如果线程调用了wait方法之后,会释放锁,当前线程置为null,计数器-1,同时进入WaitSet等待被唤醒,调用notify或者notifyAll之后,又会进入entryList队列当中竞争锁。
4.如果线程执行完毕,同样释放锁,计数器-1,当前线程置为null