13.7 线程安全

案例:3个售票窗口
class test extends Thread
{
	int tick= 40;
	
	test(String name)
	{
		super(name);
	}

	@Override
	public void run() {
		while(true)
		{
			if(0 < tick)
			{
				System.out.println(Thread.currentThread().getName()+"售了第"+tick+"票");
				tick--;
			}
			
			else
			{
				System.out.println("票买完了");
				break;
			}
		}
		
	} 
	
}


class wu
{
 public static void main(String [] args)
 {
	test t1 = new test("窗口一");
	test t2 = new test("窗口二");
	test t3 = new test("窗口三");
	t1.start();
	t2.start();
	t3.start();
	
 }
}




这样的远行结果就是一共买了40*3=120张票,所以,在定义票时,应该把票设定为static,不然每次new test类时,就相当于把票的数字翻一倍。




再加了static后,运行了,然后就会发现,还有另一个问题,就是信息安全的问题,例如我们现实票已经买完了,但是还可以卖票。


用synchronized控制同步代码块。其的参数是任意的对象都可以作为锁对象。


同步代码块注意事项:
1、任何一个对象都可以做为锁的对象。
2、在同步代码中调用了sleep方法不是释放了对象。
3、只有真正在线程安全问题的时候才使用同步代码块,否则会降低效率。
4、线程操作的锁对象必须是唯一共享的,否则无效。

出现线程安全问题的根本原因:
1、存在两个或者两个以上的线程对象,而且线程之间共享着一个资源。
2、有多个语气操作了共享资源。

class test extends Thread
{
	static int tick= 40;
	static Object d = new  Object();//这个对象资源为静态的,不然无效
	test(String name)
	{
		super(name);
	}

	@Override
	public void run() {
		
		while(true)
		{
			synchronized(d)//这个对象,也可以是“文字”例如:synchronized("锁对象")
			{
				if(0 < tick)
				{
					System.out.println(Thread.currentThread().getName()+"售了第"+tick+"票");
					tick--;
				}
				
				else
				{
					System.out.println("票买完了");
					break;
				}
			}
		}		
	}
}

class wu
{
 public static void main(String [] args)
 {
	test t1 = new test("窗口一");
	test t2 = new test("窗口二");
	test t3 = new test("窗口三");
	t1.start();
	t2.start();
	t3.start();
	
 }
}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值