// 21点游戏 分为人机对战和人人对战
// 玩家每次抽一张牌 牌的点数为1-10点随机数 谁更接近21点谁就获胜
let readline = require("readline-sync");
// 清屏函数
let clear = () => process.stdout.write(process.platform === 'win32' ? '\x1Bc' : '\x1B[2J\x1B[3J\x1B[H');
// 抽牌函数 接收一个参数 玩家对象
let takeCard = function (player) {
let point = Math.floor(Math.random() * 10 + 1); // 随机获取一个点数
console.log(`你抽取到的点数为:${point}`);
player.titlePoint += point; // 改变玩家对象的总点数
}
// 结果比较函数 接收2个参数 两个玩家对象
let judge = function (player1, player2) {
if (player1.titlePoint > player2.titlePoint) {
return 1;
}
else if (player1.titlePoint < player2.titlePoint) {
return 2;
}
else {
return 0;
}
}
// 主函数
let main = function () {
let useSys = true; // 是否使用系统 默认值true代表一直使用
while (useSys) {
clear();
console.log("欢迎来到21点的世界");
console.log("请选择游戏选项: 1.单人模式 2.双人模式 3.游戏说明 4.退出游戏");
let funcSelect = parseInt(readline.question(""));
switch (funcSelect) {
// 单人模式
case 1:
{
// 首先初始化两个对象,一个玩家,一个电脑
let player = { victory: 0, titlePoint: 0 };
let computer = { victory: 0, titlePoint: 0 };
let continuePlay = true; // 是否继续单人模式
while (continuePlay) {
let isTakeCard = true; // 玩家是否有资格抽取下一张牌 默认是true 有资格抽取
// 玩家抽牌的循环
while (isTakeCard) {
clear();
console.log(`当前的总点数为:${player.titlePoint}`);
console.log("是否要抽取一张牌?(Y/N)");
let isTake = readline.question("");
switch (isTake) {
case "Y":
case "y":
takeCard(player); // 用户已经抽取了一张牌,里面有点数了
// 玩家每抽取一张牌,都要进行是否超过21点的判断 如果超过21点,就不能继续抽牌了
// 直接将isTakeCard值改为false
if (player.titlePoint > 21) {
console.log("你已经爆炸了!");
readline.question("");
isTakeCard = false;
break;
}
console.log("\n按回车键继续");
readline.question("");
break;
case "N":
case "n":
isTakeCard = false;
break;
default:
console.log("输入有误,请重新输入");
readline.question("");
}
}
// 如果退出了上面的while,说明两种情况,一是玩家爆了,二是玩家主动退出来的
// 所以我们首先判断玩家的总点数是否大于21
if (player.titlePoint > 21) {
console.log("电脑获胜");
computer.victory++;
}
else {
// 如果没有进入上面的if,说明玩家是主动退出来的
// 那么我们就让电脑对象生成总点数,然后将两个对象传入judge函数进行比较
computer.titlePoint = Math.floor(Math.random() * 7 + 15); // 电脑随机的总点数为15-21的随机数
console.log(`你的总点数为${player.titlePoint},电脑的总点数为${computer.titlePoint}`);
let result = judge(player, computer);
// 根据judge函数返回的结果来显示不同的信息
switch (result) {
case 0:
console.log("本局为平局,不加分");
break;
case 1:
console.log("玩家获胜");
player.victory++;
break;
case 2:
console.log("电脑获胜");
computer.victory++;
break;
}
}
// 打印信息
console.log("\n当前比分如下:");
console.log(`玩家:${player.victory} vs 电脑:${computer.victory}`);
readline.question("");
// 询问玩家是否要继续单人模式 如果不继续了,直接将continuePlay修改为false
while (true) {
console.log("是否继续游戏?(Y/N)");
let isAgain = readline.question("");
// 如果要继续单人模式 那么需要将两个对象的总点数重置为0
if (isAgain === 'Y' || isAgain === 'y') {
player.titlePoint = 0;
computer.titlePoint = 0;
break;
}
else if (isAgain === 'N' || isAgain === 'n') {
continuePlay = false;
break;
}
else {
console.log("输入有误,请重新输入\n");
}
}
}
// 能够来到这里,说明玩家输入的n,这时continuePlay的值为false
// 打印最终结果即可
clear();
console.log("最终比分如下:");
console.log(`玩家:${player.victory} vs 电脑:${computer.victory}`);
console.log("\n按回车键返回");
readline.question("");
}
break;
// 双人模式
case 2:
{
// 和单人游戏模式一样,还是初始化两个对象,不过这两个对象多了一个isTakeCard属性
let player1 = { victory: 0, titlePoint: 0, isTakeCard: true };
let player2 = { victory: 0, titlePoint: 0, isTakeCard: true };
// continuePlay变量表示是否继续双人模式
let continuePlay = true;
while (continuePlay) {
// 根据两个玩家对象的isTakeCard属性值判断是否抽牌,只要两个对象有一个属性值为true就会进入while循环
while (player1.isTakeCard || player2.isTakeCard) {
// 根据玩家1的isTakeCard属性值来判断玩家1是否抽牌
if (player1.isTakeCard) {
let inputValidate = true; // 此变量为输入验证 用于后面验证玩家的输入
while (inputValidate) {
clear();
// 在抽牌之前首先打印当前的点数信息,然后开始询问玩家是否抽牌
console.log("总点数:")
console.log(`玩家A:${player1.titlePoint} vs 玩家B:${player2.titlePoint}`);
console.log("玩家A 是否抽牌?(Y/N)");
let isTake = readline.question("");
switch (isTake) {
case "Y":
case "y":
inputValidate = false; // 做输入验证
takeCard(player1); // 抽一张牌
readline.question("");
// 大于21,直接将两个玩家的isTakeCard都修改为false,这样可以退出上上个while
if (player1.titlePoint > 21) {
player1.isTakeCard = false;
player2.isTakeCard = false;
}
break;
case "N":
case "n":
// 代表玩家不抽牌了,将isTakeCard属性改为false
inputValidate = false;
player1.isTakeCard = false;
break;
default:
console.log("输入有误,请重新输入");
readline.question("");
}
}
}
// 和上面相同
if (player2.isTakeCard) {
let inputValidate = true;
while (inputValidate) {
clear();
console.log("总点数:")
console.log(`玩家A:${player1.titlePoint} vs 玩家B:${player2.titlePoint}`);
console.log("玩家B 是否抽牌?(Y/N)");
let isTake = readline.question("");
switch (isTake) {
case "Y":
case "y":
inputValidate = false;
takeCard(player2);
readline.question("");
if (player2.titlePoint > 21) {
player1.isTakeCard = false;
player2.isTakeCard = false;
}
break;
case "N":
case "n":
inputValidate = false;
player2.isTakeCard = false;
break;
default:
console.log("输入有误,请重新输入");
readline.question("");
}
}
}
}
// 代码来到这里说明两个玩家的isTakeCard都为false,接下来直接进行比较即可
if (player1.titlePoint > 21) {
console.log("玩家A已爆炸!");
console.log("玩家B获胜");
player2.victory++;
}
else if (player2.titlePoint > 21) {
console.log("玩家B已爆炸!");
console.log("玩家A获胜");
player1.victory++;
}
else {
let result = judge(player1, player2);
switch (result) {
case 0:
console.log("本局为平局,不加分");
break;
case 1:
console.log("玩家A获胜");
player1.victory++;
break;
case 2:
console.log("玩家B获胜");
player2.victory++;
break;
}
}
console.log("\n当前比分如下:");
console.log(`玩家A:${player1.victory} vs 玩家B:${player2.victory}`);
readline.question("");
while (true) {
console.log("是否继续游戏?(Y/N)");
let isAgain = readline.question("");
if (isAgain === 'Y' || isAgain === 'y') {
player1.titlePoint = 0;
player2.titlePoint = 0;
player1.isTakeCard = true;
player2.isTakeCard = true;
break;
}
else if (isAgain === 'N' || isAgain === 'n') {
continuePlay = false;
break;
}
else {
console.log("输入有误,请重新输入\n");
}
}
}
clear();
console.log("最终比分如下:");
console.log(`玩家A:${player1.victory} vs 玩家B:${player2.victory}`);
console.log("\n按回车键返回");
readline.question("");
}
break;
// 游戏说明
case 3:
{
clear();
console.log("单人模式规则:");
console.log("单人模式下玩家可以连续抽牌,每回合抽取的点数为1-10点中任意一个点数");
console.log("所有点数相加不能超过21点,然后总点数和电脑点数进行比较,谁更接近21点谁就算胜利\n");
console.log("双人模式规则:");
console.log("两名玩家轮流抽牌,在不超过21点的前提下进行总点数的比较,谁更接近21点谁就算胜利\n");
console.log("按回车键返回");
readline.question("");
}
break;
// 退出游戏
case 4:
{
// 要退出游戏 直接将useSys修改为false
useSys = false;
}
break;
}
}
clear();
console.log("Thank you for playing");
}
main();

JavaScript中纯JS写21点游戏的更多相关文章

  1. 纯JS写的2048游戏,分享之

    这几天玩儿着2048这个游戏,突然心血来潮想练习下敲代码的思路.于是乎就模仿做了一个,到眼下位置还没有实现动态移动,不是非常好看,只是玩儿着自己模仿的小游戏还是蛮爽的,哈哈 假设没有玩儿过这个游戏,最 ...

  2. javascript中外部js文件取得自身完整路径得办法

    原文:javascript中外部js文件取得自身完整路径得办法 有时候我们需要引入一个外部js文件,这个js文件又需要用到自己的路径或者是所在的目录,别问怎么又这么变态的需求,开发做久了各种奇葩需求也 ...

  3. 纯JS单页面赛车游戏代码分享

    分享一个以前写的小游戏,纯js游戏,代码很简单.欢迎大家来喷呦! 效果图: 代码展示://直接复制到html文件即可 支持IE9+版本 <!DOCTYPE html> <html&g ...

  4. 纯js写图片轮播插件

    最近终于写成了自己创作的图片轮播插件,使用原生js编写.与目前网上流行的轮播插件相比,功能和效果稍弱,但是使用起来相当方便. 先看html代码 <!DOCTYPE html> <ht ...

  5. 使用纯js写的一个分页

    上图晒效果: 网上确实有很多分页的插件以及开源代码,单本是一个后台开发猿,前台css等样式还驾驭不住,所以就开始自己去写了.其实这个分页原理很简单,就是用ajax往后台传值(当前页码),后台使用lim ...

  6. 用html5 canvas和JS写个数独游戏

    为啥要写这个游戏? 因为我儿子二年级数字下册最后一章讲到了数独.他想玩儿. 因为我也想玩有提示功能的数独. 因为我也正想决定要把HTML5和JS搞搞熟.熟悉一个编程平台,最好的办法,就是了解其原理与思 ...

  7. 纯JS写最简单的图片轮播

    非常简单的一个大图轮播,通过将控制显示位置来进行轮播效果,写来给正在学习的新手朋友们参考交流. 先看效果:(实际效果没有这么快) 先看布局: <div id="display" ...

  8. 纯JS写出日历

    封装代码: (function(cmf){ cmf.showDcalendar=function(){ var fnname=cmf.fn var id="cmfrili2" va ...

  9. 纯JS写动态分页样式效果

    效果图如下: html: <body> <div> <table id="btnbox"> <tbody> <tr>&l ...

随机推荐

  1. 《提高c++性能的编程技术》读书笔记

    一个程序的执行效率是取决于改程序翻译成汇编语言之后的执行的机器指令的条数.而每一个机器指令的执行的周期是一定的.C语言和C++都是高于汇编语言的高级语言,其中,C语言源代码与其相应的机器指不是完全同一 ...

  2. wc - 输出文件中的字节数、单词数、行数

    SYNOPSIS 总览 wc [选项列表]... [文件名列表]... DESCRIPTION 描述 对每个文件输出行.单词.和字节统计数,如果指定了多于一个文件则还有一个行数的总计.没有指定文件或指 ...

  3. Linux fork创建子进程

    1.  pid_t fork(void); 功能:创建父子进程 参数:无 返回值:成功:在父进程中:返回值为子进程的PID 在子进程中:返回值为0 失败:-1 注意: 1)fork函数是用来创建进程的 ...

  4. Mac电脑最常见的办公软件是什么?Notion for Mac多功能办公笔记软件使用方法

    Notion for Mac是一款最新的高效率.办公类软件,相信许多用户在办公的时候需要打开特别多的在线工具,譬如Google Drive.Dropbox Paper.Confluence.GitHu ...

  5. Batch - %~dp0 vs %cd%

    总结 %~dp0 只表示将要“运行的”bat命令的folder,不包含bat自己的名称. %cd%表示,“运行处”的folder . 示例脚本内容 cd-dp0.bat存放在f盘 @echo off ...

  6. Magento获取当前页面URL地址

    Magento获取当前页面URL地址 http://www.sunhaibing.com/?p=1260 在Magento中,可以通过core/url助手类中的getCurrentUrl()方法获取当 ...

  7. 阿里巴巴持续投入,etcd 正式加入 CNCF

    摘要: 2018 年 12 月 11 日,在 KubeCon + CloudNativeCon 北美峰会上,etcd 项目正式加入 CNCF. 2018 年 12 月 11 日,在 KubeCon + ...

  8. sqoop简介和原理分析

    Sqoop简介 Sqoop是一款开源的工具,主要用于在Hadoop(Hive)与传统的数据库(mysql.postgresql...)间进行数据的传递,可以将一个关系型数据库(例如 : MySQL , ...

  9. go beego

    一. 引入 go get github.com/astaxie/beego go get gitgub.com/beego/bee go get -u gitxxx.... 更新框架 编写 packa ...

  10. Qt无边框窗口的移动、拉伸边框、鼠标滚轮缩放大小

    主要是处理窗口上鼠标的几个事件,具体代码请看下面的截图, 完整代码的下载链接在此:http://download.csdn.net/detail/beyond0824/9657110, 本示例代码中, ...