**同步函数用的是哪个锁呢?
函数需要对对象调用,那么函数都有一个所属对象引用,就是this.
所以同步函数使用的锁是this**
通过该程序进行验证
下面想通过flag的值来执行不同的代码,让线程1执行if,让线程2执行else
使用两个线程卖票,一个线程在同步代码块if中,一个线程在同步代码块else中,都在执行卖票动作。
public class Demo implements Runnable{
private int tick=20;
Object obj=new Object();
public boolean flag=true;
public void run() {
if(flag) {
while(true) {
synchronized(obj) {
if(tick>0) {
try {Thread.sleep(10); } catch (Exception e) { }
System.out.println(Thread.currentThread().getName()+"code*****:"+tick--);
}
}
}
}
else {
while(true) {
show();
}
}
}
public synchronized void show()
{
if(tick>0) {
try {
Thread.sleep(10);
} catch (Exception e) {
// TODO Auto-generated catch block
}
System.out.println(Thread.currentThread().getName()+"show-----"+tick--);
}
}
}
public class ThreadDemo {
public static void main(String[] args) {
// TODO Auto-generated method stub
Demo d=new Demo();
Thread t1=new Thread(d);
Thread t2=new Thread(d);
t1.start();
d.flag=false;
t2.start();
}
}
**执行完后呢,没有code输出,只有show。
原因是**
现在存在三个线程:主线程、t1、t2。
当线程创建以后,并不是立刻开启,t1.start();开启不一定执行。
很可能不执行,继续执行d.flag=false;这个主线程。然后主线程执行完毕。但是呢flag已经更改为false,都执行show
为了能够看到code是执行了的,在主函数的t1.start();后面写 try {Thread.sleep(10); } catch (Exception e) { }
这样就可以看到code执行了。
下图为执行结果:
这样code就输出了,但是为啥是交替的,
线程1用的是obj这个锁。原因是if一旦进去,就是个无线循环,因为没别的线程争这个锁(obj),但是中间有睡眠,可以让线程2执行show方法,所以交替了。
但是不安全,打印了错票0。
既然同步出现了问题,说明不满足同步的前提:
同步的前提:
1必须要有两个或者两个以上的线程ok
2.必须是多个线程使用同一个锁no
3必须保证同步中只能有一个线程在运行
code用的obj锁,show用的this锁,若将code锁改成this
即synchronized(this),就不会出现打印错票的问题