* 如何解决线程安全问题呢?
*
* 要想解决问题,就要知道哪些原因会导致出问题:(而且这些原因也是以后我们判断一个程序是否会有线程安全问题的标准)
* A:是否是多线程环境
* B:是否有共享数据
* C:是否有多条语句操作共享数据
*
* 我们来回想一下我们的程序有没有上面的问题呢?
* A:是否是多线程环境 是
* B:是否有共享数据 是
* C:是否有多条语句操作共享数据 是
*
* 由此可见我们的程序出现问题是正常的,因为它满足出问题的条件。
* 接下来才是我们要想想如何解决问题呢?
* A和B的问题我们改变不了,我们只能想办法去把C改变一下。
* 思想:
* 把多条语句操作共享数据的代码给包成一个整体,让某个线程在执行的时候,别人不能来执行。
* 问题是我们不知道怎么包啊?其实我也不知道,但是Java给我们提供了:同步机制。
*
* 同步代码块:
* synchronized(对象){
* 需要同步的代码;
* }
*
* A:对象是什么呢?
* 我们可以随便创建一个对象试试。
* B:需要同步的代码是哪些呢?
* 把多条语句操作共享数据的代码的部分给包起来
*
* 注意:
* 同步可以解决安全问题的根本原因就在那个对象上。该对象如同锁的功能。
* 多个线程必须是同一把锁。
同步的特点:
* 前提:
* 多个线程
* 解决问题的时候要注意:
* 多个线程使用的是同一个锁对象
* 同步的好处
* 同步的出现解决了多线程的安全问题。
* 同步的弊端
* 当线程相当多时,因为每个线程都会去判断同步上的锁,这是很耗费资源的,无形中会降低程序的运行效率。
1 public class SellTicketDemo { 2 public static void main(String[] args) { 3 // 创建资源对象 4 SellTicket st = new SellTicket(); 5 6 // 创建三个线程对象 7 Thread t1 = new Thread(st, "窗口1"); 8 Thread t2 = new Thread(st, "窗口2"); 9 Thread t3 = new Thread(st, "窗口3"); 10 11 // 启动线程 12 t1.start(); 13 t2.start(); 14 t3.start(); 15 } 16 } 17 public class SellTicket implements Runnable { 18 // 定义100张票 19 private int tickets = 100; 20 //创建锁对象 21 private Object obj = new Object(); 22 23 // @Override 24 // public void run() { 25 // while (true) { 26 // synchronized(new Object()){ 27 // if (tickets > 0) { 28 // try { 29 // Thread.sleep(100); 30 // } catch (InterruptedException e) { 31 // e.printStackTrace(); 32 // } 33 // System.out.println(Thread.currentThread().getName() + "正在出售第" 34 // + (tickets--) + "张票"); 35 // } 36 // } 37 // } 38 // } 39 40 @Override 41 public void run() { 42 while (true) { 43 synchronized (obj) { 44 if (tickets > 0) { 45 try { 46 Thread.sleep(100); 47 } catch (InterruptedException e) { 48 e.printStackTrace(); 49 } 50 System.out.println(Thread.currentThread().getName() 51 + "正在出售第" + (tickets--) + "张票"); 52 } 53 } 54 } 55 } 56 }