Java多线程之并发安全经典案例-卖票

4,503 阅读2分钟

#线程相关知识
1.创建线程的两种方式

  • 继承Thread类。
  • 实现Runnable接口。(这种方式较为常用)

2.实现Runnable接口的好处

  • 将线程的任务从线程的子类中分离出来,进行了单独的封装。按照面向对象的思想将任务的封装成对象。
  • 避免了java单继承的局限性。

#多线程并发安全之卖票

  • 代码

/**
 * Created by yuandl on 2016-09-30.
 */
public class RunnableTest implements Runnable {
    private int tick = 60;

    @Override
    public void run() {
        while (true) {
                if (tick == 0) {
                    break;
                }
                try {
                    Thread.sleep(10);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println(Thread.currentThread().getName() + "=========" + tick--);
        }
    }
    public static void main(String[] args) {
        RunnableTest runnableTest=new RunnableTest();
        new Thread(runnableTest).start();
        new Thread(runnableTest).start();


    }
}
  • 打印结果

    Thread-1=========11
    Thread-1=========10
    Thread-0=========9
    Thread-1=========8
    Thread-0=========7
    Thread-0=========6
    Thread-1=========5
    Thread-0=========4
    Thread-1=========3
    Thread-0=========2
    Thread-1=========1
    Thread-0=========0
    Thread-0=========-1
    Thread-0=========-2
    Thread-0=========-3

  • 发现问题,卖票竟然出现了负数,这肯定是有问题的

  • 线程安全问题产生的原因:
    • 多个线程在操作共享的数据。
    • 操作共享数据的线程代码有多条。
    • 当一个线程在执行操作共享数据的多条代码过程中,其他线程参与了运算。就会导致线程安全问题的产生。
  • 解决思路:
    • 就是将多条操作共享数据的线程代码封装起来,当有线程在执行这些代码的时候,其他线程时不可以参与运算的。必须要当前线程把这些代码都执行完毕后,其他线程才可以参与运算。在java中,用同步代码块就可以解决这个问题。
  • 同步代码块的格式

    synchronized(对象)
    {
    需要被同步的代码 ;
    }

  • 同步的好处:解决了线程的安全问题。

  • 同步的弊端:相对降低了效率,因为同步外的线程的都会判断同步锁。
  • 同步的前提:同步中必须有多个线程并使用同一个锁。

    #最终线程安全同步的代码


/**
 * Created by yuandl on 2016-09-30.
 */
public class RunnableTest implements Runnable {
    private int tick = 60;

    @Override
    public void run() {
        while (true) {
            synchronized (this) {
                if (tick == 0) {
                    break;
                }
                try {
                    Thread.sleep(10);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println(Thread.currentThread().getName() + "=========" + tick--);
            }
        }
    }
    public static void main(String[] args) {
        RunnableTest runnableTest=new RunnableTest();
        new Thread(runnableTest).start();
        new Thread(runnableTest).start();
    }
}
  • 执行结果

    Thread-1=========10
    Thread-1=========9
    Thread-1=========8
    Thread-1=========7
    Thread-1=========6
    Thread-1=========5
    Thread-1=========4
    Thread-1=========3
    Thread-1=========2
    Thread-1=========1

Process finished with exit code 0

完美解决以上问题