为什么使用synchronized?
因为线程调用的操作并不是原子操作,即方法中步骤可能在没有运行完就被剥夺了运行权。
假设两个线程同时执行指令
i++;
i++并不是原子操作,该指令可能有以下三步:
- i加载到寄存器
- 给i增加1
- 结果写回i
如果第一个线程执行1、2步,他就被剥夺了运行权。线程二执行1-3。然后,线程一被唤醒然后完成第3步,此时,线程一的操作抹去了线程二的操作,出现线程安全问题。
那么,要解决这个问题,就必须保证一个线程执行操作时不能被打断,也就是说确保线程失去运行权之前运行完成,那么可以用到synchronized关键字。
synchronized关键字的原理
我们知道,Java中的每一个对象都有一个内部锁,若是一个方法用synchronized修饰,则该对象的锁将保护此方法。每个实例对象都有自己的锁。(即一个对象线程锁被拿去,其他线程还可以拿同一个类的另外一个对象的锁)
synchronized关键字的使用方法
目的:使得同一个对象上达到同步
public class Syn_demo{
private static final Object staticLockObj = new Object();
public synchronized void s1(){
//同实例方法争夺该锁
}
public void s2(){
synchronized(this){
//同实例方法争夺该锁
}
public void s3(){
synchronized(staticLockObj){
//同类争夺该锁
}
}
public void s4(){
synchronized(Syn_demo.class){
//同类争夺该锁
}
}
public static synchronized void s5() {
//同类争夺该锁
}
}
}