吾爱破解 - 52pojie.cn

 找回密码
 注册[Register]

QQ登录

只需一步,快速开始

查看: 1315|回复: 22
收起左侧

[求助] java多线程问题,求解答

[复制链接]
黑白客 发表于 2021-3-4 14:44
本帖最后由 黑白客 于 2021-3-4 14:50 编辑
package com.wang.syn1;

/**
 * @author: 王海新
 * @Date: 2021/2/28 16:40
 * @Description:  修改为安全的买票,
 * 添加synchronized 锁 在buy方法上。就等于在buy的对象上设置了锁。
 * 因为buy里面有延时,所以会让先进来的线程一直调用buy。如果票少,其它的线程就没有机会
 *可以将延时放到run方法中,这样其它在buy执行完
 */
public class SafeBuyTicket {
    public static void main(String[] args) {
        BuyTicket buyTicket = new BuyTicket();

        new Thread(buyTicket,"小明").start();
        new Thread(buyTicket,"小红").start();
        new Thread(buyTicket,"小芳").start();
        new Thread(buyTicket,"小蓝").start();
    }
}

class BuyTicket implements Runnable{

    //票
    private int ticketNums = 10;
    boolean flage = true;//外部停止方法

    @Override
    public void run() {
        //买票
        while (flage) {
            try {
                Thread.sleep(1000);
                buy();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
//添加synchronized 锁
    private synchronized void buy() throws InterruptedException {
        //判断是否有票
        if (ticketNums <= 0) {
            flage = false;
            return;
        }
        //模拟延时
        //Thread.sleep(1000);
        //买票
        System.out.println(Thread.currentThread().getName() + "拿到了第" + ticketNums -- +"张");
    }
}

问题描述

在上面的代码中,我在buy上加了synchronized。但是每次运行都是一个人把票全拿了。
如果把票数调到1000,就会有其他人拿到。

请问这是为什么?

--

我在网上看了很久,有人说
添加synchronized 锁 在buy方法上。就等于在buy的对象上设置了锁。
因为buy里面有延时(sleep),所以会让先进来的线程一直调用buy。如果票少,其它的线程就没有机会
可以将延时放到run方法中,这样其它在buy执行完。

问题二:一个线程已经进入到里面sleep了,怎么还会调用buy呢?

----------
还有人说把sleep放到run方法里就好了。

问题三:

如果sleep加到run中,如果第一个线程先进去买票了,那么剩下的线程跑sleep然后等待,等到第一个线程跑完,跑sleep的时候。其它线程就开始抢占锁了。那是不是以后的每次竞争。对刚出来的线程都是不利的。

另外sleep放到了buy里面的时候,虽然第一个进去的线程在里面抱着锁睡觉。但是出来之后,还是要平等竞争啊。为什么在票数少的情况下。还是这一个锁拿到所有的票

发帖前要善用论坛搜索功能,那里可能会有你要找的答案或者已经有人发布过相同内容了,请勿重复发帖。

yaojixin 发表于 2021-3-4 14:47
偏向锁了解下
 楼主| 黑白客 发表于 2021-3-4 14:51
hualonghongyan 发表于 2021-3-4 14:57
ygh0309 发表于 2021-3-4 15:16
本帖最后由 ygh0309 于 2021-3-4 15:18 编辑

wait 会释放锁,sleep 睡觉了,抱着锁睡觉,不会释放!,还有就是买票的线程没必要睡觉吧
小年轻在奋斗 发表于 2021-3-4 15:35
本帖最后由 小年轻在奋斗 于 2021-3-4 15:36 编辑

把buy() 里面的sleep换成wait,sleep没有释放对象锁,wait会释放对象锁在等待时间过去后重新进入等待队列,进行竞争;
在修改这个wait之后,你应该还要在扣票处加上判断,不然容易出现负数票出现
maojiuming 发表于 2021-3-4 15:50
看了大家的讲解,学习了
 楼主| 黑白客 发表于 2021-3-4 15:55

将sleep关闭掉吗?但是我还是不理解上面的问题
 楼主| 黑白客 发表于 2021-3-4 15:57
ygh0309 发表于 2021-3-4 15:16
wait 会释放锁,sleep 睡觉了,抱着锁睡觉,不会释放!,还有就是买票的线程没必要睡觉吧

这只是举个栗子。就是它抱着锁睡觉了。那为什么只有十张票的时候。每次都是只有一个人全部取走票呀
 楼主| 黑白客 发表于 2021-3-4 15:59
小年轻在奋斗 发表于 2021-3-4 15:35
把buy() 里面的sleep换成wait,sleep没有释放对象锁,wait会释放对象锁在等待时间过去后重新进入等待队列, ...

就是,我不理解为什么它抱着锁睡觉了。已经睡觉了。为什么在只有十张票的情况下,他还是会全部拿到
您需要登录后才可以回帖 登录 | 注册[Register]

本版积分规则

返回列表

RSS订阅|小黑屋|处罚记录|联系我们|吾爱破解 - LCG - LSG ( 京ICP备16042023号 | 京公网安备 11010502030087号 )

GMT+8, 2024-11-26 06:35

Powered by Discuz!

Copyright © 2001-2020, Tencent Cloud.

快速回复 返回顶部 返回列表