先来看看synchronized这个关键字是什么意思:
同步,同步什么?
要同步的就是这个关键字后面紧跟的花括号里的代码,也就是代码块。
为什么要同步?
来看一个例子:
public static int i=0;
public static void main(String[] args) throws InterruptedException {
var t1=new Thread(()->{
for (int i1 = 0; i1 < 10000; i1++) {
i++;
}
});
var t2 = new Thread(()->{
for (int i1 = 0; i1 < 10000; i1++) {
i++;
}
});
t1.start();
t2.start();
t1.join();
t2.join();
System.out.println(i);
}
这段代码理论上最后输出的i的值应该是20000,
但是本次运行之后的结果却是17146,为什么?
网上的相关说明有很多,总结一下就是一段线程代码运行到一半,这时候线程调度器又跑去运行另外一个线程,导致线程内存中所做的修改没来得及刷新到主内存,从而导致数据不一致。
这个时候,就轮到我们的主角登场了,synchronized关键字的作用就是锁住某一个对象,当这个对象没被解锁之前,别的线程无法锁住这个对象。
那么,我们来尝试一下:
这是修改之后的部分代码:
var t1=new Thread(()->{
synchronized (Main1.class){
for (int i1 = 0; i1 < 10000; i1++) {
i++;
}
}
});
var t2 = new Thread(()->{
synchronized (Main1.class){
for (int i1 = 0; i1 < 10000; i1++) {
i++;
}
}
});
使用synchronized锁住了Main1.class这个对象,这样我们就能保证t1和t2能从头到尾完整地运行一遍,不会出现之前那样的情况。
无论运行多少遍,结果肯定为20000