算法详解——回溯法

news2024/11/19 0:40:21

一、回溯法概述——问题背景

  回溯法是一种解决约束满足问题的方法,特别适用于解决组合问题、搜索优化问题等。它通过逐步构建候选解决方案并且在这个解决方案不再可能满足约束或条件时进行剪枝和回溯。具体来说,回溯法可以应用于以下类型的问题:

  • 排列和组合问题:例如求解全排列、子集合等问题,回溯法可以系统地生成所有可能的组合并逐一验证。
  • 决策问题:如八皇后问题、数独填充、图的着色问题等,这些问题需要在多个选择中找到符合特定约束的解。
  • 优化问题:如旅行商问题(TSP)和其他需要找到最优解的问题,回溯法可以遍历所有可能的解,找到成本最低或收益最高的解。
  • 分割问题:例如分割等和子集、装箱问题等,需要将集合分割成满足特定条件的多个子集。

二、回溯法思想——基本过程

  回溯法的基本思想是在包含问题所有可能解的解空间树中,采用深度优先搜索策略来探索解决方案。这一过程从解空间树的根节点开始,逐层深入每一个节点进行探索。在探索到任何一个节点时,首先需要判断该节点是否可能包含问题的解。如果判断为是,算法将继续从这个节点深入,探索所有可能的子节点;如果判断为否,则表明以当前节点为根的子树不会包含问题的有效解,因此无需进一步探索这一路径。

  这时,算法将执行“剪枝”操作,即停止当前路径的进一步探索并退回到其父节点,继续探索其他可能的子节点。通过这种方式,回溯法避免了无效搜索和不必要的计算,从而提高了搜索效率。整个过程不断重复,直到探索完整个解空间树或找到满足条件的解为止。这种策略确保了算法可以系统地覆盖所有可能的解决方案,同时在不满足条件的情况下及时撤退,节省资源。

三、回溯法应用——四皇后问题

   N N N 皇后问题是一种经典的组合优化问题,目标是在一个 N × N N×N N×N 的棋盘上放置 N N N 个皇后棋子,使得这些皇后彼此之间不发生冲突。在国际象棋中,皇后可以在水平、垂直或任意45°斜线方向上移动。因此,解决这一问题的关键在于确保任意两个皇后不能位于同一行、同一列或同一对角线上。为了达成这一目标,我们需要找到一种放置方法,使得棋盘上的每一行、每一列以及所有的主要和次要对角线上都至多只有一个皇后。解决这一问题可以帮助研究和理解更广泛的约束满足和搜索优化问题,同时也是探索算法设计和问题求解技巧的一种方式。如下图即使四皇后问题的一种解:

在这里插入图片描述

  四皇后问题可以通过构建和探索一个解空间树来求解。解空间树是一个抽象结构,用于表示所有可能的解决方案路径。在四皇后问题中,每个节点代表棋盘的一个状态,即某些皇后已经放置在棋盘上的位置。根节点是一个空棋盘,而每个子节点表示在棋盘上新增一个皇后的尝试。解空间树的每一层对应于棋盘的一行,其中每个节点尝试将一个皇后放在那一行的不同列中。我们从根节点开始,按照深度优先的策略探索这棵树。

解空间树的探索过程

  1. 根节点与层次结构:

  根节点代表一个空棋盘,即开始状态。从这里出发,第一层的四个子节点分别尝试在第一行的四列中放置一个皇后。每个子节点又将生成自己的子节点,这些子节点尝试在第二行放置皇后,而且必须确保不与第一行的皇后冲突。

  1. 深度优先搜索与回溯:

  搜索从根节点开始,深入到解空间树的第一层,尝试在第一行的每一列中放置一个皇后。对于每个位置,如果安全,则递归地进入下一层,也就是下一行,尝试放置另一个皇后。

  • 安全性检查:在尝试在棋盘的某行某列放置皇后之前,会检查该位置是否与已放置的皇后有列冲突或对角线冲突。

  • 递归深入:如果当前节点(即棋盘配置)安全,则在下一层继续放置皇后。

  • 回溯:如果在当前行无法找到一个安全的列位置,这表明当前路径不可能导致解决方案,因此算法会回溯到上一层,即上一行,移除上一行的皇后,尝试在其他列放置。

  1. 剪枝:

  在搜索过程中,如果一个节点已确定不可能导致有效解(例如,如果第一行和第二行的皇后在同一列),则无需探索这个节点的任何子节点。这种“剪枝”操作减少了不必要的计算,优化了搜索过程。

  1. 找到解决方案:

  当到达解空间树的底部,即成功在所有四行中放置了皇后,并且所有放置都是安全的,那么这个路径(从根到当前节点的路径)就是一个解决方案。如果未到达底部就无路可走,则需要回溯。

  1. 枚举所有可能的解决方案:

  通过从根节点到解空间树的每一个叶节点的遍历,我们可以找出所有可能的解决方案。棋盘上每一行的每一个合法位置都将被考虑一次,确保全面性。

  通过这种方式,四皇后问题的解空间树提供了一种系统的方法来探索所有可能的解决方案,并有效地避开了不可能的解决方案路径。这不仅显著提高了效率,而且确保了解决方案的完整性。

解决过程与原理

  1. 棋盘初始化:
      棋盘是一个4x4的二维数组,其中的每个元素初始设为0,表示该位置没有放置皇后。棋盘的每一行将尝试放置一个皇后。

  2. 安全性检查:

  在尝试将一个皇后放置在棋盘的某个位置时,必须确保该位置与已放置的其他皇后没有冲突。这涉及到三个方向的检查:

  • 列检查:确保当前列上没有其他皇后。

  • 左上对角线检查:检查从当前位置开始向左上方延伸的对角线上是否有其他皇后。

  • 右上对角线检查:检查从当前位置开始向右上方延伸的对角线上是否有其他皇后。

  这些检查确保任何两个皇后都不会在同一行、列或对角线上,从而避免了相互攻击。

  1. 递归与回溯:

  使用递归函数来尝试在每一行放置一个皇后。对于每一行,函数遍历所有列,并使用安全性检查判断是否可以在当前列放置皇后:

  • 如果当前位置安全,则放置皇后并递归地调用该函数以尝试在下一行放置另一个皇后。

  • 如果成功在所有行放置了皇后,则当前的棋盘配置是一个有效解。

  • 如果在某行中找不到安全的列来放置皇后,则函数将回溯到上一行,移除那一行的皇后,并尝试下一个列位置。

  • 这个过程将持续递归和回溯,直到所有可能的行和列的组合都被尝试过。

  1. 打印解决方案:

  一旦找到一个有效的棋盘配置,即所有皇后都安全地放置,便打印该配置。如果解空间中存在多个解,每个解都会被打印出来。

Python代码实现

def is_safe(board, row, col):
    # 检查列冲突
    for i in range(row):
        if board[i][col] == 1:
            return False

    # 检查左上对角线冲突
    for i, j in zip(range(row, -1, -1), range(col, -1, -1)):
        if board[i][j] == 1:
            return False

    # 检查右上对角线冲突
    for i, j in zip(range(row, -1, -1), range(col, len(board))):
        if board[i][j] == 1:
            return False

    return True

def solve_n_queens(board, row):
    if row == len(board):
        print_solution(board)
        return True

    solution_found = False
    for col in range(len(board)):
        if is_safe(board, row, col):
            board[row][col] = 1
            if solve_n_queens(board, row + 1):
                solution_found = True
            board[row][col] = 0

    return solution_found

def print_solution(board):
    for row in board:
        print(" ".join('Q' if x == 1 else '.' for x in row))
    print("")

def main():
    n = 4
    board = [[0] * n for _ in range(n)]
    if not solve_n_queens(board, 0):
        print("没有找到解决方案")

main()

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

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

相关文章

【高阶数据结构】图--最短路径问题

图--最短路径问题 一、单源最短路径--Dijkstra算法1、简介2、解析3、代码4、测试用例5、打印最小路径代码和测试6、缺陷:不能使用负路径 二、单源最短路径--Bellman-Ford算法1、简介2、解析(1)详情i、负权问题:一个点只跑一趟找最…

整除C++

题目&#xff1a; 思路&#xff1a; 循环遍历7(可以被 7 整除的正整数最小为7)~n&#xff0c;如果i%70&#xff0c;就说明i可以被7整除. 代码&#xff1a; #include<iostream> using namespace std; int n;//一个正整数 int main(){scanf("%d",&n);//输入…

Vue.js【路由】

初识路由 提到路由&#xff08;Route&#xff09;&#xff0c;一般我们会联想到网络中常见的路由器&#xff08;Router&#xff09;&#xff0c;那么路由和路由器之间有什么关联呢&#xff1f;路由是指路由器从一个接口接收到数据&#xff0c;根据数据的目的地址将数据定向传送…

git-删除workspace.xml的跟踪

问题描述 .gitignore 文件内容如下&#xff1a; .pyc *.pyc user_files/ .vscode/ __pycache__//.idea/misc.xml /.idea/modules.xml /.idea/inspectionProfiles/profiles_settings.xml /.idea/inspectionProfiles/Project_Default.xml /.idea/batrp_webbackend-server-dev.i…

Java 开发 框架安全:Spring 漏洞序列.(CVE-2022-22965)

什么叫 Spring 框架. Spring 框架是一个用于构建企业级应用程序的开源框架。它提供了一种全面的编程和配置模型&#xff0c;可以简化应用程序的开发过程。Spring 框架的核心特性包括依赖注入&#xff08;Dependency Injection&#xff09;、面向切面编程&#xff08;Aspect-Or…

【Linux笔记】 基础指令(二)

风住尘香花已尽 日晚倦梳头 重命名、剪切指令 -- mv 简介&#xff1a; mv 命令是 move 的缩写&#xff0c;可以用来移动文件或者将文件改名&#xff0c;是 Linux 系统下常用的命令&#xff0c;经常用来备份文件或者目录 语法&#xff1a; mv [选项] 源文件或目录 目标文件或目录…

笨方法自学python(三)-数学计算

数字和数学计算 这章练习里有很多的数学运算符号。我们来看一遍它们都叫什么名字 plus 加号-minus 减号/ slash 斜杠*asterisk 星号% percent 百分号< less-than 小于号greater-than 大于号< less-than-equal 小于等于号 greater-than-equal 大于等于号 print ("I …

刷t2、、、

、、 public class ThisTest {public static void main(String args[]) {int i;for (;;) {System.out.println(1);}} } while()的循环条件等于for中循环条件。循环体会有一个条件改变等于for中类似自增条件。while()判断条件一般在while前面会初始化跟for中初始化一样。这样 w…

【讲解下目标追踪】

&#x1f308;个人主页: 程序员不想敲代码啊 &#x1f3c6;CSDN优质创作者&#xff0c;CSDN实力新星&#xff0c;CSDN博客专家 &#x1f44d;点赞⭐评论⭐收藏 &#x1f91d;希望本文对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出指正&#xff0c;让我们共…

ECharts系列文章汇总(持续更新中)

ECharts介绍 ECharts是一款基于JavaScript的数据可视化图表库&#xff0c;提供了直观、生动、可交互、可个性化定制的数据可视化图表。以下是关于ECharts的详细介绍&#xff1a; 发展历程&#xff1a; ECharts最初由百度团队开源&#xff0c;并在2018年初捐赠给Apache基金会&…

软件工程经济学--期末复习资料

软件工程经济学--期末复习资料 前言第一章 绪论第二章 软件工程经济学基础第三章 软件的成本管理与定价分析第四章 软件工程项目评价方法与经济效果评价第五章 软件生产函数、效益分析及不确定性分析第六章 软件工程项目进度计划的制定结尾总结 前言 软件工程经济学&#xff0…

Github2024-05-10开日报 Top10

根据Github Trendings的统计&#xff0c;今日(2024-05-10统计)共有10个项目上榜。根据开发语言中项目的数量&#xff0c;汇总情况如下&#xff1a; 开发语言项目数量Python项目4TypeScript项目4JavaScript项目1Lua项目1C项目1Rust项目1Dart项目1 RustDesk: 用Rust编写的开源远…

如何映射公司的BS架构系统,出差也能远程访问?

在现代企业运营中&#xff0c;员工出差和分支机构的协同工作变得越来越普遍。然而&#xff0c;如何确保在不同地点的员工都能安全、便捷地访问公司内网的BS&#xff08;Browser/Server&#xff09;架构办公系统&#xff0c;是一个亟待解决的问题。 贝锐花生壳内网穿透服务提供…

vue 路由url中去掉#

修改前效果 想要去掉/# 如何实现&#xff1f; 1、typeScript中去掉url中# 找到项目中的router/index.ts-----------去掉createWebHashHistory中的Hash 将createWebHashHistory修改为createWebHistory 2、javaScript中去掉url中# 找到项目中的router/index.js-----------添加…

如何批量将十六进制数据转成bin文件

最近在做新项目遇到一个问题&#xff0c;我们要通过上位机把一堆数据通过串口发送给下位机存储&#xff0c;而上位机需要Bin文件。 解决办法&#xff1a; 1)创建一个记事本文件&#xff0c;然后将其后缀修改成.bin 2)然后打开notepad,新建一个文件&#xff0c;随便写下数据 我…

2024年第九届数维杯大学生数学建模挑战赛B 题思路1.0版本

B题&#xff1a;生物质和煤共热解问题的研究 数维杯分享资料&#xff08;问题一代码论文思路&#xff09;链接&#xff08;18点更新&#xff09;&#xff1a; 链接&#xff1a;https://pan.baidu.com/s/1GSv9dkqcY6G-XUnd0sJe_A 提取码&#xff1a;sxjm 题目分析&#xff1…

RFID在汽车制造中的应用如何改变行业

随着工业4.0和中国制造2025的推进&#xff0c;企业对于智能化、自动化的需求日益增长&#xff0c;RFID射频技术在制造业中已经相当普遍了。在如今这瞬息万变的行业与时代中&#xff0c;RFID技术可以帮助企业获得竞争优势&#xff0c;简化日益复杂的生产流程&#xff0c;推动企业…

Ansible的安装与基础命令的使用

Ansible Ansible 是一个开源的自动化工具&#xff0c;用于配置管理、应用部署和任务自动化。它由 Michael DeHaan 于 2012 年创建&#xff0c;后来被 Red Hat 收购。Ansible 的设计理念是简单易用&#xff0c;不需要在受管节点上安装任何代理软件&#xff0c;它通过 SSH&#…

使用 scrapyd 部署 scrapy

1.scrapyd 是什么&#xff1f; Scrapyd 是一个用于部署和运行 Scrapy 爬虫项目的服务器应用程序。它使得你可以通过 HTTP 命令来部署、管理和执行多个 Scrapy 爬虫&#xff0c;非常适合持续集成和生产环境中的爬虫部署。 2.安装scrapyd 并使用 2.1 安装 scrapyd F:\scrapydTes…

飞跨电容型的三电平(FC-NPC)逆变器simulink仿真模型

本人搭建了飞跨电容型的三电平逆变器simulink仿真模型&#xff0c;相较于二极管钳位型三电平逆变器而言&#xff0c;钳位二极管变为飞跨的电容。采用SPWM调制和均流均压控制&#xff0c;通过搭建仿真模型得到三电平波形。 三电平拓扑中的飞跨电容是指在电路的输出端使用电容来实…