JS使用普通队列实现击鼓传花游戏

167 篇文章 3 订阅

最近复习到了数据结构中的普通队列部分,来实现一个击鼓传花游戏的应用。

循环队列的一个例子就是击鼓传花(hot potato),在这个游戏中,孩子们围成一个圆圈,把花尽快地传递给旁边的人。某一时刻传话停止,这个时候花在谁手里,谁就退出圆圈、结束游戏。重复这个过程,直到只剩一个孩子(胜者)

首先我们需要实现队列这个数据结构。

class Queue {
    constructor() {
        // count属性控制队列的大小
        this.count = 0;
        // 追踪第一个元素
        this.lowestCount = 0;
        // 首先我们需要一个用于存储队列中元素的数据结构,我们可以使用数组,就像Stack一样.
        // 但是,为了写出一个在获取元素时更高效的数据结构,我们将使用一个对象来存储我们的元素
        this.items = {};        
    }

    enqueue(element) {
        this.items[this.count] = element;
        this.count++;
    }

    dequeue() {
        if(this.isEmpty()) {
            return undefined;
        }
        const result = this.items[this.lowestCount];
        delete this.items[this.lowestCount];
        this.lowestCount++;
        return result;
    }
    isEmpty() {
        return this.count - this.lowestCount === 0;
    }
    peek() {
        if(this.isEmpty()) {
            return undefined;
        }
        return this.items[this.lowestCount];
    }
    size() {
        return this.count - this.lowestCount;
    }
    clear() {
        this.items = {};
        this.count = 0;
        this.lowestCount = 0;
    }
    toString() {
        if(this.isEmpty()) {
            return '';
        }
        let objString = `${this.items[this.lowestCount]}`;
        for(let i = this.lowestCount + 1; i < this.count; i++) {
            objString = `${objString},${this.items[i]}`;
        }
        return objString;
    }
}

然后说一下通过队列实现这个游戏的思路,先把所有人添加到队列中,然后循环遍历队列,从队列头部移除一项,再将其添加到队列末尾,模拟击鼓传花的过程,一旦达到给定的传递次数,拿着花的那个人就被淘汰。

function hotPotato(elementList,num) {
    const queue = new Queue();
    const elimitatedList = [];
    // 把名单的名字全都加入队列中
    for(let i = 0; i < elementList.length; i++) {
        queue.enqueue(elementList[i]);
    }
    while(queue.size() > 1) {
        for(let i = 0; i < num; i++) {
            // 从队列开头移除一项,再将其添加到队列末尾,模拟击鼓传花
            // (如果你把花传给了旁边的人,你被淘汰的威胁就立即解除了)。
            // 一旦达到给定的传递次数,拿着花的那个人就被淘汰了
            queue.enqueue(queue.dequeue());
        }
        // 被淘汰,从队列中移除
        elimitatedList.push(queue.dequeue());
    }
    return {
        elimitatedList:elimitatedList,
        winner:queue.dequeue()
    };
}

const names = ['John','Jack','Camila','Ingrid','Carl'];
const result = hotPotato(names,7);
result.elimitatedList.forEach(item => {
    console.log(`${item}在击鼓传花游戏中被淘汰。`);
})

console.log(`胜利者:${result.winner}`);

在这里插入图片描述
通过测试用例可以看到结果是正确的。

有任何问题都可以加我联系方式沟通交流。

QQ:505417246
微信:18331092918
微信公众号:Code程序人生
个人博客:https://Creator12333.github.io(详细总结一些面试题,以及前端进阶的内容,CSDN写偏基础的内容)

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

CreatorRay

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值