回溯算法:解锁多种问题的解决之门
经典回溯算法
回溯算法是一种基于深度优先搜索的算法,通过探索所有可能的候选解来找出所有可能的解。当候选解不满足条件时,会回溯到上一步,尝试其他的候选解。下面将介绍回溯算法在组合问题、切割问题、排列问题、子集问题、棋盘问题和图的遍历等方面的应用。
组合问题
如从 N 个数中选出 k 个数的所有组合方式。以从数组 [1,2,3,4] 中选 2 个数为例,其核心思想是通过回溯的方式尝试所有可能的组合。具体步骤如下:
- 初始化 :定义结果集和路径变量,结果集用于存储所有满足条件的组合,路径变量用于存储当前递归路径上的元素。
- 递归函数 :参数包括起始位置 startindex、目标个数 k、数组等。终止条件是路径的长度等于 k 时,将路径添加到结果集中。在循环中,从 startindex 开始遍历数组,依次将元素添加到路径中,并递归调用函数,起始位置加 1,继续选择下一个元素。回溯时移除路径中的最后一个元素。
- 代码演示 :
func combine(n int, k int) [][]int {result := [][]int{}path := []int{}var backtrack func(startIndex int)backtrack = func(startIndex int) {if len(path) == k {temp := make([]int, k)copy(temp, path)result = append(result, temp)return}for i := startIndex; i <= n; i++ {path = append(path, i)backtrack(i + 1)path = path[:len(path)-1]}}backtrack(1)return result
}
切割问题
一个字符串按一定规则有几种切割方式。例如将字符串 “aab” 切割成所有可能的子字符串组合。其核心思想是通过回溯的方式尝试所有可能的切割位置。具体步骤如下:
- 初始化 :定义结果集和路径变量。
- 递归函数 :参数包括起始位置 startindex、字符串等。终止条件是起始位置等于字符串长度时,将路径添加到结果集中。在循环中,从 startindex 开始遍历字符串,依次切割子字符串,判断是否符合要求,如果符合,就将其添加到路径中,并递归调用函数,起始位置更新为 i+1。回溯时移除路径中的最后一个元素。
- 代码演示 :
func partition(s string) [][]string {result := [][]string{}path := []string{}var backtrack func(startIndex int)backtrack = func(startIndex int) {if startIndex >= len(s) {temp := make([]string, len(path))copy(temp, path)result = append(result, temp)return}for i := startIndex; i < len(s); i++ {if isPalindrome(s[startIndex : i+1]) {path = append(path, s[startIndex:i+1])backtrack(i + 1)path = path[:len(path)-1]}}}backtrack(0)return result
}
func isPalindrome(s string) bool {for i := 0; i < len(s)/2; i++ {if s[i] != s[len(s)-1-i] {return false}}return true
}
排列问题
如 N 个数的所有排列方式。以数组 [1,2,3] 的全排列为例。其核心思想是通过回溯的方式尝试所有可能的排列。具体步骤如下:
- 初始化 :定义结果集和路径变量,同时定义一个 used 数组来记录元素是否被使用。
- 递归函数 :参数包括 used 数组等。终止条件是路径的长度等于数组长度时,将路径添加到结果集中。在循环中,依次选择未使用的元素,将其添加到路径中,并标记为已使用,递归调用函数。回溯时移除路径中的最后一个元素,并标记为未使用。
- 代码演示 :
func permute(nums []int) [][]int {result := [][]int{}path := []int{}used := make([]bool, len(nums))var backtrack func()backtrack = func() {if len(path) == len(nums) {temp := make([]int, len(path))copy(temp, path)result = append(result, temp)return}for i := 0; i < len(nums); i++ {if !used[i] {used[i] = truepath = append(path, nums[i])backtrack()path = path[:len(path)-1]used[i] = false}}}backtrack()return result
}
子集问题
如从 N 个数中选出所有符合条件的子集。以数组 [1,2,3] 的所有子集为例。其核心思想是通过回溯的方式尝试所有可能的子集。具体步骤如下:
- 初始化 :定义结果集和路径变量。
- 递归函数 :参数包括起始位置 startindex、数组等。终止条件是当起始位置大于等于数组长度时结束。在循环中,从 startindex 开始遍历数组,依次将元素添加到路径中,并添加到结果集中,然后递归调用函数,起始位置加 1,继续向下搜索。回溯时移除路径中的最后一个元素。
- 代码演示 :
func subsets(nums []int) [][]int {result := [][]int{}path := []int{}var backtrack func(startIndex int)backtrack = func(startIndex int) {temp := make([]int, len(path))copy(temp, path)result = append(result, temp)for i := startIndex; i < len(nums); i++ {path = append(path, nums[i])backtrack(i + 1)path = path[:len(path)-1]}}backtrack(0)return result
}
棋盘问题
经典的八皇后问题是回溯算法在棋盘问题中的典型应用。其核心思想是通过回溯的方式尝试在棋盘上放置皇后。具体步骤如下:
- 初始化 :定义棋盘和结果集。
- 递归函数 :参数包括行号 row 等。终止条件是当行号等于棋盘大小时,将当前棋盘状态添加到结果集中。在循环中,依次尝试在当前行的每一列放置皇后,判断是否满足条件,如果满足,就在此位置放置皇后,并递归调用函数,行号加 1。回溯时撤销皇后的位置。
- 代码演示 :
func solveNQueens(n int) [][]string {result := [][]string{}board := make([][]string, n)for i := range board {board[i] = make([]string, n)for j := range board[i] {board[i][j] = "."}}var backtrack func(row int)backtrack = func(row int) {if row == n {temp := make([]string, n)for i := range board {temp[i] = strings.Join(board[i], "")}result = append(result, temp)return}for col := 0; col < n; col++ {if isValid(board, row, col) {board[row][col] = "Q"backtrack(row + 1)board[row][col] = "."}}}backtrack(0)return result
}
func isValid(board [][]string, row, col int) bool {for i := 0; i < row; i++ {if board[i][col] == "Q" {return false}}for i, j := row-1, col-1; i >= 0 && j >= 0; i, j = i-1, j-1 {if board[i][j] == "Q" {return false}}for i, j := row-1, col+1; i >= 0 && j < len(board); i, j = i-1, j+1 {if board[i][j] == "Q" {return false}}return true
}
图的遍历
深度优先搜索(DFS)是回溯算法在图的遍历中的典型应用。其核心思想是通过回溯的方式深度优先地遍历图。具体步骤如下:
- 初始化 :定义访问标记数组,用于记录节点是否被访问过。
- 递归函数 :参数包括当前节点等。终止条件是当所有节点都被访问过时结束。将当前节点标记为已访问,并依次访问其邻接节点,若邻接节点未被访问过,则递归调用函数。回溯时将当前节点标记为未访问。
- 代码演示 :
func DFS(graph [][]int, start int) {visited := make([]bool, len(graph))var dfs func(v int)dfs = func(v int) {visited[v] = truefmt.Print(v, " ")for _, w := range graph[v] {if !visited[w] {dfs(w)}}}dfs(start)
}
相关文章:
回溯算法:解锁多种问题的解决之门
经典回溯算法 回溯算法是一种基于深度优先搜索的算法,通过探索所有可能的候选解来找出所有可能的解。当候选解不满足条件时,会回溯到上一步,尝试其他的候选解。下面将介绍回溯算法在组合问题、切割问题、排列问题、子集问题、棋盘问题和图的…...

国产频谱仪性能如何?矢量信号分析仪到底怎么样?
矢量信号分析仪是一种高性能的电子测量设备,具备频谱分析、矢量信号分析、实时频谱分析、脉冲信号分析、噪声系数测量、相位噪声测量等多种功能。它能够对各类复杂信号进行精确的频谱特性分析、调制质量评估、信号完整性检测以及干扰源定位等操作。广泛应用于通信、…...

熔断器(Hystrix,Resilience4j)
熔断器 核心原理 熔断器通过监控服务调用失败率,在达到阈值时自动切断请求,进入熔断状态(类似电路保险丝)。其核心流程为: 关闭状态(Closed):正常处理请求,统计失…...
贪心算法套路模板+详细适用场景+经典题目清单
1. 排序 贪心选择 适用场景: 任务调度问题:需要安排多个任务,尽量完成更多任务或最小冲突。 区间调度问题:选出最多互不重叠的区间。 区间覆盖问题:用最少区间覆盖某个范围。 合并区间问题:合并重叠区…...

C++23 容器从其他兼容范围的可构造性与可赋值性 (P1206R7)
文章目录 背景与动机提案内容与实现细节提案 P1206R7实现细节编译器支持 对开发者的影响提高灵活性简化代码向后兼容性 总结 C23标准引入了对容器构造和赋值的新特性,这些特性使得容器能够更灵活地从其他兼容范围初始化,并支持从范围赋值。这些改进由提案…...

多通道振弦式数据采集仪MCU安装指南
设备介绍 数据采集仪 MCU集传统数据采集器与5G/4G,LoRa/RS485两种通信功能与一体的智能数据采集仪。该产品提供振弦、RS-485等的物理接口,能自动采集并存储多种自然资源、建筑、桥梁、城市管廊、大坝、隧道、水利、气象传感器的实时数据,利用现场采集的数…...
Axios中POST、PUT、PATCH用法区别
在 Axios 中,POST、PUT 和 PATCH 是用于发送 HTTP 请求的三种不同方法,它们的核心区别源自 HTTP 协议的设计语义。以下是它们的用法和区别: 1. POST 语义:用于创建新资源。 特点: 非幂等(多次调用可能产生…...
synchronized 实现原理
1. 对象头与 Mark Word 每个 Java 对象在内存中分为三部分:对象头、实例数据 和 对齐填充。 对象头 是核心部分,包含以下信息: Mark Word(标记字段):存储对象的哈希码、分代年龄、锁状态等。Klass Pointe…...

SOC-ESP32S3部分:9-GPIO输入按键状态读取
飞书文档https://x509p6c8to.feishu.cn/wiki/L6IGwHKV6ikQ08kqwAwcAvhznBc 前面我们学习了GPIO的输出,GPIO输入部分其实也是一样的,这里我们使用按键作为GPIO输入例程讲解,分三步走。 查看板卡原理图,确定使用的是哪个GPIO查看G…...
前端(小程序)学习笔记(CLASS 2):WXML模板语法与WXSS模板样式
1、数据绑定 数据绑定的基本原则 1、在data中定义数据 在页面对应的.js文件中,把数据定义到data对象中即可: Page({data: {//字符串类型的数据info: init data,//数组类型的数据msgList: [{msg: hello}, {msg: world}]} }) 2、在WXML中使用数据(Mus…...

Ubuntu20.04的安装(VMware)
1.Ubuntu20.04.iso文件下载 下载网址:ubuntu-releases-20.04安装包下载_开源镜像站-阿里云 2.创建虚拟环境 2.1打开VMware与创建新虚拟机 点击创建新虚拟机 如果没下好可以点击稍后安装操作系统 选择linux版本选择Ubuntu 64位然后点击下一步。 注意这里需要选择一…...

【论文阅读】LLaVA-OneVision: Easy Visual Task Transfer
LLaVA-OneVision: Easy Visual Task Transfer 原文摘要 研究背景与目标 开发动机: 基于LLaVA-NeXT博客系列对数据、模型和视觉表征的探索,团队整合经验开发了开源大型多模态模型 LLaVA-OneVision。 核心目标: 突破现有开源LMM的局限…...

Spring Boot 项目多数据源配置【dynamic datasource】
前言: 随着互联网的发展,数据库的读写分离、数据迁移、多系统数据访问等多数据源的需求越来越多,我们在日常项目开发中,也不可避免的为了解决这个问题,本篇来分享一下在 Spring Boot 项目中使用多数据源访问不通的数据…...

JAVA查漏补缺(2)
AJAX 什么是Ajax Ajax(Asynchronous Javascript And XML),即是异步的JavaScript和XML,Ajax其实就是浏览器与服务器之间的一种异步通信方式 异步的JavaScript 它可以异步地向服务器发送请求,在等待响应的过程中&…...

【Web前端】JavaScript入门与基础(二)
Javascript对象 什么是对象?对象(object)是 JavaScript 语言的核心概念,也是最重要的数据类型。简单说,对象就是一组“键值对”(key-value)的集合,是一种无序的复合数据集合。 var…...
取消 Conda 默认进入 Base 环境
在安装 Conda 后,每次打开终端时默认会进入 base 环境。可以通过以下方法取消这一默认设置。 方法一:使用命令行修改配置 在终端中输入以下命令,将 auto_activate_base 参数设置为 false: conda config --set auto_activate_ba…...

Electron+vite+vue3 从0到1搭建项目,开发Win、Mac客户端
随着前端技术的发展,出现了所谓的大前端。 大前端则是指基于前端技术延伸出来的各种终端平台及应用场景,包括APP、桌面端、手表终端、服务端等。 本篇文章主要是和大家一起学习一下使用Electron 如何打包出 Windows 和 Mac 所使用的客户端APPÿ…...
《深度揭秘:解锁智能体大模型自我知识盲区探测》
当面对超出其训练数据边界和固有知识范畴的问题时,智能体大模型往往会陷入困境,却浑然不知,这便是知识盲区带来的隐患。如何构建能够自动发现自身知识盲区的智能体大模型,成为当下人工智能领域亟待攻克的前沿难题,它关…...
打卡Day33
简单的神经网络 数据的准备 # 仍然用4特征,3分类的鸢尾花数据集作为我们今天的数据集 from sklearn.datasets import load_iris from sklearn.model_selection import train_test_split import numpy as np# 加载鸢尾花数据集 iris load_iris() X iris.data # …...
计算机组成原理-基本运算部件定点数的运算
2.2基本运算部件 整理自up主beokayy_ 1.加法器 一位全加器 全加器是最基本的加法单元: 三个输入端:加数Ai,加数Bi,低位传进来的进位C1-1两个输出端:本位和S,向高位的进位C 全加器的逻辑表达式: SiAi⊕Bi⊕Ci-1CiAiBi(Ai⊕Bi)C…...

python打卡day34@浙大疏锦行
知识点回归: CPU性能的查看:看架构代际、核心数、线程数GPU性能的查看:看显存、看级别、看架构代际GPU训练的方法:数据和模型移动到GPU device上类的call方法:为什么定义前向传播时可以直接写作self.fc1(x) ①CPU性能查…...

SOC-ESP32S3部分:8-GPIO输出LED控制
飞书文档https://x509p6c8to.feishu.cn/wiki/OSQWwh95niobqUkKyDQcVgsbnFg 这节课,我们将会以ESP32S3外设GPIO的使用为例,带大家学习如何从零开始学会ESP32外设的使用。 例如,这节课我们的需求是,需要通过GPIO控制指示灯的亮灭&…...

05算法学习_59. 螺旋矩阵 II
05算法学习_59. 螺旋矩阵 II 05算法学习_59. 螺旋矩阵 II题目描述:个人代码:学习思路:第一种写法:题解关键点: 个人学习时疑惑点解答: 05算法学习_59. 螺旋矩阵 II 力扣题目链接: 59. 螺旋矩阵 II 题目描…...
绘制音频信号的各种频谱图,包括Mel频谱图、STFT频谱图等。它不仅能够绘制频谱图librosa.display.specshow
librosa.display.specshow 是一个非常方便的函数,用于绘制音频信号的各种频谱图,包括Mel频谱图、STFT频谱图等。它不仅能够绘制频谱图,还能自动设置轴标签和刻度,使得生成的图像更加直观和易于理解。 ### 函数签名 python libros…...

Linux `>`/`>>` 重定向操作符深度解析与高阶应用指南
Linux `>`/`>>` 重定向操作符深度解析与高阶应用指南 一、核心功能解析1. 基础重定向2. 标准流描述符二、高阶重定向技巧1. 多流重定向2. 文件描述符操作3. 特殊设备操作三、企业级应用场景1. 日志管理系统2. 数据管道处理3. 自动化运维四、安全与权限管理1. 防误操作…...

【自定义类型-联合和枚举】--联合体类型,联合体大小的计算,枚举类型,枚举类型的使用
目录 一.联合体类型 1.1--联合体类型的声明 1.2--联合体的特点 1.3--相同成员的结构体和联合体对比 1.4--联合体大小的计算 1.5--联合体练习 二.枚举类型 2.1--枚举类型的声明 2.2--枚举类型的优点 2.3--枚举类型的使用 🔥个人主页:草莓熊Lotso…...

李宏毅《深度学习》:Self-attention 自注意力机制
一,问题分析: 什么情况下需要使用self-attention架构,或者说什么问题是CNN等经典网络架构解决不了的问题,我们需要开发新的网络架构? 要解决什么问题《——》对应开发self-attention架构的目的? 1&#…...

C++初阶-list的使用1
目录 1.std::list简介 2.成员函数 2.1构造函数的使用 2.2list::operator的使用 3.迭代器 4.容量 4.1list::empty函数的使用 4.2list::size函数的使用 4.3list::max_size函数的使用 5.元素访问 6.修饰符 6.1list::assign函数的使用 6.2push_back和pop_back和push_fr…...
Linux中的tty与login之间的关系
agetty 进程和 login 进程之间的关系: 一、简要概括 agetty 是登录前的终端初始化程序。 login 是处理用户登录认证的程序。 关系:agetty 启动后等待用户输入用户名,然后调用 login 进程进行用户认证。 二、详细过程 1. agetty 的作用 a…...

Python web 开发 Flask HTTP 服务
Flask 是一个轻量级的 Web 应用框架,它基于 Python 编写,特别适合构建简单的 Web 应用和 RESTful API。Flask 的设计理念是提供尽可能少的约定和配置,从而让开发者能够灵活地构建自己的 Web 应用。 https://andi.cn/page/622189.html...