用同步代码块解决实现Runnable接口创建线程的线程安全问题
此处用买票窗口举例说明
package com.xiancheng.bean;
/*
* 用实现Runnable接口的方式来创建多线程
*
* 存在线程安全问题
* 1.问题:买票过程中,出现了重票、错票
* ①问题出现的原因是什么
* 当某个线程操作车票的过程中,尚未操作完成,其它线程参与进来,也操作车票
* ②如何解决
* 思路:当一个线程a在操作ticket的时候,其它线程不能参与进来,直到线程a操作完ticket时
* 线程才可以操作ticket,这种情况即使线程a出现了阻塞,也不能被改变
*
* 方案:在Java中,我们通过同步机制,来解决线程的安全问题
*
* 方式一:同步代码块
* synchronized(同步监视器){
* //需要被同步的代码(请参考以下说明)
*
* }
* 说明:①操作共享数据的代码,即为需要被同步的代码
* ②共享数据:多个线程共同操作的变量即为共享数据
* ③同步监视器(锁):任何一个类的对象,都可以充当锁(要求多个线程必须公用同一把锁)
*
* 好处:解决了代码的安全问题
* 局限性:操作同步代码时,只能有一个线程参与,其它线程等待,相当于是一个单线程的过程,效率低
*
* 补充:在实现Runnable接口创建多线程的方式中,我们也可以考虑使用this充当同步监视器
* 即把obj直接换成this,此时无需专门创建一个Object类的对象
* */
public class TicketTwo {
public static void main(String[] args) {
TestFive t = new TestFive();
Thread t1 = new Thread(t);
Thread t2 = new Thread(t);
Thread t3 = new Thread(t);
t1.setName("窗口一");
t2.setName("窗口二");
t3.setName("窗口三");
t1.start();
t2.start();
t3.start();
}
}
class TestFive implements Runnable{
private int ticket = 100;
final Object obj = new Object(); //注意:在这里要加上final,意味着锁不能被改变
public void run(){
while(true){
synchronized(obj){
if(ticket>0){
System.out.println(Thread.currentThread().getName()+"卖的票号为:"+ticket);
ticket--;
}
else{
break;
}
}
}
}
}