
在double-choco谜题中,棋盘由一个个单元格组成,这些单元格可以是白色或灰色。为了有效地表示棋盘状态和块的边界,我们采用一个2d数组来存储自定义的cell对象。每个cell对象不仅包含其在棋盘上的坐标,还承载了颜色、数字(如果适用)、边界信息以及是否已被分配到某个块的状态。
一个cell对象的核心属性定义如下:
let cell = {
x: Number, // 单元格的X坐标
y: Number, // 单元格的Y坐标
color: "white" | "gray", // 单元格的颜色
number: null | Number, // 如果有数字提示,表示该颜色区域的单元格数量
top: true | false, // 顶部是否有边界线 (true: 有边界, false: 无边界)
bottom: true | false, // 底部是否有边界线
left: true | false, // 左侧是否有边界线
right: true | false, // 右侧是否有边界线
taken: false, // 是否已被分配到某个完成的谜题块中
blockId: null // 所属谜题块的唯一ID,或存储整个块的引用
};关键属性解析:
在谜题生成过程中,我们需要能够根据已定义的边界线(即cell对象的top/bottom/left/right属性)来识别和提取独立的谜题块。这可以通过一个递归的洪水填充(Flood-Fill)算法来实现。
该算法从一个未被标记的单元格开始,递归地访问所有与其相连(即之间没有边界线)的相邻单元格,直到遇到边界线或已访问过的单元格。所有被访问到的单元格共同构成一个完整的谜题块。
/**
* 递归地提取一个连通的谜题块。
* @param {Array<Array<cell>>} cells - 整个棋盘的2D单元格数组。
* @param {cell} currentCell - 当前正在处理的单元格。
* @param {Array<cell>} currentBlock - 用于存储当前块中所有单元格的数组。
* @param {number} blockId - 当前块的唯一标识符。
*/
function extractBlock(cells, currentCell, currentBlock, blockId) {
// 边界检查:确保单元格在棋盘范围内
if (!currentCell || currentCell.taken) {
return;
}
currentCell.taken = true; // 标记为已访问/已分配
currentCell.blockId = blockId; // 分配块ID
currentBlock.push(currentCell); // 将单元格添加到当前块中
const { x, y } = currentCell;
const rows = cells.length;
const cols = cells[0].length;
// 向上移动:如果顶部没有边界线且上方单元格存在
if (!currentCell.top && y > 0) {
extractBlock(cells, cells[y - 1][x], currentBlock, blockId);
}
// 向下移动:如果底部没有边界线且下方单元格存在
if (!currentCell.bottom && y < rows - 1) {
extractBlock(cells, cells[y + 1][x], currentBlock, blockId);
}
// 向左移动:如果左侧没有边界线且左侧单元格存在
if (!currentCell.left && x > 0) {
extractBlock(cells, cells[y][x - 1], currentBlock, blockId);
}
// 向右移动:如果右侧没有边界线且右侧单元格存在
if (!currentCell.right && x < cols - 1) {
extractBlock(cells, cells[y][x + 1], currentBlock, blockId);
}
}
/**
* 遍历整个棋盘,提取所有独立的谜题块。
* @param {Array<Array<cell>>} cells - 整个棋盘的2D单元格数组。
* @returns {Array<Array<cell>>} 所有提取出的谜题块的数组。
*/
function findAllBlocks(cells) {
const allExtractedBlocks = [];
let nextBlockId = 1;
for (let y = 0; y < cells.length; y++) {
for (let x = 0; x < cells[0].length; x++) {
const currentCell = cells[y][x];
if (!currentCell.taken) {
const newBlock = [];
extractBlock(cells, currentCell, newBlock, nextBlockId++);
if (newBlock.length > 0) {
allExtractedBlocks.push(newBlock);
}
}
}
}
return allExtractedBlocks;
}工作原理:
以上就是生成Double-Choco谜题:高效数据结构与算法实践的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号