C#八皇后算法:回溯法 vs 列优先法 vs 行优先法 vs 对角线优先法

目录

1.八皇后算法(Eight Queens Puzzle)

2.常见的八皇后算法解决方案

(1)列优先法(Column-First Method):

(2)行优先法(Row-First Method):

(3)对角线优先法(Diagonal-First Method):

(4)回溯法(Backtracking):


1.八皇后算法(Eight Queens Puzzle)

       皇后问题是一个古老而著名的问题,它实质上就是使棋盘上的8个皇后不能在同一行、同一列或同一条斜线上,共有92种方案。

2.常见的八皇后算法解决方案

        八皇后算法的解决方案有多种,以下是一些常见的解决方案:

(1)列优先法(Column-First Method):

        八皇后问题是一个经典的回溯算法问题,可以使用列优先法(或称为逐列解决法)来解决。 首先选择一个空的棋盘,然后从第一行开始,尝试将皇后放置在每一列。如果当前列没有被攻击,那么就将皇后放置在该列。否则,尝试下一列。当找到一个有效的列时,将皇后放置在该列的最下方。重复这个过程,直到所有的皇后都被放置在棋盘上。

// 使用列优先法解决八皇后问题
// 列优先算法也叫逐列解决法namespace _144_4
{class Program{private const int Size = 8;private readonly int[] queens = new int[Size];               // 存储每行皇后的列位置private readonly bool[] columns = new bool[Size];            // 列是否已被占用private readonly bool[] diagonals1 = new bool[2 * Size - 1]; // 主对角线是否已被占用private readonly bool[] diagonals2 = new bool[2 * Size - 1]; // 副对角线是否已被占用public void Solve(){PlaceQueen(0);}int solnum = 0;private void PlaceQueen(int row){if (row == Size){solnum += 1;PrintQueens(solnum);  // 所有皇后都已放置,打印结果return;}for (int col = 0; col < Size; col++){if (IsSafe(row, col)){queens[row] = col;columns[col] = true;diagonals1[row - col + Size - 1] = true;diagonals2[row + col] = true;PlaceQueen(row + 1);// 回溯diagonals2[row + col] = false;diagonals1[row - col + Size - 1] = false;columns[col] = false;}}}private bool IsSafe(int row, int col){return !columns[col] && !diagonals1[row - col + Size - 1] && !diagonals2[row + col];}private void PrintQueens(int solnum){Console.WriteLine("Solution{0}: ", solnum);for (int i = 0; i < Size; i++){for (int j = 0; j < Size; j++){if (queens[i] == j){Console.Write("Q ");}else{Console.Write("* ");}}Console.WriteLine();}Console.WriteLine();}public static void Main(string[] args){ArgumentNullException.ThrowIfNull(args);Program solver = new();solver.Solve();}}
}
  • 算法流程

        初始化所有数组。

        从第一行开始,尝试在每一列放置皇后。

        使用回溯法,如果在某一行找不到安全的位置,则返回到上一行并尝试下一个位置。

        当所有皇后都成功放置时,打印解决方案。

  • 方法分析

        Solve():启动算法,从第一行开始放置皇后。

        PlaceQueen(int row):递归方法,尝试在当前行的每一列放置皇后。如果找到一个安全的位置,则递归地尝试放置下一个皇后。如果所有皇后都已成功放置,则打印解决方案。

        IsSafe(int row, int col):检查在给定位置 (row, col) 放置皇后是否安全。如果列、主对角线和副对角线都没有被占用,则返回 true。

        PrintQueens(int solnum):打印解决方案。对于每一行,如果当前列有皇后,则打印 "Q",否则打印 "*"。

(2)行优先法(Row-First Method):

        与列优先法类似,但不同之处在于,该方法从第一列开始,尝试将皇后放置在每一行。如果当前行没有被攻击,那么就将皇后放置在该行的最右侧。否则,尝试下一行。当找到一个有效的行时,将皇后放置在该行的当前列。重复这个过程,直到所有的皇后都被放置在棋盘上。

// 使用行优先法解决八皇后问题
// 行优先算法也叫回溯法
namespace _144_3
{class Program{static void Main(string[] args){ArgumentNullException.ThrowIfNull(args);List<int[]> solutions = [];int[] solution = new int[8];Solve(0, solution, solutions);foreach (int[] sol in solutions){for (int i = 0; i < 8; i++){for (int j = 0; j < 8; j++){if (j == sol[i]){Console.Write("Q ");}else{Console.Write(". ");}}Console.WriteLine();}Console.WriteLine();}}static void Solve(int row, int[] solution, List<int[]> solutions){if (row == solution.Length){//solutions.Add(solution.ToArray()); // Add a copy of the solutionsolutions.Add([.. solution]);return;}for (int col = 0; col < solution.Length; col++){if (IsSafe(row, col, solution)){solution[row] = col;Solve(row + 1, solution, solutions);}}}static bool IsSafe(int row, int col, int[] solution){for (int i = 0; i < row; i++){if (solution[i] == col ||Math.Abs(solution[i] - col) == Math.Abs(i - row)){return false;}}return true;}}
}

        代码使用了行优先法(也称为回溯法)来解决八皇后问题,这是一个经典的递归回溯问题。 代码分析:

  • Main 方法

        Main 方法是程序的入口点。

        它首先检查 args 是否为空,如果为空则抛出异常。

        初始化一个空列表 solutions 来存储所有解决方案。

        调用 Solve 方法来寻找解决方案。

        遍历 solutions 列表,并打印出每一个解决方案。

  • Solve 方法

        Solve 方法是递归函数,用于寻找八皇后问题的解决方案。

        如果当前行 row 等于解决方案数组的长度(即8),则找到一个解决方案,并将其添加到 solutions 列表中。

        对于当前行的每一列,检查是否安全(没有冲突),如果安全则在该列放置皇后,并递归调用 Solve 方法处理下一行。

  • IsSafe 方法

        IsSafe 方法用于检查在当前位置 (row, col) 放置皇后是否安全。

        它遍历已经放置皇后的行,检查当前列和左上方对角线是否有冲突。

        如果没有冲突,返回 true;否则返回 false。

  • 注意事项

        在 Main 方法中,使用了 ArgumentNullException.ThrowIfNull(args); 来检查 args 是否为空。这通常用于命令行应用程序,但在没有实际命令行参数需求的程序中是不必要的。

        在 Solve 方法中,使用了 solutions.Add([.. solution]); 来添加解决方案的副本到 solutions 列表。这是C# 9.0引入的切片语法,用于创建数组或列表的浅拷贝。用这个简洁的方式来避免直接添加引用到同一个数组。

(3)对角线优先法(Diagonal-First Method):

        该方法首先选择一个空的棋盘,然后从左上角开始,尝试将皇后放置在对角线上。如果当前对角线没有被攻击,那么就将皇后放置在该对角线的最下方。否则,尝试下一个对角线。当找到一个有效的对角线时,将皇后放置在该对角线的当前列。重复这个过程,直到所有的皇后都被放置在棋盘上。

// 八皇后算法_对角线优先法
namespace _144_2
{class Program{static void Main(string[] args){ArgumentNullException.ThrowIfNull(args);int n = 8;int[] queens = new int[n];List<int[]> solutions = [];SolveQueens(n, queens, 0, solutions);Console.WriteLine("Total solutions: " + solutions.Count);foreach (int[] solution in solutions){for (int i = 0; i < n; i++){for (int j = 0; j < n; j++){if (j == solution[i]){Console.Write("Q ");}else{Console.Write(". ");}}Console.WriteLine();}Console.WriteLine();}}static void SolveQueens(int n, int[] queens, int index, List<int[]> solutions){if (index == n){solutions.Add([.. queens]);return;}for (int i = 0; i < n; i++){if (CanPlaceQueen(queens, index, i)){queens[index] = i;SolveQueens(n, queens, index + 1, solutions);}}}static bool CanPlaceQueen(int[] queens, int index, int col){for (int i = 0; i < index; i++){if (queens[i] == col || Math.Abs(queens[i] - col) == index - i){return false;}}return true;}}
}

        在这个示例中,使用一个整数数组queens来表示棋盘上皇后的位置。queens数组的每个元素表示对应行上皇后的列位置。

        使用递归函数SolveQueens来解决八皇后问题。该函数接受当前皇后的位置、当前行号和已找到的解作为参数。在递归过程中,尝试在每一列上放置皇后,并检查是否满足问题的约束条件。如果满足,则将皇后放置在当前行的对应列上,并继续递归处理下一行。如果当前行的所有列都满足约束条件,则表示找到一个解,并将解添加到已找到的解列表中。

        函数CanPlaceQueen用于检查在给定位置放置皇后是否满足约束条件。它接受棋盘大小、当前皇后的位置、当前行号和当前列号作为参数。在函数中,遍历当前行号之前的行,检查当前列号是否已经放置了皇后,或者当前行和列上的皇后是否在同一条对角线上。如果满足任一条件,则表示不能在当前位置放置皇后。

        在Main函数中,输出找到的解决方案。对于每个解决方案,遍历8行8列,如果当前列是皇后的位置,则输出"Q",否则输出"."。

(4)回溯法(Backtracking):

        该方法通过递归的方式尝试所有可能的皇后位置。算法步骤如下:

  • 选择一个空的棋盘。
  • 选择一个皇后,将其放置在棋盘的第一行的任意一列。
  • 选择下一个皇后,将其放置在下一行的任意一列,但不能与第一个皇后位于同一列或同一对角线上。
  • 重复步骤3,直到所有的皇后都被放置在棋盘上。
// 八皇后算法_回溯法
namespace _144
{class Program{#region 八皇后算法/// <summary>/// 解决八皇后问题/// </summary>/// <param name="size">皇后数量</param>static void QueenArithmetic(int size){int[] Queen = new int[size];//每行皇后的位置int y, x, i, j, d, t = 0;y = 0;Queen[0] = -1;while (true){for (x = Queen[y] + 1; x < size; x++){for (i = 0; i < y; i++){j = Queen[i];d = y - i;//检查新皇后是否能与以前的皇后相互攻击if ((j == x) || (j == x - d) || (j == x + d))break;}if (i >= y)break;     //不攻击}if (x == size)     //没有合适的位置{if (0 == y){Console.WriteLine("Over");        //回溯到了第一行break;     //结束}Queen[y] = -1; //回溯y--;}else{Queen[y] = x;   //确定皇后的位置y++;            //下一个皇后if (y < size)Queen[y] = -1;else{Console.WriteLine("\n" + ++t + ':');//所有的皇后都排完了,输出for (i = 0; i < size; i++){for (j = 0; j < size; j++)Console.Write(Queen[i] == j ? 'Q' : '*');Console.WriteLine();}y = size - 1;//回溯}}}Console.ReadLine();}#endregionstatic void Main(string[] args){ArgumentNullException.ThrowIfNull(args);int size = 8;           //皇后数QueenArithmetic(size);}}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/751316.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【计算机网络_应用层】https协议——加密和窃密的攻防

文章目录 1.https协议的介绍2. 加密和解密2.1 什么是加密2.2 常见的加密方式2.2.1 对称加密2.2.2 非对称加密 2.3 数据摘要&#xff08;数据指纹&#xff09;2.4 数字签名 3. https协议的加密和解密方案一&#xff1a;使用对称加密&#xff08;❌&#xff09;方案二&#xff1a…

Cesium模拟真实湖泊效果

1. 实现效果 2.实现方法 官方文档 Cesium的material中提供了水water的相关材质&#xff1a;传送门 详细代码&#xff1a; const getWater (data) > {// 根据几何实例创建贴地面图元waterPrimitive new Cesium.GroundPrimitive({geometryInstances: new Cesium.GeometryIn…

2024年腾讯云免费服务器申请入口,个人和企业均可申请

腾讯云免费服务器申请入口 https://curl.qcloud.com/FJhqoVDP 免费服务器可选轻量应用服务器和云服务器CVM&#xff0c;轻量配置可选2核2G3M、2核8G7M和4核8G12M&#xff0c;CVM云服务器可选2核2G3M和2核4G3M配置&#xff0c;腾讯云服务器网txyfwq.com分享2024年最新腾讯云免费…

基于单片机的DDS函数信号发生器

单片机DDS函数信号发生器设计 该设计以AT89S52为主控芯片&#xff0c;通过控制高性能DDS&#xff08;直接数字频率合成&#xff09;芯片AD9834产生不同频率的信号&#xff0c;经过6阶巴特沃兹低通滤波电路&#xff0c;连接数字电位器的运算放大电路&#xff0c;输出信号。可以…

理解接口,

关键点 接口可抽象出重要的行为标准&#xff08;用抽象方法来表示&#xff09;。 可以把实现接口的类的对象的引用赋值给接口变量&#xff0c;体 现该类根据接口里的行为标准给出具体行为。 ~接口的思想~&#xff1a;可以要求某些类&#xff08;类的父类可不同&#xff09;有…

监视和内存观察

监视和内存观察 5.监视和内存观察5.1 监视5.2 内存 5.监视和内存观察 在调试的过程中我们&#xff0c;如果要观察代码执行过程中&#xff0c;上下文环境中的变量的值&#xff0c;有哪些方法呢&#xff1f; 这些观察的前提条件一定是开始调试后观察&#xff0c;比如&#xff1…

Python小白笔记

输入 # 一行输入多个数字&#xff0c;空格隔开&#xff0c;存入列表a中 a list(map(int, input().split())) print(a) >>>21 22 34 54 67 >>>[21, 22, 34, 54, 67] 输出 数据&#xff1a; print(%d%10.3f%(x,y)) y的精度为3&#xff0c;宽度为10 %0 …

python中pyinstaller打包带资源的程序-pgzreo

本文使用创作助手。 PyInstaller是一个用于将Python脚本打包成可执行文件的工具。它可以将Python脚本、模块和依赖库打包成独立的可执行文件&#xff0c;方便在没有安装Python解释器的环境中运行。下面是PyInstaller的详细用法&#xff1a; 安装PyInstaller&#xff1a; 使用p…

Oracle 部署及基础使用

1. Oracle 简介 Oracle Database&#xff0c;又名 Oracle RDBMS&#xff0c;简称 Oracle Oracle系统&#xff0c;即是以Oracle关系数据库为数据存储和管理作为构架基础&#xff0c;构建出的数据库管理系统。是目前最流行的客户/服务器&#xff08;client/server&#xff09;或…

SpringTask实现的任务调度与XXL-job实现的分布式任务调度【XXL-Job工作原理】

目录 任务调度 分布式任务调度 分布式任务调度存在的问题以及解决方案 使用SpringTask实现单体服务的任务调度 XXL-job分布式任务调度系统工作原理 XXL-job系统组成 XXL-job工作原理 使用XXL-job实现分布式任务调度 配置调度中心XXL-job 登录调度中心创建执行器和任务 …

测试用例要如何写

1、测试点与测试用例 测试点不等于测试用例&#xff0c;这是我们首先需要认识到的。 问题1&#xff1a;这些测试点在内容上有重复&#xff0c;存在冗余。 问题2&#xff1a;一些测试点的测试输入不明确&#xff0c;不知道测试时要测试哪些。 问题3&#xff1a;总是在搭相似…

【pynput】监控是否打开百度贴吧网页

文章目录 简介Demo 简介 有网友提过一个要求&#xff0c;用 Python 实现一个 电脑打开某网站就自动关机的功能。 想到的思路有两个&#xff1a; 【windows 平台】, 获取活动的窗口标题&#xff0c;如果标题里包含了某些网站名称, 那就使用关机命令 可以定时拉取标题, 也可以使…

滑块验证码

1.这里针对滑块验证给了一个封装的组件verifition&#xff0c;使用直接可以调用 2.组件目录 3.每个文件的内容 3.1 Api文件中只有一个index.js文件&#xff0c;用来存放获取滑块和校验滑块结果的api import request from /router/axios//获取验证图片 export function reqGe…

Docker出现容器名称重复如何解决

假如你的重复容器名称是mysql5 删除已存在的容器&#xff1a;如果你不再需要那个已经存在的名为“mysql5”的容器&#xff0c;你可以删除它。使用下面的命令&#xff1a; docker rm -f mysql5这条命令会强制删除正在运行的容器。一旦容器被删除&#xff0c;你就可以重新使用这个…

Java 面向对象(类与对象 成员方法 方法重载 可变参数 构造方法 / 构造器 this关键字 包 访问修饰符)

目录 一、类与对象1. 类与对象的定义2. 类和对象的内存分配机制 二、成员方法1. 成员方法的定义2. 方法的调用机制3. 成员方法传参机制 三、方法重载四、可变参数1. 基本概念2. 基本语法3. 应用 五、 构造方法 / 构造器1. 特点2. 使用案例3. 对象创建的流程 六、this关键字1. 运…

P6安装:安装P6提示1433端口无效

错误描述 尝试运行 Microsoft SQL Server 2005 的 Primavera P6 数据库时&#xff0c;遇到以下错误&#xff1a; SQLServerException: The TCP/IP connection to the host [name], port 1433 has failed. Error: “Connection refused: connect. Verify the connection prope…

MATLAB的多项式相加

多项式的加减在阶次相同的情况下可直接运算&#xff0c;若两个相加减的多项式阶次不同&#xff0c;则低阶多项式必须用零填补高阶项系数&#xff0c;使其与高阶多项式有相同的阶次。而且通常情况下&#xff0c;进行加减的两个多项 式的阶次不会相同&#xff0c;这时可以自定义一…

视频素材库大全免费无水印必备素材网站,整理分享抖音视频素材剪辑必备

想要做出容易上热门的抖音视频作品&#xff0c;你一定要找到合适的短视频剪辑素材&#xff0c;今天我就给你收集了视频素材库大全免费无水印必备素材网站&#xff0c;都给你提供了海量的短视频素材库大全免费无水印&#xff0c;从风景&#xff0c;美食&#xff0c;动漫&#xf…

从排序算法的艺术看C语言qsort函数的魅力:一场数据的时空穿越

欢迎来到白刘的领域 Miracle_86.-CSDN博客 系列专栏 C语言知识 先赞后看&#xff0c;已成习惯 创作不易&#xff0c;多多支持&#xff01; 目录 一 、回调函数 二、qsort函数 1.qsort函数排序整型数据 2.qsort函数排序结构数据 一 、回调函数 何为回调函数&#xff1…

【LGR-179-Div.2】复旦勰码 3 月月赛 II ZHYOI Round 4(A~B)

T431492 农场 有n个农场&#xff0c;告诉你农场的对角顶点坐标&#xff0c;农场都是矩阵。 要找出一个最大的矩阵&#xff08;平行于x和y轴&#xff09;把所有农场都圈起来。 那就保证最下面的点和最上面的点&#xff0c;最左边的点&#xff0c;最右边的点都能被圈到。 用l…