以卖票的例子来介绍多线程和资源共享,下面我们来看看为什么要用卖票作为例子。
卖票是包含一系列动作的过程,有各种操作,例如查询票、收钱、数钱、出票等,其中有一个操作是每次卖掉一张,就将总的票数减去1。有10张票,如果一个人卖票,先做查票、收钱、数钱等各种操作,再将总的票数减去1,效率很低。如果多个人卖票,每个人都是做同样的操作,数钱、检查钱,最后将总的票数减1,这样效率高。但是有一个问题,如果出现两个人同时将总的票数减掉了1,例如,A、B两个人同时读取到票的总数是10,A从中减去1,同时B也从中减去1,总数显示是9,其实票只有8张。导致数据错误。
按照正常逻辑,同一时刻只允许一个人来从总票数中减去1,A读取总票数,再减去1的过程中,B必须等待,等A操作完了,B才能进行。其实票就是共享资源,一次只能由一个人访问。这里就要用到同步机制,即锁机制,使用关键词synchronized将读取总的票数,并减去1的操作锁定,使得一次只能由一个人访问。每个售票员就是一个线程,多个售票员进行同一项卖票任务。
synchronized原理是,执行synchronized部分代码的时候必须需要对象锁,而一个对象只有一个锁,只有执行完synchronized里面的代码后释放锁,其他线程才可以获得锁,那么就保证了同一时刻只有一个线程访问synchronized里面的代码。使得资源共享的关键是,只有一个实例,synchronized使用的是同一把锁,用实例的锁或者定义一个实例。这就需要使用实现Runnable接口的方式&#x