代码随想录算法训练day59---图论系列4

embedded/2025/2/21 7:59:57/

代码随想录算法训练

—day59

文章目录

  • 代码随想录算法训练
  • 前言
  • 一、110.字符串接龙
  • 二、105.有向图的完全可达性
    • dfs版本1
    • dfs版本2
    • bfs版本
  • 三、100. 岛屿的最大面积
    • 方法一
    • 方法二
  • 总结


前言

今天是算法营的第59天,希望自己能够坚持下来!
今天继续图论part!今日任务:
● 110.字符串接龙
● 105.有向图的完全可达性
● 106.岛屿的周长


一、110.字符串接龙

卡码网题目链接
leetcode题目链接
文章讲解

思路:
从 startStr 到 endStr,找在 strList 中最短的路径,本题只需要求出最短路径的长度就可以了,不用找出具体路径。
所以这道题要解决两个问题:

  • 图中的线是如何连在一起的
  • 起点和终点的最短路径长度

1.由题目可得,strList中的字符串是以“ 只变化一个字符“作为连接
2.并且适合用广搜,因为广搜只要到达终点就是最短距离

另外需要有一个注意点:

  • 本题是一个无向图,需要用标记位,标记着节点是否走过,否则就会死循环
  • 使用set来检查字符串是否出现在字符串集合里更快一些

代码如下:

#include<iostream>
#include<unordered_set>
#include<unordered_map>
#include<queue>
#include<string>
using namespace std;//需要求最短距离,适合用广度优先搜索
//strList中的字符串是以“ 只变化一个字符“作为连接
int main() {int n;string beginStr, endStr, str;cin >> n;cin >> beginStr >> endStr;unordered_set<string> strSet; //用set方便查找while (n--) {cin >> str;strSet.insert(str);}//记录strSet里的字符串是否被访问过,同时记录路径长度unordered_map<string, int> visitMap; //<记录的字符串,路径长度>//初始化队列queue<string> que;que.push(beginStr);visitMap.insert(pair<string,int>(beginStr, 1));while (!que.empty()) {string word = que.front();que.pop();int path = visitMap[word]; //这个字符串在路径中的长度//开始在这个str中,挨个字符去替换for (int i = 0; i < word.size(); i++) {string newWord = word; //用一个新字符串替换str,因为每次要置换一个字符//遍历26字母for(int j = 0; j < 26; j++) {newWord[i] = j + 'a'; //从a开始换,替换掉str的第i个字母if (newWord == endStr) { //替换字母后,与重点字符串相同cout << path + 1 << endl; //找到路径了return 0;}//字符串集合中有newWord,并且newWord没有被访问过if (strSet.find(newWord) != strSet.end() && visitMap.find(newWord) == visitMap.end()) {visitMap[newWord] = path + 1; //添加访问信息que.push(newWord); //将新字符串放到队列中}}}}//没找到路径,输出0cout << 0 << endl;
}

二、105.有向图的完全可达性

卡码网题目链接
文章讲解

思路:
本题是一个有向图搜索全路径的问题。只能用深搜(DFS)或者广搜(BFS)来搜。

深搜三部曲:
1.确认递归函数,参数:需要地图,当前位置,标记访问过的位置的数组
2.确认终止条件:递归函数可以是处理当前节点,也可以是处理下一个节点。根据不同的写法有不同的终止条件。我这里是按照当前节点来写。那么终止条件就是当前节点已经访问过了,所以直接返回。
3.处理目前搜索节点出发的路径:处理当前路径就是对当前路径标记成true,并且找到下一个路径
注意:这里不需要回溯,因为题目求的是需要到达所有的节点,不需要退回标记。而回溯是找不同的路径时,到了终点需要掉头的时候就需要回溯,才能得出一条条不一样的路。

dfs版本1

代码如下:

// 写法一:dfs 处理当前访问的节点
#include <iostream>
#include <vector>
#include <list>
using namespace std;void dfs(const vector<list<int>>& graph, int key, vector<bool>& visited) {if (visited[key]) {return;}visited[key] = true;list<int> keys = graph[key];for (int key : keys) {// 深度优先搜索遍历dfs(graph, key, visited);}
}int main() {int n, m, s, t;cin >> n >> m;// 节点编号从1到n,所以申请 n+1 这么大的数组vector<list<int>> graph(n + 1); // 邻接表while (m--) {cin >> s >> t;// 使用邻接表 ,表示 s -> t 是相连的graph[s].push_back(t);}vector<bool> visited(n + 1, false);dfs(graph, 1, visited);//检查是否都访问到了for (int i = 1; i <= n; i++) {if (visited[i] == false) {cout << -1 << endl;return 0;}}cout << 1 << endl;
}

dfs版本2

第二种写法注意有注释的地方是和写法一的区别:

写法二:dfs处理下一个要访问的节点
#include <iostream>
#include <vector>
#include <list>
using namespace std;void dfs(const vector<list<int>>& graph, int key, vector<bool>& visited) {list<int> keys = graph[key];for (int key : keys) {if (visited[key] == false) { // 确认下一个是没访问过的节点visited[key] = true;dfs(graph, key, visited);}}
}int main() {int n, m, s, t;cin >> n >> m;vector<list<int>> graph(n + 1);while (m--) {cin >> s >> t;graph[s].push_back(t);}vector<bool> visited(n + 1, false);visited[1] = true; // 节点1 预先处理dfs(graph, 1, visited);for (int i = 1; i <= n; i++) {if (visited[i] == false) {cout << -1 << endl;return 0;}}cout << 1 << endl;
}

bfs版本

#include <iostream>
#include <vector>
#include <list>
#include <queue>
using namespace std;int main() {int n, m, s, t;cin >> n >> m;vector<list<int>> graph(n + 1);while (m--) {cin >> s >> t;graph[s].push_back(t);}vector<bool> visited(n + 1, false);visited[1] = true; //  1 号房间开始queue<int> que;que.push(1); //  1 号房间开始// 广度优先搜索的过程while (!que.empty()) {int key = que.front(); que.pop();list<int> keys = graph[key];for (int key : keys) {if (!visited[key]) {que.push(key);visited[key] = true;}}}for (int i = 1; i <= n; i++) {if (visited[i] == false) {cout << -1 << endl;return 0;}}cout << 1 << endl;
}

三、100. 岛屿的最大面积

卡码网题目链接
leetcode题目链接
文章讲解

方法一

遍历每一个空格,遇到岛屿则计算其上下左右的空格情况。
当岛屿周边是边界或者是水域的话,说明找到了一条边。
在这里插入图片描述
在这里插入图片描述

代码如下:

#include<iostream>
#include<vector>
using namespace std;int main() {int n, m;cin >> n >> m;vector<vector<int>> graph(n, vector<int>(m, 0));for (int i = 0; i < n; i++) {for (int j = 0; j < m; j++) {cin >> graph[i][j];}}int dir[4][2] = {0, 1, 1, 0, -1, 0, 0, -1};int result = 0;//遍历每个节点for (int i = 0; i < n; i++) {for (int j = 0; j < m; j++) {//如果是岛屿的话if (graph[i][j] == 1) {//遍历岛屿四周for (int k = 0; k < 4; k++) {int x = i + dir[k][0];int y = j + dir[k][1];//xy位置在边界上,或者是水域,则周长+1if (x < 0 || x >= n || y < 0 || y >= m || graph[x][y] == 0) {result++;}}}}}cout << result << endl;
}

方法二

在这里插入图片描述
假设一个岛屿有4条边,
有一对相邻两个陆地,边的总数就要减2,
那么result = 岛屿数量 * 4 - cover * 2;

代码如下:

#include<iostream>
#include<vector>
using namespace std;int main() {int n, m;cin >> n >> m;vector<vector<int>> graph(n, vector<int>(m, 0));for (int i = 0; i < n; i++) {for (int j = 0; j < m; j++) {cin >> graph[i][j];}}int sum = 0; //陆地数量int cover = 0; //相邻数量for (int i = 0; i < n; i++) {for (int j = 0; j < m; j++) {if (graph[i][j] == 1) {sum++;//统计当前岛屿左边相邻陆地if (i-1 >= 0 && graph[i-1][j] == 1) cover++;//统计当前岛屿左边相邻陆地if(j-1 >= 0 && graph[i][j-1] == 1) cover++;//不需要统计下边和右边,因为会重复计算}}}//岛屿周长= 岛屿数量*4 - 相邻的岛屿数*2cout << sum*4 - 2*cover << endl;
}

总结

  • 字符串接龙 :求最短路径,考虑用广搜。根据题意可知节点的连接是“只替换一个字符”,遍历字符串,再分别替换成26字母,并用set检查是否在list中。
  • 有向图的完全可达性:需要有一个数组来记录到达过的路径,注意这里不需要回溯。
  • 岛屿的最大面积:方法一:岛屿周围是水或者边界就是边长 方法二:周长= 岛屿数量4 - 相邻的岛屿数2

明天继续加油!


http://www.ppmy.cn/embedded/164019.html

相关文章

《95015网络安全应急响应分析报告(2024)》

2025年2月&#xff0c;95015服务平台发布了最新一期的《95015网络安全应急响应分析报告&#xff08;2024&#xff09;》。报告分别从整体形势、受害者特征、攻击者特征等方面&#xff0c;对2024年95015平台接报的739起网络安全应急响应事件展开分析&#xff0c;并给出了7个年度…

DeepSeek在linux下的安装部署与应用测试

结合上一篇文章&#xff0c;本篇文章主要讲述在Redhat linux环境下如何部署和使用DeepSeek大模型&#xff0c;主要包括ollama的安装配置、大模型的加载和应用测试。关于Open WebUI在docker的安装部署&#xff0c;Open WebUI官网也提供了完整的docker部署说明&#xff0c;大家可…

深度学习-123-综述之AI人工智能与DL深度学习简史1956到2024

文章目录 1 AI与深度学习的简史1.1 人工智能的诞生(1956)1.2 早期人工神经网络(1940-1960年代)1.3 多层感知器MLP(1960年代)1.4 反向传播(1970-1980年代)1.5 第二次黑暗时代(1990-2000年代)1.6 深度学习的复兴(21世纪末至今)1.6.1 CNN卷积神经网络(1980-2010)1.6.2 RNN递归神经…

使用verilog 实现 cordic 算法 ----- 旋转模式

1-设计流程 ● 了解cordic 算法原理&#xff0c;公式&#xff0c;模式&#xff0c;伸缩因子&#xff0c;旋转方向等&#xff0c;推荐以下链接视频了解 cordic 算法。哔哩哔哩-cordic算法原理讲解 ● 用matlab 或者 c 实现一遍算法 ● 在FPGA中用 verilog 实现&#xff0c;注意…

HTML元素

HTML文档是由各种各样功能的元素标签构成的&#xff0c;接下来这些元素可能你没有见过&#xff0c;不要担心&#xff0c;后面会逐一介绍它们&#xff0c;这里作为一个组略的了解&#xff0c;除了上一节我们介绍的span&#xff0c;h1&#xff0c;p标签外&#xff0c;HTML还有很多…

Java开发实习面试笔试题(含答案)

在广州一家中大公司面试&#xff08;BOSS标注是1000-9999人&#xff0c;薪资2-3k&#xff09;&#xff0c;招聘上写着Java开发&#xff0c;基本没有标注前端要求&#xff0c;但是到场知道是前后端分离人不分离。开始先让你做笔试&#xff08;12道问答4道SQL题&#xff09;&…

(萌新入门)如何从起步阶段开始学习STM32 —— 0.碎碎念

目录 前言与导论 碎碎念 所以&#xff0c;我到底需要知道哪些东西呢 从一些基础的概念入手 常见的工具和说法 ST公司 MDK5 (Keil5) CubeMX 如何使用MDK5的一些常用功能 MDK5的一些常见的设置 前言与导论 非常感谢2301_77816627-CSDN博客的提问&#xff0c;他非常好奇…

如何写出优秀的测试用例?

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