小博..... 2022-07-12 22:19 采纳率: 50%
浏览 17
已结题

多线程 同步锁Lock

在做多线程 同步时,使用Lock锁实现同步。以买票的例子。把线程休眠的时间放在 加锁Lock.lock()前,每个线程都能运行买票。启动3个线程
代码


package com.example.demo.therad.synchronizedThread;
//买票
import java.util.concurrent.locks.ReentrantLock;
public class BuyTicket  implements Runnable{
   private int ticketNum=10;//票数   
   Boolean flag=true;//线程外部结束标志 
   private final ReentrantLock lock=new ReentrantLock();    
    public void run() {
        while(flag) {        
        try {
                **Thread.sleep(1000);**
                lock.lock();//加锁                
                if (ticketNum<=0) {
                    flag=false;
                    return;            
                }                
                System.out.println(Thread.currentThread().getName()+"买到了第:"+ticketNum-- +"张票");                
                    
            } catch (Exception e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }finally {
                lock.unlock();//解锁
            }
        }        
    }
}

运行结果:线程3买到了第:10张票
线程2买到了第:9张票
线程1买到了第:8张票
线程2买到了第:7张票
线程3买到了第:6张票
线程1买到了第:5张票
线程3买到了第:4张票
线程2买到了第:3张票
线程1买到了第:2张票
线程2买到了第:1张票

但是 把线程休眠的时间放在 加锁Lock.lock()后,运行的结果,只有一个线程在运行买票
代码

package com.example.demo.therad.synchronizedThread;
//买票
import java.util.concurrent.locks.ReentrantLock;
public class BuyTicket  implements Runnable{
   private int ticketNum=10;//票数   
   Boolean flag=true;//线程外部结束标志  
   private final ReentrantLock lock=new ReentrantLock();    
    public void run() {
        while(flag) {        
            try {            
                lock.lock();//加锁                
                if (ticketNum<=0) {
                    flag=false;
                    return;            
                }
                Thread.sleep(1000);
                System.out.println(Thread.currentThread().getName()+"买到了第:"+ticketNum-- +"张票");                                    
            } catch (Exception e) {
                // TODO Auto-generated catch block
                e.printStackTrace();
            }finally {
                lock.unlock();//解锁
            }
        }
        
    }
}

结果
线程2买到了第:10张票
线程2买到了第:9张票
线程2买到了第:8张票
线程2买到了第:7张票
线程2买到了第:6张票
线程2买到了第:5张票
线程2买到了第:4张票
线程2买到了第:3张票
线程2买到了第:2张票
线程2买到了第:1张票

求解答一下疑问.. 感谢

  • 写回答

1条回答 默认 最新

  • 皮蛋不吃粥 2022-07-13 09:37
    关注

    代码二
    概率事件并不是每次都是线程2一直买到,把声明的lock改为公平锁可以看看,其实把票数加大也会发现,不是一直线程2

    本回答被题主选为最佳回答 , 对您是否有帮助呢?
    评论

报告相同问题?

问题事件

  • 系统已结题 7月21日
  • 已采纳回答 7月13日
  • 创建了问题 7月12日

悬赏问题

  • ¥15 c程序不知道为什么得不到结果
  • ¥40 复杂的限制性的商函数处理
  • ¥15 程序不包含适用于入口点的静态Main方法
  • ¥15 素材场景中光线烘焙后灯光失效
  • ¥15 请教一下各位,为什么我这个没有实现模拟点击
  • ¥15 执行 virtuoso 命令后,界面没有,cadence 启动不起来
  • ¥50 comfyui下连接animatediff节点生成视频质量非常差的原因
  • ¥20 有关区间dp的问题求解
  • ¥15 多电路系统共用电源的串扰问题
  • ¥15 slam rangenet++配置