用node.js怎样做命令行游戏?实现原理是什么?
Admin 2021-10-19 群英技术资讯 1358 次浏览
用node.js怎样做命令行游戏?其实要实现一个命令行游戏并不困难,这篇文章就以实现贪吃蛇游戏为例,给大家分享一下实现命令行游戏的原理及实现代码,感兴趣的朋友可以参考。

命令行输入
画面渲染
使用过 yarn upgrade-interactive 命令更新 npm 依赖, 或者使用过 vue-cli 等脚手架创建过新项目的同学应该都见过: 这些工具会在命令行输出很多选项, 通过上下按键可以移动焦点, 通过空格键可以选择
那么这些操作是如何实现的呢? 下面通过 readline 和 process.stdin 来实现命令行监听按键事件:
process.stdin 是一个可读流, 通过 readline.emitKeypressEvents 可以给可读流注册 keypress 事件, 通过 keypress 事件就能获取到按键的值
readline.emitKeypressEvents(process.stdin) // 注册 keypress 事件
process.stdin.setRawMode(true) // 开启原始模式, 使输入的每个字符带上各种详细属性
process.stdin.on('keypress', (...args) => {
console.log(args)
// 按下方向键会输出
// [
// undefined,
// {
// sequence: '\u001b[A',
// name: 'up',
// ctrl: false,
// meta: false,
// shift: false,
// code: '[A'
// }
// ]
})
注意: setRawMode 会使命令行按下 ctrl + c 不再发送终止信号, 可能需要自行处理退出逻辑
输出到命令行的游戏画面默认为 30 行 x 50 列, 将其划分为一个二维数组, 每隔一段时间将二维数组的值打印出来并擦除之前打印的值, 即完成一次帧画面的渲染
process.stdout 是一个可写流, 调用 process.stdout.write 可以向命令行写入数据, nodejs 中 console.log 其实就是将数据写入到 process.stdout 并换行
通过向命令行写入开头为 ANSI 转义序列 的字符串可以 光标移动/滚动屏幕/擦除显示/颜色文本 等等功能, 想要深入了解可以自行搜索关键字学习, 本文使用 ansi-escapes npm 包实现擦除功能
const ansiEscapes = require('ansi-escapes')
function clear(lines) {
process.stdout.write(ansiEscapes.eraseLines(lines)) // 可以擦除指定行数的输出
}
根据游戏画面的宽高定义一个二维数组, 小蛇的头和身体视为画面中的点, 值为非空值, 空白画面则为空字符串
let dots = []
for (let col = 0; col < wall.height; col++) {
dots[col] = new Array(wall.width).fill(' ')
}
在每一帧中, 小蛇的头会向前进的方向前进一个, 头接着的第一节身体则会移动到上一帧头所在的位置, 以此类推每一节身体都会移动到前一节身体的位置上, 所以需要定义一个数据记录之前的头和身体的位置
const SNAKE_HEAD = '@' // 头的符号
const SNAKE_BODY = '○' // 身体的符号
function drawFrame() {
let dots = []
for (let col = 0; col < wall.height; col++) {
dots[col] = new Array(wall.width).fill(' ')
}
let nextBody = []
let head = next(snake.body[0]) // next 方法传入当前点的 x, y 坐标, 返回向前进方向前进一个的 x, y 坐标
nextBody.push(head)
dots[head.y][head.x] = SNAKE_HEAD
for (let i = 1; i < snake.length; i++) {
let body = snake.body[i - 1]
dots[body.y][body.x] = SNAKE_BODY
nextBody.push(body)
}
screen.draw(dots) // 将二维数组中的点输出到命令行中
// 更新蛇的状态
snake.body = nextBody
snake.head = snake.body[0]
}
小蛇每吃到一个鸟蛋, 身体会长一节, 并在画面中随机生成另一个鸟蛋. 到了这一步其实就很简单了, 随机生成一个点作为鸟蛋的位置, 插入到之前的二维数组中.
function layAEgg() {
let x = ~~(wall.width * Math.random())
let y = ~~(wall.height * Math.random())
return { x, y }
}
当小蛇的头的位置与鸟蛋的位置相同时, 则视为蛇吃到鸟蛋, 蛇的长度加一, 并在尾部增加一节上一帧蛇尾的节点位置
const SNAKE_HEAD = '@'
const SNAKE_BODY = '○'
const BIRD_EGG = '●'
function drawFrame() {
let dots = []
for (let col = 0; col < wall.height; col++) {
dots[col] = new Array(wall.width).fill(' ')
}
let nextBody = []
let head = next(snake.body[0])
nextBody.push(head)
dots[head.y][head.x] = SNAKE_HEAD
for (let i = 1; i < snake.length; i++) {
let body = snake.body[i - 1]
dots[body.y][body.x] = SNAKE_BODY
nextBody.push(body)
}
// 判断蛇头位置在上一帧中是否为鸟蛋位置, 为真视为蛇吃到鸟蛋
if (prevDots && prevDots[head.y][head.x] === BIRD_EGG) {
let body = snake.body[snake.length - 1]
dots[body.y][body.x] = SNAKE_BODY
nextBody.push(body)
snake.length += 1
egg = null
prevDots = null
}
if (!egg) {
egg = layAEgg()
while (dots[egg.y][egg.x] !== ' ') {
egg = layAEgg()
}
}
dots[egg.y][egg.x] = BIRD_EGG
prevDots = dots // 保存上一帧的数据, 用于下次绘制时判断逻辑
screen.draw(dots)
snake.body = nextBody
snake.head = snake.body[0]
}
至此, 命令行贪吃蛇游戏基本逻辑都已实现, 剩下的就是使用定时器每隔一段时间绘制一次帧画面. 其实几乎任何像素游戏(如俄罗斯方块/吃豆人等)都可以按照这个流程实现, 不同的只是帧画面的处理逻辑而已. 如果感兴趣的话, 可以去我的 github 查看该 贪吃蛇游戏源码
以上就是用node.js怎样做命令行游戏的相关的介绍啦,上述贪吃蛇的示例具有一定的借鉴价值,有需要的朋友可以参考学习,希望对大家有帮助,想要了解更多可以继续浏览群英网络其他相关的文章。
文本转载自脚本之家
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:mmqy2019@163.com进行举报,并提供相关证据,查实之后,将立刻删除涉嫌侵权内容。
猜你喜欢
这篇文章主要介绍了Vue3之 页面,菜单,路由的使用,文章围绕Vue3页面,菜单,路由相关资料展开详细内容,需要的朋友可以参考一下
在 React中是不存在双向数据绑定的机制的,需要我们自己对其进行实现。本文主要介绍一下React双向数据绑定,感兴趣的可以了解一下
这篇文章给大家详细的介绍vue的diff算法,对很多新手来说vue的diff算法是比较难理解的内容,下面小编就分享一些实例,对大家学习vue的diff算法有一定的帮助,接下来一起跟随小编看看吧。
本文主要给大家分享的是Javascript建造者模式,建造者模式是设计模式中的一种,之前我们也了解其他的设计模式,感兴趣的朋友可以了解看看,接下来我们就看Javascript建造者模式的定义、优点、作用和实现等等。
这篇文章主要介绍了angularjs循环对象属性实现动态列的思路详解,本文给大家分享一个demo代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
成为群英会员,开启智能安全云计算之旅
立即注册Copyright © QY Network Company Ltd. All Rights Reserved. 2003-2020 群英 版权所有
增值电信经营许可证 : B1.B2-20140078 粤ICP备09006778号 域名注册商资质 粤 D3.1-20240008