news 2026/6/12 2:03:08

【每日算法】LeetCode 51. N 皇后

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
【每日算法】LeetCode 51. N 皇后

对前端开发者而言,学习算法绝非为了“炫技”。它是你从“页面构建者”迈向“复杂系统设计者”的关键阶梯。它将你的编码能力从“实现功能”提升到“设计优雅、高效解决方案”的层面。从现在开始,每天投入一小段时间,结合前端场景去理解和练习,你将会感受到自身技术视野和问题解决能力的质的飞跃。------ 算法:资深前端开发者的进阶引擎

LeetCode 51. N 皇后

1. 题目描述

1.1 问题定义

N 皇后问题要求在一个n × n的棋盘上放置n个皇后,使得它们彼此之间无法相互攻击。皇后可以攻击同一行、同一列或同一对角线(包括主对角线和副对角线)上的任何棋子。给定整数n,返回所有不同的解决方案。每个解决方案是一个棋盘布局,用'Q'表示皇后,'.'表示空位。

1.2 输入输出示例

  • 输入:n = 4
  • 输出:
[[".Q..",// 解决方案 1"...Q","Q...","..Q."],["..Q.",// 解决方案 2"Q...","...Q",".Q.."]]

解释: 4 皇后问题有两个不同的解决方案。每个解决方案是一个字符串数组,表示棋盘行。

2. 问题分析

2.1 问题本质

N 皇后是一个典型的组合搜索问题,需要枚举所有可能的放置方式并过滤无效解。由于皇后不能共享行、列或对角线,这形成了严格的约束条件,适合用回溯算法(深度优先搜索 + 剪枝)解决。核心是逐行放置皇后,利用约束避免无效搜索。

2.2 前端关联场景

在前端开发中,类似问题出现在:

  • UI 布局与约束:如拖放组件时确保不重叠、网格系统布局(类似 CSS Grid 或 Flexbox 的自动排列)。
  • 游戏与交互:棋盘类游戏(如数独、象棋)的规则验证和求解器。
  • 状态管理:复杂表单或多步骤向导中,处理依赖关系和回退逻辑(类似 React 状态机或 Vuex 的异步操作)。
    学习回溯算法能帮助你设计更高效的状态更新和渲染逻辑,提升应用性能。

3. 解题思路

3.1 回溯算法介绍

回溯算法通过递归尝试所有候选解,并在不满足约束时回溯(撤销选择)。对于 N 皇后:

  1. 逐行放置:从第 0 行开始,每行放置一个皇后。
  2. 约束检查:在放置前,检查当前列和两个对角线是否已被占用。
  3. 递归与回溯:如果安全则放置,递归进入下一行;完成后回溯,尝试其他列。
  4. 记录解:当所有行都放置皇后,保存当前棋盘布局。

3.2 复杂度分析

  • 时间复杂度: 最坏情况下为 O(N!),但通过剪枝(早期检测冲突),实际运行远少于 N!。每行有 N 种选择,但受列和对角线约束减少。
  • 空间复杂度: O(N),用于递归调用栈(深度 N)和存储当前棋盘(N × N 数组)。

3.3 最优解说明

回溯算法是 N 皇后的最优解,因为它系统地搜索所有可能解并利用剪枝避免无效路径。尽管存在优化(如位运算加速),但回溯思想是核心,且在前端中由于 n 通常较小(n ≤ 10),朴素回溯已足够高效。

4. 各思路代码实现

4.1 思路一:朴素回溯法(推荐前端使用)

使用数组记录列、主对角线和副对角线的占用状态,实现简单易读。

/** * @param {number} n * @return {string[][]} */varsolveNQueens=function(n){constresult=[];// 初始化棋盘,全部为'.'constboard=Array.from({length:n},()=>Array(n).fill('.'));// 检查位置 (row, col) 是否安全constisSafe=(row,col)=>{// 检查列冲突for(leti=0;i<row;i++){if(board[i][col]==='Q')returnfalse;}// 检查左上对角线冲突for(leti=row-1,j=col-1;i>=0&&j>=0;i--,j--){if(board[i][j]==='Q')returnfalse;}// 检查右上对角线冲突for(leti=row-1,j=col+1;i>=0&&j<n;i--,j++){if(board[i][j]==='Q')returnfalse;}returntrue;};constbacktrack=(row)=>{if(row===n){// 所有行放置完毕,记录解决方案result.push(board.map(rowArr=>rowArr.join('')));return;}for(letcol=0;col<n;col++){if(isSafe(row,col)){board[row][col]='Q';// 放置皇后backtrack(row+1);// 递归下一行board[row][col]='.';// 回溯,移除皇后}}};backtrack(0);returnresult;};

4.2 思路二:优化回溯法(位运算)

使用位掩码记录列和对角线占用,加速冲突检查,但 JavaScript 中位运算有局限性(基于 32 位整数)。

varsolveNQueens=function(n){constresult=[];constboard=Array.from({length:n},()=>Array(n).fill('.'));constbacktrack=(row,cols,diag1,diag2)=>{if(row===n){result.push(board.map(rowArr=>rowArr.join('')));return;}// 计算可用位置:cols, diag1, diag2 是位掩码,1 表示占用letavailablePositions=(~(cols|diag1|diag2))&((1<<n)-1);while(availablePositions){constposition=availablePositions&-availablePositions;// 获取最低位的 1constcol=Math.floor(Math.log2(position));// 转换为列索引board[row][col]='Q';// 递归更新掩码:diag1 左移,diag2 右移backtrack(row+1,cols|position,(diag1|position)<<1,(diag2|position)>>1);board[row][col]='.';availablePositions&=availablePositions-1;// 移除最低位的 1}};backtrack(0,0,0,0);returnresult;};

注意:位运算版本在 JavaScript 中对于 n > 31 可能失效,但前端场景通常 n 较小。建议优先使用朴素回溯以保持代码可读性。

5. 各实现思路的复杂度、优缺点对比表格

思路时间复杂度空间复杂度优点缺点
朴素回溯法O(N!)O(N)(递归栈)+ O(N²)(棋盘存储)实现简单,易于理解和调试;适合前端快速开发和面试场景对于 n 较大时效率较低,但前端中 n 通常较小
优化回溯法(位运算)O(N!)O(N)(递归栈)+ O(N²)(棋盘存储)冲突检查为 O(1),常数时间优化;适合性能敏感场景实现复杂,JavaScript 位运算有局限;可读性差,维护困难

6. 总结

6.1 实际应用场景

回溯算法在前端开发中具有广泛的应用价值:

  • UI 组件与布局:在动态网格或拖放系统中,确保组件满足约束(如不重叠),类似 N 皇后的位置冲突检测。
  • 游戏开发:构建棋盘游戏(如数独、八数码)的求解器,使用回溯搜索所有可能状态。
  • 状态管理与工作流:处理复杂表单验证或多步骤流程,回溯帮助实现“撤销”和“重试”逻辑。
  • 算法面试与进阶:作为前端开发者,掌握回溯算法能提升问题解决能力,助力技术面试和系统设计。
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/11 21:35:50

Docker入门

Docker是一款快速构建、运行、管理应用的工具自动搜索并下载应用镜像。镜像不仅包含应用本身&#xff0c;还包含应用运行所需要的环境&#xff0c;配置&#xff0c;系统函数库、Docker会在运行镜像时创建一个隔离环境&#xff0c;称为容器。镜像仓库&#xff1a;Docker Hub入门…

作者头像 李华
网站建设 2026/6/11 14:49:14

22、正则表达式:从基础到高级应用

正则表达式:从基础到高级应用 1. 正则表达式基础 正则表达式除了字面字符外,还包含元字符,用于指定更复杂的匹配规则。常见的元字符有: ^ $ . [ ] { } - ? * + ( ) | \ ,其他字符则被视为字面字符。不过,反斜杠字符在某些情况下用于创建元序列,也可让元字符被当作字…

作者头像 李华
网站建设 2026/6/8 11:41:57

取能模块:无源自取电创新解决方案

提及无源自取电&#xff0c;很多人会第一时间联想到智能手机的无线充电功能&#xff0c;原则上理解的没错&#xff0c;只是相较于无源自取电这种技术&#xff0c;二者本质上应用的领域有所区别。前者目前适用于中高压和低压输电线路上&#xff0c;以非接触式取能的核心优势&…

作者头像 李华
网站建设 2026/6/10 22:02:42

30、编写Shell脚本入门与项目实践

编写Shell脚本入门与项目实践 1. 编写第一个脚本 首先,我们来编写一个简单的“Hello World”脚本。在命令行中输入以下代码: [me@linuxbox ~]$ echo Hello World! # This is a comment too Hello World!这里的注释在命令行中作用不大,但在脚本里是很有用的。脚本的第一行…

作者头像 李华
网站建设 2026/6/11 3:10:20

启动进程,并返回新进程id(C++源码)

1、概述 启动进程,并返回新进程id,支持场景: 1、支持绝对路径: C:\Program Files\Google\Chrome\Application\chrome.exe 2、支持环境变量路径:%localappdata%\\Feishu\\Feishu.exe 1.1、调用示例 // 调用示例// 启动进程,并返回新进程id// 支持绝对路径: C:\Program Fil…

作者头像 李华
网站建设 2026/6/11 5:41:56

PDF压缩

winnzip项目pdf压缩部分/*** 压缩PDF文件* param inputFile 输入PDF文件路径* param outputFile 输出PDF文件路径* param compressionLevel 压缩等级: 0小尺寸, 1中等尺寸, 2大尺寸* param lossless 是否无损压缩* return 压缩是否成功*/使用Ghostscript命令行方式进行pdf压缩&…

作者头像 李华