synchnorized ,多线程对同一资源进行读写操作,会出现数据不一致的问题,为了保证线程安全,需要同步代码块。
synchnorized 锁 是可重入锁,当前线程获得锁,并且后面操作也是需要同样的锁时,锁可以重入,就是锁+1 +1,-1,-1。
锁的是对象,不是锁代码块,拿到锁的线程才能执行同步代码块,拿不到锁的线程等待锁释放,获得锁。
java对象的结构 可以看转载文章Java对象结构与锁实现原理及MarkWord详解,其中是使用对象结构中的对象头里的markword进行标识对象的线程锁状态。
在hotspot中 jdk早起的synchnorized的实现是重量级锁即os(系统锁)级别的。
后来的改进,synchnorized的实现锁升级,即由偏向锁->自旋锁->重量级锁。
偏向锁 锁对象标记一下线程的id,不需要上锁, 如果下次来的还是同一个线程,那么直接执行同步代码块,否则就升级为自旋锁。
自旋锁 是线程在一直循环询问锁释放了没有,消耗cpu,当询问10次之后锁就升级重量级锁。
重量级锁 没有获得锁的线程在就绪队列里等待,不消耗cpu
自旋锁 消耗cpu 用户态 快,AtomicL、 Lock使用自旋锁。
重量级锁 不消耗cpu,消耗系统资源, 用户态到内核态 慢。
所以执行时间短(加锁代码)、线程数少 用自旋。
执行时间长、线程数多,用系统锁。
注意点,syschnorized(object) objec不能用String常量、Integer、Long ,会引发奇怪的问题。
发生异常,锁会释放,因此在处理高并发的时候,要格外小心不一致的问题。