只需检查水平、垂直、主对角线(\)、副对角线(/)4个方向,以落子点为中心各查前后2格共5点,逐方向统计连续同色长度,边界须显式判断,坐标系统一为boardi(i行j列),平局检测应在赢棋检测之后进行。

判断五子连珠时,别直接遍历全部8个方向
二维数组里找连续5个相同棋子,最容易卡在“方向太多、重复检查、边界越界”上。实际只需检查4个方向:水平、垂直、主对角线(\)、副对角线(/),因为反向是同一组连线——比如从(i,j)往右下查5个,和从(i+4,j+4)往左上查5个,判的是同一条线。
实操建议:
- 每次落子后,只以该位置为中心,朝这4个方向分别延伸检查(各查2格前 + 2格后,共5点),避免全盘扫描
- 每个方向用独立的
for循环,起始偏移量设为-2到+2,但必须加边界判断:if x = board.length || y = board[0].length - 别用
try/catch捕获越界来简化逻辑——JS里数组越界返回undefined,可能误判成空位;Python会抛IndexError,但性能差且掩盖问题
checkWin(board, row, col, player)函数里,计数器不能重置太早
常见错误是:一遇到不同值就清零计数器,结果○○●●●这种跨中点的连珠被漏掉。关键在于——要先往一个方向走到底,统计连续同色长度,而不是边走边清零。
实操建议:
- 每个方向单独写一段逻辑,例如主对角线(\):先往
(row-2,col-2)到(row+2,col+2)收集5个值,再用filter(x => x === player).length === 5判断(适合小数据);或用累加器,从row-2开始正向扫,遇到非player就重置count = 0,否则count++,过程中一旦count === 5立即返回true - 别把
row和col当绝对坐标去套公式,比如board[row + i][col + i]在i=-2时可能越界,必须提前校验 - 如果棋盘用
0(空)、1(黑)、2(白)表示,注意别把0当成假值直接跳过——if (board[x][y])会把黑子1当真、白子2当真、空位0当假,但你得区分1和2
数组索引顺序搞反会导致整条线判错
JavaScript/Python里board[row][col]是常规写法,但有人习惯按“x,y”传参,写成board[y][x],结果主对角线查成了副对角线,或者水平变垂直。这不是逻辑错,是坐标系映射错。
实操建议:
- 统一用
board[i][j],其中i是行号(纵向位置),j是列号(横向位置);落子输入如果是clickX, clickY像素坐标,先转成列、行:col = Math.floor(clickX / cellSize),row = Math.floor(clickY / cellSize) - 写方向偏移数组时,明确标注含义:
[[0,1],[1,0],[1,1],[1,-1]]对应“右、下、右下、右上”,别写成[[1,0],[0,1],...]然后脑内倒推 - 调试时打印某条线上的值,例如主对角线5点:
console.log([board[row-2][col-2], board[row-1][col-1], board[row][col], board[row+1][col+1], board[row+2][col+2]]),一眼看出是否取对了
平局判断不能只看棋盘满不满
15×15棋盘共225个点,但五子棋极少下满——往往中间就赢了,或者某一方故意留空逼对手犯错。真到终局,更可能是“无空位且无人五连”,这时才算平局。但直接board.flat().every(x => x !== 0)不够,因为可能存在无效位置(比如带边框的渲染数组)或未初始化内存。
实操建议:
- 维护一个
emptyCount变量,每落一子就--,归零时再触发平局检测;比每次遍历快,也避免因数组结构不一致出错 - 平局检测必须在赢棋检测之后做——先跑一遍
checkWin,返回false再看是否emptyCount === 0 - 如果支持悔棋,记得同步更新
emptyCount,否则悔一步空位没补回,下次就误判平局
最麻烦的不是算连珠,是方向偏移和边界检查那几行代码写错后,bug只在特定位置触发,比如第3行第7列落子才漏判——这种得靠固定测试用例压,而不是靠运行几次看出来。










