玩转并发-单线程执行设计模式

概述

Single Thread Execution模式是指同一时刻只能有一个线程去访问共享资源,就像独木桥一次通过一个人一样。简单来说,Single Thread Execution就是采用排他式操作保证同一时刻只能有一个线程访问共享资源。


有一个门始终只能一个人通过

先写个线程不安全的栗子:

package com.Reyco.MyThread;

public class Gate {
	private int count=0;
	private String name="Nobody";
	private String address="Nowhere";
	
	//门通过
	public void pass(String name,String address) {
		this.name=name;
		this.address=address;
		count++;
		verify();
	}
	
	//校验
	private void verify() {
		if(this.name.charAt(0)!=this.address.charAt(0)) {
			System.out.println("================");
			System.out.println("broken"+toString());
		}
	}
	
	@Override
	public String toString() {
		return "NO."+count+":"+name+address;
	}
}


public class Person	extends Thread{
	private final String name;
	private final String address;
	private final Gate gate;
	
	public Person(String name, String address, Gate gate) {
		this.name = name;
		this.address = address;
		this.gate = gate;
	}
	
	@Override
	public void run() {
		System.out.println(name+"Begin!");
		while(true) {
			this.gate.pass(name, address);
		}
	}
}

测试类:

public static void main(String[] args) {
		Gate gate = new Gate();
		Person person1 = new Person("Beyco","Beijing",gate);
		Person person2 = new Person("Gbenl","Guangzhou",gate);
		Person person3 = new Person("Seyco","Shenzhen",gate);
		
		person1.start();
		person2.start();
		person3.start();
	}

打印结果:

brokenNO.1112588:BeycoBeijing
================
brokenNO.1113089:BeycoShenzhen
================
brokenNO.1113562:BeycoShenzhen

我们发现id顺序发生错误,且name和address不匹配;即使name和address首字母相同,仍然发生broke。其实这两个现象很好解释,比如A线程传入参数Beyco和Beijing,当向pass方法传入Beyco时,A线程被B线程打断,B线程传入Shenzhen,此时verify自然就broke;另一种情况是,继前面一种情况之后,此时参数已经是Beyco和Shenzhen,在准备toString时,B线程又将address传入实参Beijing,所以就出现了首字母相同还是broke的情况。
解决方法很简单:将pass方法和toString方法同步即可

public synchronized String toString() {
		return "NO."+count+":"+name+address;
	}
/**
	 * 临界值
	 * @param name
	 * @param address
	 */
	public synchronized void pass(String name,String address) {
		//线程竞争
		this.name=name;
		this.address=address;
		count++;
		verify();
	}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值