创建线程是为了程序具有单独执行的路径,让多部分代码同时执行。那么多个线程共同操作同一组数据,执行结果和单线程的执行结果是一样的吗?会不会产生一些错误呢?
就拿我们身边的例子来说,买火车票,我们可以在12306官网购票,可以去售票厅买,还可以打电话预约……但是票数是固定的,相同的时间会有很多人买票,那么这个票到底卖给谁,12306铁道到底是如何决定的如何操作这些数据的,也许我们不是很清楚,但可以确定的是,同一张票只能售给一个人,否则好多人拿着相同的票去坐车,用相同的座位,岂不是天下大乱。该如何解决这个问题呢?那就是线程同步。
【线程同步】
线程同步可以解决,保证线程安全。如果有多个线程在同时运行,而这些线程会同时运行一段代码。程序每次运行结果和单线程运行的结果是一样的,而且其他的变量的值也和预期的是一样的,就是线程安全的。
【同步分类】
与关键字:synchronized分不开
【代码实现】
private static int ticket=100;假设共享数据时ticket
---同步代码块(任意对象)
synchronized (obj) {
if(ticket==0){
break;
}else{
try {
Thread.sleep(10);
} catch (InterruptedException e) {
System.out.println(e.getMessage());
e.printStackTrace();
}//为了看到线程不安全问题,让程序走到这里睡眠10毫秒
System.out.println(Thread.currentThread().getName()+"正在售出第"+ticket--+"票");
}
}
---静态同步方法(类名.class)
public static synchronized void synchronizenFunction_1(){
if(ticket==0){
return ; //break;//要在循环中写
}else{
try {
Thread.sleep(10);//为了看到线程不安全问题,让程序走到这里睡眠10毫秒
} catch (InterruptedException e) {
System.out.println(e.getMessage());
e.printStackTrace();
}
System.out.println(Thread.currentThread().getName()+"正在售出第"+ticket--+"票");
}
}
---非静态同步方法(this)
public synchronized void synchronizenFunction(){
if(ticket==0){
return ; //break;//要在循环中写
}else{
try {
Thread.sleep(10);
} catch (InterruptedException e) {
System.out.println(e.getMessage());
e.printStackTrace();
}//为了看到线程不安全问题,让程序走到这里睡眠10毫秒
System.out.println(Thread.currentThread().getName()+"正在售出第"+ticket--+"票");
}
}
---Lock锁接口
public void run() {
while(true){
l.lock();
if(ticket==0){
break;
}else{
try {
Thread.sleep(10);
//为了看到线程不安全问题,让程序走到这里睡眠10毫秒
System.out.println(Thread.currentThread().getName()+"正在售出第"+ticket--+"票");
} catch (InterruptedException e) {
System.out.println(e.getMessage());
e.printStackTrace();
}finally{
l.unlock();
}
}
}
}
【小结】
通过线程同步就可以轻而易举的解决线程安全问题了,这样就再也不用担心几个人拿着同一张票去乘车了。需要注意的是多个线程要保证安全就必须使用同一个锁。