1 线程安全问题的原因:由于一个线程在操作共享数据过程中,未执行完毕的情况下,另外的线程有参与进来,导致共享数据存在安全问题
2 解决方法:必须让一个线程操作共享数据完毕以后,其它线程才有机会参与共享数据的操作
3 java如何实现线程的安全,现成的同步机制
synchronized(同步监视器){ //需要被同步的代码块(操作共同数据的代码)}
同步监视器:由任何一个类的对象充当,哪个线程获取此监视器,就执行大括号里被同步的代码
1)同步代码块
class Window2 implements Runnable{ private int ticket = 100; //不用static,只创建一个对象 public void run(){ while(true){ synchronized(this){ //注意一个线程进入同步代码块中,在执行的其中的代码过程中其他线程不能进入,所以称为同步锁。synchronized一定要放在共享数据前后 if(ticket>0){ //ticket是共享数据,所以同步从这个地方开始,如果放在while前,那么线程1一直占用cpu直到结束,线程2,3就进不来了
try { Thread.currentThread().sleep(20); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } //只能加try-catch,不能throws,因为子类方法异常不能大于父类 System.out.println(Thread.currentThread().getName()+"售票号码 :"+ticket--); } else{ break; }}
class Window extends Thread{ static int ticket = 100; static Object obj = new Object(); public void run(){ while(true){ synchronized (obj) { if (ticket > 0) { try { Thread.currentThread().sleep(20); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } System.out.println(Thread.currentThread().getName() + ":" + ticket--); } else { break; } }
2)同步方法: 将操作共享数据的方法声明为synchronized,此方法称为同步方法,能够保证当其中一个线程执行此方法时,其他线程在外等待直到此线程执行完此方法。只能在实现方式中能使用,继承方式3个对象,有3个this,不能用同步方法
同步方法的锁:this,不用显式指明。
package lianxi1; //方式二:实现方式创建多线程 class Window3 implements Runnable{ private int ticket = 100; //不用static,只创建一个对象 public void run(){ while(true){ show(); } } public synchronized void show(){ //不是显式调用this,因为只有一个对象,所以只有this就是同步监视器 if(ticket>0){ //ticket是共享数据,所以从这个地方开始 try { Thread.currentThread().sleep(20); } catch (InterruptedException e) { // TODO Auto-generated catch block e.printStackTrace(); } //只能加try-catch,不能throws,因为子类方法异常不能大于父类 System.out.println(Thread.currentThread().getName()+"售票号码 :"+ticket--); } } } public class TestThreadWindow3 { public static void main(String[] args) { Window3 w2 = new Window3(); Thread t1 = new Thread(w2); //将Runnable实现类对象作为形参传人Thread构造器,执行start方法 Thread t2 = new Thread(w2); Thread t3 = new Thread(w2); t1.setName("窗口1"); t2.setName("窗口2"); t3.setName("窗口3"); t1.start(); t2.start(); t3.start(); } }