
一、核心数据结构:单元格表示
在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,或存储整个块的引用
};关键属性解析:
- x, y: 单元格的二维坐标,方便定位。
- color, number: 用于表示谜题的特定规则,即白/灰区域及其大小提示。
- top, bottom, left, right: 这四个布尔值是定义谜题块边界的关键。当值为true时,表示该方向存在一条实线,将当前单元格与相邻单元格分隔开;当值为false时,表示该方向没有实线,当前单元格与相邻单元格相连,属于同一个连通区域。
- taken: 在生成过程中,用于标记已被成功分配到某个合法谜题块的单元格,避免重复处理。
- blockId: 用于在提取块后,将所有属于同一块的单元格关联起来。
二、块提取算法:基于边界的连通组件识别
在谜题生成过程中,我们需要能够根据已定义的边界线(即cell对象的top/bottom/left/right属性)来识别和提取独立的谜题块。这可以通过一个递归的洪水填充(Flood-Fill)算法来实现。
网趣购物系统静态版支持网站一键静态生成,采用动态进度条模式生成静态,生成过程更加清晰明确,商品管理上增加淘宝数据包导入功能,与淘宝数据同步更新!采用领先的AJAX+XML相融技术,速度更快更高效!系统进行了大量的实用性更新,如优化核心算法、增加商品图片批量上传、谷歌地图浏览插入等,静态版独特的生成算法技术使静态生成过程可随意掌控,从而可以大大减轻服务器的负担,结合多种强大的SEO优化方式于一体,使
该算法从一个未被标记的单元格开始,递归地访问所有与其相连(即之间没有边界线)的相邻单元格,直到遇到边界线或已访问过的单元格。所有被访问到的单元格共同构成一个完整的谜题块。
/**
* 递归地提取一个连通的谜题块。
* @param {Array>} cells - 整个棋盘的2D单元格数组。
* @param {cell} currentCell - 当前正在处理的单元格。
* @param {Array} 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>} cells - 整个棋盘的2D单元格数组。
* @returns {Array>} 所有提取出的谜题块的数组。
*/
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;
} | 工作原理:
- extractBlock函数是核心的递归函数。它接收当前单元格、一个用于累积当前块单元格的数组以及块ID。
- 在









