Java实现司机售票员线程同步模式

实现司机售票员线程同步模式,比较简单,通俗易懂。

售票员活动:关闭车门

司机的活动:启动车子

司机的活动:正常行驶;售票员活动:售票

司机的活动:到站停车;

售票员活动:打开车门;


pv操作实现

driver(){
  while(true){
    P(driver);
    start();
    driving();
    stop();
    V(Seller);
  }
}

seller(){
  while(true){
    close(door);
    V(Driver);
    sell();
    P(Seller);
    open(door);
  }
}

很容易看出上述存在死锁,因为如果seller执行完V(Driver)之后Driver还没有执行到P(Driver),那么当Driver执行到P(Driver)之后则会挂起,而Seller到了P(Seller)之后也同样会挂起,因此死锁产生。

死锁产生的原因是V操作的信号被忽略了。因此只要保持住该信号就可以唤醒Driver。因此java代码对其的改进如下所示:

package concurrency;

import java.security.interfaces.DSAKey;
import java.util.Random;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

public class DriverAndSeller {
	Object driver = new Object();
	Object seller = new Object();
	int driverNotify = 0;
	int sellerNotify = 0;
	class Driver implements Runnable{
		public void run(){
			Random random = new Random(47);
			while(true){
				try{
					synchronized (driver) {
						if(driverNotify==0)
							driver.wait();
						driverNotify--;
					}
					System.out.println("[Driver]: Starting...");
					Thread.sleep(random.nextInt(1000)+1000);
					System.out.println("[Driver]: Runing...");
					Thread.sleep(random.nextInt(6000));
					System.out.println("[Driver]: Stop...");
					synchronized (seller) {
						sellerNotify++;
						seller.notify();
					}
				}catch (InterruptedException e) {
					e.printStackTrace();
				}
			}
		}
	}
	
	class Seller implements Runnable{
		public void run(){
			Random random = new Random(47);
			while(true){
				try{
					System.out.println("[Seller]: Closing door...");					
					synchronized (driver) {
						driverNotify++;
						driver.notify();
					}
					Thread.sleep(random.nextInt(1000)+500);
					System.out.println("[Seller]: Selling tickets...");
					Thread.sleep(random.nextInt(1000)+5000);
					synchronized (seller) {
						if(sellerNotify==0)
						  seller.wait();
						sellerNotify--;
					}
					System.out.println("[Seller]: Opening door...");
					Thread.sleep(random.nextInt(300)+300);
				}catch (InterruptedException e) {
					e.printStackTrace();
				}
			}
		}
	}
	
	public static void main(String[] args){
		DriverAndSeller das = new DriverAndSeller();
		ExecutorService es = Executors.newFixedThreadPool(2);
		es.execute(das.new Driver());
		es.execute(das.new Seller());
	}
}

所得的结果是:

[Seller]: Closing door...
[Driver]: Starting...
[Seller]: Selling tickets...
[Driver]: Runing...
[Driver]: Stop...
[Seller]: Opening door...
[Seller]: Closing door...
[Driver]: Starting...
[Seller]: Selling tickets...
[Driver]: Runing...
[Driver]: Stop...
[Seller]: Opening door...
[Seller]: Closing door...
[Driver]: Starting...
[Seller]: Selling tickets...
[Driver]: Runing...
[Driver]: Stop...
[Seller]: Opening door...
[Seller]: Closing door...
[Driver]: Starting...
[Seller]: Selling tickets...



  • 1
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
汽车司机售票员之间必须协同工作,一方面,只有售票员把车门关好了司机才能开车,因此,售票员关好车门应通知司机开车。另一方面,只有当司机已经停 下,售票员才能开门上下客,故司机停车后应通知售票员。假定某辆公共汽车上有两名售票员与一名司机汽车当前正在始发站停车上客,试设必要的信号灯及赋初值,写出他们的同步过程 分析: 司机停车,通知售票员开门,售票员关门,通知司机开车 使用到的函数和信号量 HANDLE mutex; HANDLE empty; HANDLE full; 创建信号量 HANDLE CreateSemaphore( __in_opt LPSECURITY_ATTRIBUTES lpSemaphoreAttributes,// lpSemaphoreAttributes是信号量的安全属性 可为NULL __in LONG lInitialCount,// lInitialCount是初始化的信号量 __in LONG lMaximumCount,// lMaximumCount是允许信号量增加到最大值 __in_opt LPCWSTR lpName//lpName是信号量的名称 可为NULL ); 创建互斥信号量 HANDLE CreateMutex(  LPSECURITY_ATTRIBUTES lpMutexAttributes, // 指向安全属性的指针 可为NULL  BOOL bInitialOwner, // 初始化互斥对象的所有者  LPCTSTR lpName // 指向互斥对象名的指针 ); 申请一个资源 WaitForSingleObject(HANDLE full,INFINITE); 释放资源 ReleaseSemaphore( __in HANDLE hSemaphore,// hSemaphore是要增加的信号量句柄 __in LONG lReleaseCount,// lReleaseCount是增加的计数。 __out_opt LPLONG lpPreviousCount//lpPreviousCount是增加前的数值返回。 ); 释放互斥信号量 BOOL ReleaseMutex(HANDLE hMutex); DWORD WaitForMultipleObjects( DWORD nCount, // number of handles in array CONST HANDLE *lpHandles, // object-handle array BOOL bWaitAll, // wait option DWORD dwMilliseconds // time-out interval );

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值