synchronized用于实现同步处理,保证共享数据的安全性
* 数据有安全性问题的原因:1、共享数据 2、修改数据
* synchronized相对于volatile是重量级的线程安全的方法,可以保证3大特性:原子性、可见性、有序性。可以将并发操作转换为串型执行
*用于静态方法,锁对象为当前类
*public static synchronized void pp(){}
* 用于非静态方法,锁对象为当前类的对象
* public synchronized void pp(){}
* 用于代码块,锁对象为指定的对象
*synchronized(obj){}
synchronized用法:
用法1:同步代码块
private final static String LOCK="lock1";
synchronized (LOCK) {}
synchronized用法2:同步方法 在方法上添加同步关键字,当前的锁对象为当前对象no---对象锁.
以new出来的NumOper对象充当锁,当前对象内的所有synchronized方法在不同线程调用时互斥,但是可以直接访问非synchronized方法。注意synchronized允许持有锁的线程重入.
public synchronized void add() {
System.out.println(Thread.currentThread() + "....add...begin:" + this.num);
this.num++;
sub();
System.out.println(Thread.currentThread() + "....add...end" + this.num);}
public synchronized void sub() {
System.out.println(Thread.currentThread() + "....sub...begin" + this.num);
this.num--;
System.out.println(Thread.currentThread() + "....sub...end" + this.num);}
用法3:同步静态方法,以当前类Class对象作为锁---类锁
针对一个类一般只会存储一个
使用类锁,所以不管new了多少个对象,都可以得到互斥的效果
private synchronized static void add() {
System.out.println(Thread.currentThread() + "开始创建操作" + counter);
counter++;
eee();
System.out.println(Thread.currentThread() + "完成创建操作" + counter);}
使用的是对象锁,所以只能new一个对象,才可以得到互斥的效果。如果创建多个则不能达到互斥的目的
引入锁机制以解决线程安全问题:
悲观锁:当存在多个线程操作共享数据时,需要保证同一时刻有且只有一个线程在操作共享数据,其他线程必须等到该线程处理完数据后再进行
乐观锁:CAS compare and set
synchronized总结
* synchronized同步关键字,用于代码同步处理,解决线程安全问题
*synchronized同步方法 以当前对象充当锁
public synchronized void pp(){}
* synchronized同步静态方法 以当前类Class充当锁
public synchronized static void pp(){}
* - synchronized同步代码块 自定义对象充当锁
synchronized(obj){}
synchronized原理
* 在添加synchronized关键字后就可以保证在一个时刻上只有一个线程在调用某个方法或者代码块,不会出现并发的情形,达到排队执行的效果。
在Java中synchronized可保证在同一个时刻,只有一个线程可以执行某个方法或者某个代码块(主要是对方法或者代码块中存在共享数据的操作),同时还应该注意到synchronized另外一个重要的作用,synchronized可保证一个线程的变化(主要是共享数据的变化)被其他线程所看到(保证可见性,完全可以替代volatile功能),这点确实也是很重要的。
锁对比
*偏向锁: 线程在大多数情况下并不存在竞争条件,使用同步会消耗性能,而偏向锁是对锁的优化,可以消除同步,提升性能。当一个线程获得锁,会将对象头的锁标志位设为01,进入偏向模式。偏向锁可以在让一个线程一直持有锁,在其他线程需要竞争锁的时候,再释放锁 | 加锁和解锁不需要额外的消耗,和执行非同步方法相比仅存在纳秒级的差距 | 如果线程间存在锁竞争,会带来额外的锁撤销的消耗 | 适用于只有一个线程访问同步块的场景 |
*轻量级锁: 当线程A获得偏向锁后,线程B进入竞争状态,需要获得线程A持有的锁,那么线程A撤销偏向锁,进入无锁状态。线程A和线程B交替进入临界区,偏向锁无法满足,膨胀到轻量级锁,锁标志位设为00 | 竞争的线程不会阻塞,提高了程序的相应速度 | 如果始终得不到所竞争的线程,使用自旋会消耗CPU | 追求相应速度,同步块执行速度非常块 |
*重量级锁: 当多线程交替进入临界区,轻量级锁hold得住。但如果多个线程同时进入临界区,hold不住了,膨胀到重量级锁 | 线程竞争不使用自旋,不会消耗CPU | 线程阻塞,响应时间缓慢 | 追求吞吐量,同步块执行速度较慢
会同时被多个线程访问的资源,就是竞争资源(临界资源),也称为竞争条件。对于多线程共享的资源(临界资源)必须进行同步,以避免一个线程的改动被另一个线程所覆盖