完整教程:图论回溯

news/2025/10/5 9:26:01/文章来源:https://www.cnblogs.com/ljbguanli/p/19126266

完整教程:图论&回溯

图论

200.岛屿数量DFS

给你一个由 ‘1’(陆地)和 ‘0’(水)组成的的二维网格,请你计算网格中岛屿的数量。岛屿总是被水包围,并且每座岛屿只能由水平方向和/或竖直方向上相邻的陆地连接形成。此外,你可以假设该网格的四条边均被水包围。

深度优先遍历:用一个visited数组标记所有访问过的地方,遍历图,遇到第一个陆地且没被访问过,就用深搜遍历此岛屿的全部陆地。

class Solution {
private:
int dir[4][2] = {
0
, 1
, 1
, 0
, -1
, 0
, 0
, -1
}
;
void dfs(vector<vector<
bool>>
& visit, vector<vector<
char>>
& grid,
int x,
int y){
for(
int i = 0
; i <
4
; i++
) {
int nextx = x + dir[i][0]
;
int nexty = y + dir[i][1]
;
if(nextx <
0 || nextx >= grid.size(
) || nexty <
0 || nexty >= grid[0].size(
)
)
continue
;
if(!visit[nextx][nexty] && grid[nextx][nexty] == '1'
) {
visit[nextx][nexty] = true
;
dfs(visit, grid, nextx, nexty)
;
}
}
}
public:
int numIslands(vector<vector<
char>>
& grid) {
int n = grid.size(
)
, m = grid[0].size(
)
;
vector<vector<
bool>> visit = vector<vector<
bool>>(n, vector<
bool>(m, false
)
)
;
int result = 0
;
for(
int i = 0
; i < n; i++
) {
for(
int j = 0
; j < m; j++
) {
if(!visit[i][j] && grid[i][j] == '1'
) {
result++
;
visit[i][j] = true
;
dfs(visit, grid, i, j)
;
}
}
}
return result;
}
}
;

994.腐烂的橘子BFS

在给定的 m x n 网格 grid 中,每个单元格可以有以下三个值之一:
值 0 代表空单元格;
值 1 代表新鲜橘子;
值 2 代表腐烂的橘子。
每分钟,腐烂的橘子 周围 4 个方向上相邻 的新鲜橘子都会腐烂。
返回 直到单元格中没有新鲜橘子为止所必须经过的最小分钟数。如果不可能,返回 -1 。

class Solution {
public:
int orangesRotting(vector<vector<
int>>
& grid) {
int n = grid.size(
)
, m = grid[0].size(
)
;
queue<pair<
int
,
int>> q;
int fresh = 0
;
for(
int i = 0
; i < n; i++
) {
for(
int j = 0
; j < m; j++
) {
if(grid[i][j] == 2
) {
q.push({
i,j
}
)
;
}
else
if(grid[i][j] == 1
) {
fresh++
;
}
}
}
int time = 0
;
int dir[4][2] = {
1
, 0
, 0
, 1
, -1
, 0
, 0
, -1
}
;
while(!q.empty(
) && fresh >
0
) {
int size = q.size(
)
;
while(size--
) {
auto [x, y] = q.front(
)
; q.pop(
)
;
for(
int i = 0
; i <
4
;i++
) {
int nx = x + dir[i][0]
;
int ny = y + dir[i][1]
;
if(nx <
0 || ny <
0 || nx >= grid.size(
) || ny >= grid[0].size(
) || grid[nx][ny] != 1
)
continue
;
grid[nx][ny] = 2
;
q.push({
nx,ny
}
)
;
fresh--
;
}
}
time++
;
}
return fresh == 0 ? time : -1
;
}
}
;

207.课程表

你这个学期必须选修 numCourses 门课程,记为 0 到 numCourses - 1 。
在选修某些课程之前需要一些先修课程。 先修课程按数组 prerequisites 给出,其中 prerequisites[i] = [ai, bi] ,表示如果要学习课程 ai 则 必须 先学习课程 bi 。
例如,先修课程对 [0, 1] 表示:想要学习课程 0 ,你需要先完成课程 1 。
请你判断是否可能完成所有课程的学习?如果可以,返回 true ;否则,返回 false 。

本题可约化为: 课程安排图是否是 有向无环图(DAG)。
方法一:入度表(广度优先遍历)
算法流程:
统计课程安排图中每个节点的入度,生成 入度表 indegrees。
借助一个队列 queue,将所有入度为 0 的节点入队。
当 queue 非空时,依次将队首节点出队,在课程安排图中删除此节点 pre:
并不是真正从邻接表中删除此节点 pre,而是将此节点对应所有邻接节点 cur 的入度 −1,即 indegrees[cur] -= 1。
当入度 −1后邻接节点 cur 的入度为 0,说明 cur 所有的前驱节点已经被 “删除”,此时将 cur 入队。
在每次 pre 出队时,执行 numCourses–;
若整个课程安排图是有向无环图(即可以安排),则所有节点一定都入队并出队过,即完成拓扑排序。换个角度说,若课程安排图中存在环,一定有节点的入度始终不为 0。
因此,拓扑排序出队次数等于课程个数,返回 numCourses == 0 判断课程是否可以成功安排。

class Solution {
public:
bool canFinish(
int numCourses, vector<vector<
int>>
& prerequisites) {
vector<vector<
int>>
adjacency(numCourses)
;
vector<
int>
indegrees(numCourses, 0
)
;
queue<
int> q;
for(
const
auto& pair : prerequisites) {
int course = pair[0]
, pre = pair[1]
;
indegrees[course]++
;
adjacency[pre].push_back(course)
;
}
for(
int i = 0
; i < indegrees.size(
)
; i++
) {
if(indegrees[i] == 0
) q.push(i)
;
}
while(!q.empty(
)
){
int pre = q.front(
)
; q.pop(
)
;
numCourses--
;
for(
int i = 0
; i < adjacency[pre].size(
)
; i++
) {
if(--indegrees[adjacency[pre][i]] == 0
)
q.push(adjacency[pre][i]
)
;
}
}
return numCourses == 0
;
}
}
;

208.前缀树

Trie(发音类似 “try”)或者说 前缀树 是一种树形数据结构,用于高效地存储和检索字符串数据集中的键。这一数据结构有相当多的应用情景,例如自动补全和拼写检查。
请你实现 Trie 类:
Trie() 初始化前缀树对象。
void insert(String word) 向前缀树中插入字符串 word 。
boolean search(String word) 如果字符串 word 在前缀树中,返回 true(即,在检索之前已经插入);否则,返回 false 。
boolean startsWith(String prefix) 如果之前已经插入的字符串 word 的前缀之一为 prefix ,返回 true ;否则,返回 false 。路漫漫我不畏;

class Trie {
private:
bool isEnd;
Trie* next[26]
;
public:
Trie(
) {
isEnd = false
;
memset(next, 0
,
sizeof(next)
)
;
}
void insert(string word) {
Trie* node =
this
;
for(
char ch : word){
if(node->next[ch - 'a'] == NULL
) {
node->next[ch - 'a'] =
new Trie(
)
;
}
node = node->next[ch - 'a']
;
}
node->isEnd = true
;
}
bool search(string word) {
Trie* node =
this
;
for(
char ch : word) {
node = node->next[ch - 'a']
;
if(node == NULL
)
return false
;
}
return node->isEnd;
}
bool startsWith(string prefix) {
Trie* node =
this
;
for(
char ch : prefix) {
node = node->next[ch - 'a']
;
if(node == NULL
)
return false
;
}
return true
;
}
}
;
/**
* Your Trie object will be instanti`ated and called as such:
* Trie* obj = new Trie();
* obj->insert(word);
* bool param_2 = obj->search(word);
* bool param_3 = obj->startsWith(prefix);
*/

全排列

46. 全排列

给定一个不含重复数字的数组 nums ,返回其 所有可能的全排列 。你可以 按任意顺序 返回答案。
在这里插入图片描述

class Solution {
private:
vector<
int> path;
vector<vector<
int>> result;
void backtracking(vector<
int>
& nums, vector<
bool>
& used){
if(path.size(
) == nums.size(
)
) {
result.push_back(path)
;
return
;
}
for(
int i = 0
; i < nums.size(
)
; i++
){
if(used[i] == true
)
continue
;
path.push_back(nums[i]
)
;
used[i] = true
;
backtracking(nums, used)
;
used[i] = false
;
path.pop_back(
)
;
}
}
public:
vector<vector<
int>>
permute(vector<
int>
& nums) {
path.clear(
)
;
result.clear(
)
;
vector<
bool>
used(nums.size(
)
, false
)
;
backtracking(nums,used)
;
return result;
}
}
;

78.子集

给你一个整数数组 nums ,数组中的元素 互不相同 。返回该数组所有可能的子集(幂集)。
在这里插入图片描述
子集问题、组合问题、分割问题可以抽象成树,组合和分割是收集树的叶子结点,而子集是找树的所有结点。

class Solution {
public:
vector<
int> path;
vector<vector<
int>> result;
void backtracking(vector<
int>
& nums,
int startIndex) {
result.push_back(path)
;
for(
int i = startIndex; i < nums.size(
)
; i++
) {
path.push_back(nums[i]
)
;
backtracking(nums, i + 1
)
;
path.pop_back(
)
;
}
}
vector<vector<
int>>
subsets(vector<
int>
& nums) {
backtracking(nums, 0
)
;
return result;
}
}
;

17. 电话号码的字母组合

给定一个仅包含数字 2-9 的字符串,返回所有它能表示的字母组合。答案可以按 任意顺序 返回。
在这里插入图片描述

class Solution {
private:
const string letterMap[10] = {
""
,
""
,
"abc"
,
"def"
,
"ghi"
,
"jkl"
,
"mno"
,
"pqrs"
,
"tuv"
,
"wxyz"
}
;
public:
vector<string> result;string s;void backtracking(const string& digits,int index) {if(digits.size() == index) {result.push_back(s);return;}int digit = digits[index] - '0';string letter = letterMap[digit];for(int i = 0; i < letter.size(); i++){s.push_back(letter[i]);backtracking(digits, index + 1);s.pop_back();}}vector<string>letterCombinations(string digits) {s.clear();result.clear();if(digits.size() == 0)return result;backtracking(digits, 0);return result;}};

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

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

相关文章

自已创建网站要怎么做网站策划就业前景

一个整数由个位、十位、百味...组成&#xff0c;我们知道整数可以用int型表示&#xff0c;那么一个整数到底是几位数呢&#xff1f;&#xff1f; 下面这个代码就是来计算位数的&#xff1a; #include<stdio.h>// 获得num的位数 int getbit(int num) {if(num 0)return 0…

用 Whisper 打破沉默:AI 语音技术如何重塑无障碍沟通方式? - 指南

用 Whisper 打破沉默:AI 语音技术如何重塑无障碍沟通方式? - 指南pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-family: &quo…

什么事三合一网站百度免费推广

光电耦合器作为一种关键的电子连接器&#xff0c;在航天航空领域扮演着重要角色。本文将深入探讨光电耦合器在航天航空领域的应用及其技术特点。 光电耦合器在航天航空领域的应用 光电耦合器作为一种高可靠性、高速传输、抗干扰能力强的连接器&#xff0c;在航天航空领域有着广…

实用指南:【论文阅读 | PR 2024 |ICAFusion:迭代交叉注意力引导的多光谱目标检测特征融合】

实用指南:【论文阅读 | PR 2024 |ICAFusion:迭代交叉注意力引导的多光谱目标检测特征融合】pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !impo…

软件设计师难考吗网站seo规划

AutoUpdater.NET 是一款用于WPF、Winform软件版本更新的框架&#xff0c;类似框架还有Squirrel、WinSparkle、NetSparkle、Google Omaha。 一、安装AutoUpdater.NET 首先&#xff0c;您需要在项目中安装AutoUpdater.NET库。您可以通过NuGet包管理器来安装它。在Visual Studio中…

做网站一般要了解哪些重庆妇科医院排名最好的医院

android提高UI的流畅度Android中所有的界面绘制工作都是在UI线程中进行的&#xff0c;提高UI流畅度的最核心根本在于释放UI线程。即:不在主线程中做耗时的操作。很多人都知道&#xff0c;耗时的操作要放到子线程中去做&#xff0c;比如访问网络&#xff0c;比如读写sd卡。像这类…

生成式AI改进极端多标签分类技术

本文介绍利用生成式AI改进极端多标签分类的新方法,通过层次化标签聚类解决长尾分布问题,提出XLGen-BCL和XLGen-MCG两种架构,在多个数据集上验证了聚类引导模型在整体性能和罕见标签分类上的优势。会议信息 EACL 202…

2025.10.5——1绿

普及+/提高 P2216 [HAOI2007] 理想的正方形 单调队列优化的类似悬线法的题。

NIO----JAVA - 教程

NIO----JAVA - 教程2025-10-05 09:08 tlnshuju 阅读(0) 评论(0) 收藏 举报pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-fa…

【清晰教程】利用Git工具将本地项目push上传至GitHub仓库中 - 指南

【清晰教程】利用Git工具将本地项目push上传至GitHub仓库中 - 指南pre { white-space: pre !important; word-wrap: normal !important; overflow-x: auto !important; display: block !important; font-family: "…

建设工程敎育那个网站青岛网站建设官网

消息对话框 QMessageBox&#xff1a;消息对话框&#xff0c;用于向用户展示简单的信息&#xff0c;警告&#xff0c;错误或者询问对话框。 信息框 QMessageBox::information(nullptr,"标题","信息"); 警告框 QMessageBox::warning(nullptr,"标题&…

建设银行的网站用户名服务器的作用

编辑 ∑Gemini来源&#xff1a;国家自然科学基金委关于印发《国家自然科学基金项目科研不端行为调查处理办法》的通知国科金发诚〔2020〕96号各局&#xff08;室&#xff09;、科学部&#xff0c;机关党委&#xff0c;各直属单位&#xff1a;《国家自然科学基金项目科研不端行为…

题解:2025.10.信友队.智灵班选拔面试题目

2025.10.信友队.智灵班选拔面试题目题解 T1 题目描述 现在有25匹马赛跑,场地中有5个跑道(即一场比赛最多有5匹马参赛),赛马时你不能掐表,只能看到马的先后顺序,问至少比赛多少场能知道跑得最快的3匹马 错误思路1…

做义工旅行有哪些网站上海财务外包公司

Oracle提示错误消息ORA-28001: the password has expired&#xff0c;是由于Oracle11G的新特性所致&#xff0c; Oracle11G创建用户时缺省密码过期限制是180天&#xff08;即6个月&#xff09;&#xff0c; 如果超过180天用户密码未做修改则该用户无法登录。 Oracle公司是为了数…

电影网站网页设计手机微网站

1&#xff0c;检查是否有mysql残留文件查找出安装的mysql软件包和依赖包rpm -pa | grep mysql依次删除yum remove mysql-xxx-xxx-查找出所用的配置文件find / -name mysql依次删除rm -rf /var/lib/mysql2&#xff0c;删除MariaDB的文件&#xff0c;装MySQL的话会和MariaDB的文件…

如何用模板建站wordpress删除登录

题干 LCR 023. 相交链表 的头节点 headA 和 headB &#xff0c;请找出并返回两个单链表相交的起始节点。如果两个链表没有交点&#xff0c;返回 null 。 图示两个链表在节点 c1 开始相交&#xff1a; 题目数据 保证 整个链式结构中不存在环。 注意&#xff0c;函数返回结果…

南昌网站建设资讯公司官网制作教程

编程题总结 题目一&#xff1a;输出无重复的3位数 题目描述 从{1,2,3,4,5,6,7,8,9}中随机挑选不重复的5个数字作为输入数组‘selectedDigits’&#xff0c;能组成多少个互不相同且无重复数字的3位数?请编写程》序&#xff0c;从小到大顺序&#xff0c;以数组形式输出这些3位…

三合一网站怎么建立东莞做网站微信巴巴

应用分发&#xff08;App Distribution&#xff09;或APP分发&#xff0c;通常指的是将移动应用程序&#xff08;如iOS、Android或其他平台的应用&#xff09;通过各种渠道提供给最终用户进行下载和安装的过程。这个过程涉及多个环节&#xff0c;包括应用开发、测试、发布、推广…

MX WEEK4

训练赛 A 今日未完成被 luogu P3225 创飞大学习。 B 今日未完成被 luogu P10953 创飞大学习。 C 今日未完成被 luogu P2272 创飞大学习。 D 今日未完成被 luogu P8867 创飞大学习。 E 今日未完成被 luogu P4126 创飞大…