今日学习内容总结如下:
synchronized总结
synchronized同步关键字,用于代码同步处理,解决线程安全问题
synchronized同步方法 以当前对象充当锁
public synchronized void pp(){}
synchronized同步静态方法 以当前类Class充当锁
public synchronized static void pp(){}
synchronized同步代码块 自定义对象充当锁
synchronized(obj){}
synchronized原理
在添加synchronized关键字后就可以保证在一个时刻上只有一个线程在调用某个方法或者代码块,不会出现并发的情形,达到排队执行的效果。
在Java中synchronized可保证在同一个时刻,只有一个线程可以执行某个方法或者某个代码块(主要是对方法或者代码块中存在共享数据的操作),同时还应该注意到synchronized另外一个重要的作用,synchronized可保证一个线程的变化(主要是共享数据的变化)被其他线程所看到(保证可见性,完全可以替代volatile功能),这点确实也是很重要的。
在JDK1.6之前一般不建议使用synchronized,因为相比较Lock接口而言,是重量级的
jdk6之前是重量级锁,JDK6开始优化锁的状态总共有四种,无锁状态(使用乐观锁CAS,没有synchronized)、偏向锁、轻量级锁和重量级锁。锁状态的改变是根据竞争激烈程度进行的,在几乎无竞争的条件下,会使用偏向锁,在轻度竞争的条件下,会由偏向锁升级为轻量级锁, 在重度竞争的情况下,会升级到重量级锁。 随着锁的竞争,锁可以从偏向锁升级到轻量级锁,再升级的重量级锁,但是锁的升级是单向的,也就是说只能从低到高升级,不会出现锁的降级
对象在内存中存储时可以分为对象头、实例数据和对齐字节三部分。对象头数据一般包含标识字mark word和类型指针klass point;实例数据就是具体对象的成员数据,一般按照4B为的单位进行数据存储;最后的对齐字节用于将对象存储的数据凑够8字节的整数倍。
Mark Word:默认存储对象的HashCode,分代年龄和锁标志位信息。这些信息都是与对象自身定义无关的数据,所以Mark Word被设计成一个非固定的数据结构以便在极小的空间内存存储尽量多的数据。它会根据对象的状态复用自己的存储空间,也就是说在运行期间Mark Word里存储的数据会随着锁标志位的变化而变化
Klass Point:对象指向它的类元数据的指针,虚拟机通过这个指针来确定这个对象是哪个类的实例。
synchronized用的锁是存在Java对象头里的,存在锁对象的对象头的Mark Word中