synchronized方法

本文通过售票系统的案例探讨了在多线程环境下synchronized关键字的使用技巧。文章详细解释了将synchronized应用在run方法上的影响,并对比了使用显式锁(ReentrantLock)和synchronized块的效果。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >


在看多线程这块,突然有个想法,把synchronized关键字直接加到run方法上,会有什么效果。以售票为例,结果就出问题了。

MyRunnable m = new MyRunnable();
		
Thread t1 = new Thread(m,"1");
Thread t2 = new Thread(m,"2");
Thread t3 = new Thread(m,"3");
Thread t4 = new Thread(m,"4");
Thread t5 = new Thread(m,"5");
			
t1.start();
t2.start();
t3.start();
t4.start();
t5.start();

private int tickets = 2000;
	
	@Override
	public synchronized void run(){
		while(true){
			try{
				if(tickets>0){
					try {
						Thread.sleep(5);
					} catch (Exception e) {
						e.printStackTrace();
					}
					System.err.println("窗口"+Thread.currentThread().getName()+"正在出售第"+(tickets--)+"张票");
				}
			}catch(Exception e){
				e.printStackTrace();
			}
		}
	}
结果:

效果就是只有一个线程执行,其余线程不执行。后来查了一下,被synchronized关键字修饰的方法,任何时候最多只能被一个线程执行。问题应该就出在这里。当一开始其中一个线程抢到执行权后,由于该方法中有一个while循环一直执行,所以该线程就一直执行,直到结束,退出方法,其他线程才有机会获得执行权,但该例子中此时程序已经执行结束了。一直处于等待状态。


private int tickets = 100;	
	Lock lock = new ReentrantLock();
	@Override
	public  void run() {
		while(true){
			try{
				lock.lock();
				if(tickets>0){
					try{
						Thread.sleep(10);
					}catch(Exception e){
						e.printStackTrace();
					}
					System.err.println("窗口"+Thread.currentThread().getName()+"正在出售第"+(tickets--)+"张票");
				}
			}catch (Exception e) {
				e.printStackTrace();
			}finally{
				lock.unlock();
			}
		}
	}

或者:

Object o = new Object();
	
	@Override
	public  void run(){
			while(true){
				synchronized(o){
				try{
					if(tickets>0){
						try {
							Thread.sleep(300);
						} catch (Exception e) {
							e.printStackTrace();
						}
						System.err.println("窗口"+Thread.currentThread().getName()+
								"正在出售第"+(tickets--)+"张票");
					}
				}catch(Exception e){
					e.printStackTrace();
				}
			}
		}
		
	}

将之前的代码改为上面代码执行就正常了。总之,就是把同步锁加在while循环内就可以了,这样每进行一次while循环,其他线程都有机会获得锁对象。


结果:

由于sleep时间短,效果不是很明显






评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值