代码随想录Day 53|题目:110. 字符串接龙、105.有向图的完全可达性、106. 岛屿的周长

ops/2024/9/24 23:52:11/

提示:DDU,供自己复习使用。欢迎大家前来讨论~

文章目录

    • 题目一:110. 字符串接龙
    • 思路
    • 题目二:105.有向图的完全可达性
    • 题目三:106. 岛屿的周长
    • 思路
    • 解法一:
    • 解法二:
    • 总结
      • 1. 有向图遍历
      • 2. 岛屿周长计算
      • 3. 组合方法计算岛屿周长

题目一:110. 字符串接龙

110. 字符串接龙 (kamacoder.com)

思路

问题的要点:

  1. 图的构建:将每个字符串看作一个节点,如果两个字符串只有一个字符不同,就在它们之间连一条边。

  2. 使用BFS:因为BFS能够找到从起点到终点的最短路径。

  3. 实现步骤

    • 把开始字符串加入队列,标记为访问过。
    • 用队列进行层次遍历,每到一个新字符串,就把它未访问的邻居加入队列。
    • 一旦到达终点字符串,返回当前遍历的步数。
  4. 注意事项

    • 用集合快速检查字符串是否存在于字典中。
    • 用标记记录每个字符串是否访问过,避免循环。

完整C++代码如下:

#include <iostream>
#include <vector>
#include <string>
#include <unordered_set>
#include <unordered_map>
#include <queue>
using namespace std;
int main() {string beginStr, endStr, str;int n;cin >> n;unordered_set<string> strSet;cin >> beginStr >> endStr;for (int i = 0; i < n; i++) {cin >> str;strSet.insert(str);}// 记录strSet里的字符串是否被访问过,同时记录路径长度unordered_map<string, int> visitMap; // <记录的字符串,路径长度>// 初始化队列queue<string> que;que.push(beginStr);// 初始化visitMapvisitMap.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';if (newWord == endStr) { // 发现替换字母后,字符串与终点字符串相同cout <<  path + 1 << endl; // 找到了路径 return 0;}// 字符串集合里出现了newWord,并且newWord没有被访问过if (strSet.find(newWord) != strSet.end()&& visitMap.find(newWord) == visitMap.end()) {// 添加访问信息,并将新字符串放到队列中visitMap.insert(pair<string, int>(newWord, path + 1));que.push(newWord);}}}}// 没找到输出0cout << 0 << endl;}

当然本题也可以用双向BFS,就是从头尾两端进行搜索。

题目二:105.有向图的完全可达性

105. 有向图的完全可达性 (kamacoder.com)

要解决这个问题,我们可以使用深度优先搜索(DFS)或广度优先搜索(BFS)来判断从节点1出发能否到达图中的所有节点。

  1. 构建图:首先,根据输入构建有向图,使用邻接表存储每个节点的出边。

  2. 选择搜索算法

    • DFS:从节点1开始,递归地访问所有可达的节点。
    • BFS:从节点1开始,层次地访问所有可达的节点。
  3. 实现步骤

    • 初始化:创建一个布尔数组visited来标记每个节点是否被访问过。
    • 遍历:使用DFS或BFS遍历图,标记所有从节点1出发可达的节点。
    • 检查:遍历结束后,检查visited数组,如果所有节点都被访问过,则输出1,否则输出-1。
  4. DFS的两种写法

    • 写法一:在递归之前检查当前节点是否已被访问,如果已访问则停止递归。
    • 写法二:在递归中检查下一个节点是否未被访问,如果未访问则标记为已访问并继续递归。
  5. 回溯:在这个问题中,我们不需要回溯操作,因为我们只关心是否能到达所有节点,而不关心具体的路径。

  6. 注意点:确保遍历过程中不会重复访问已经标记为已访问的节点,以避免无限循环。

通过上述步骤,我们可以确定从节点1是否可以到达有向图中的所有节点。

以上分析完毕,DFS整体实现C++代码如下:

// 写法一: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;
}

题目三:106. 岛屿的周长

106. 岛屿的周长 (kamacoder.com)

思路

解法一:

要解决这个问题,我们可以遵循以下思路:

  1. 遍历矩阵:检查每个单元格,如果单元格是岛屿(值为1),则进一步检查其边界。

  2. 检查边界:对于每个岛屿单元格,检查其上下左右四个方向:

    • 如果相邻单元格是水(值为0),则当前单元格是岛屿的边界,周长加1。
    • 如果相邻单元格出界(即越界),则同样认为当前单元格是岛屿的边界,周长加1。
  3. 计算周长:将所有岛屿边界单元格的数量加起来,即为岛屿的周长。

  4. 注意细节

    • 只需要检查岛屿的边界,不需要使用BFS或DFS搜索整个岛屿。
    • 确保在检查边界时不要重复计算同一个边界单元格。
  5. 实现步骤

    • 初始化周长计数器为0。
    • 遍历矩阵中的每个单元格。
    • 对于每个岛屿单元格,检查其四个方向:
      • 如果是水或出界,则周长计数器加1。
    • 遍历结束后,输出周长计数器的值。

C++代码如下:(详细注释)

#include <iostream>
#include <vector>
using namespace std;
int main() {int n, m;cin >> n >> m;vector<vector<int>> grid(n, vector<int>(m, 0));for (int i = 0; i < n; i++) {for (int j = 0; j < m; j++) {cin >> grid[i][j];}}int direction[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 (grid[i][j] == 1) {for (int k = 0; k < 4; k++) {       // 上下左右四个方向int x = i + direction[k][0];int y = j + direction[k][1];    // 计算周边坐标x,yif (x < 0                       // x在边界上|| x >= grid.size()     // x在边界上|| y < 0                // y在边界上|| y >= grid[0].size()  // y在边界上|| grid[x][y] == 0) {   // x,y位置是水域result++;}}}}}cout << result << endl;}

解法二:

要解决这个问题,我们可以遵循以下思路:

  1. 遍历矩阵:检查每个单元格,如果单元格是岛屿(值为1),则进一步检查其边界。

  2. 检查边界:对于每个岛屿单元格,检查其上下左右四个方向:

    • 如果相邻单元格是水(值为0),则当前单元格是岛屿的边界,周长加1。
    • 如果相邻单元格出界(即越界),则同样认为当前单元格是岛屿的边界,周长加1。
  3. 计算周长:将所有岛屿边界单元格的数量加起来,即为岛屿的周长。

  4. 注意细节

    • 只需要检查岛屿的边界,不需要使用BFS或DFS搜索整个岛屿。
    • 确保在检查边界时不要重复计算同一个边界单元格。
  5. 实现步骤

    • 初始化周长计数器为0。
    • 遍历矩阵中的每个单元格。
    • 对于每个岛屿单元格,检查其四个方向:
      • 如果是水或出界,则周长计数器加1。
    • 遍历结束后,输出周长计数器的值。

通过上述步骤,我们可以计算出岛屿的周长。

#include <iostream>
#include <vector>
using namespace std;
int main() {int n, m;cin >> n >> m;vector<vector<int>> grid(n, vector<int>(m, 0));for (int i = 0; i < n; i++) {for (int j = 0; j < m; j++) {cin >> grid[i][j];}}int sum = 0;    // 陆地数量int cover = 0;  // 相邻数量for (int i = 0; i < n; i++) {for (int j = 0; j < m; j++) {if (grid[i][j] == 1) {sum++; // 统计总的陆地数量// 统计上边相邻陆地if(i - 1 >= 0 && grid[i - 1][j] == 1) cover++;// 统计左边相邻陆地if(j - 1 >= 0 && grid[i][j - 1] == 1) cover++;// 为什么没统计下边和右边? 因为避免重复计算}}}cout << sum * 4 - cover * 2 << endl;}

总结

1. 有向图遍历

  • 图的表示:理解邻接表或邻接矩阵在表示图中的作用。
  • 深度优先搜索(DFS):掌握DFS的递归实现,用于遍历图中的节点。
  • 广度优先搜索(BFS:掌握BFS的层次遍历方法,用于查找最短路径。

2. 岛屿周长计算

  • 矩阵遍历:遍历整个矩阵以识别岛屿的边界。
  • 边界条件:理解如何识别岛屿的边界,包括水域和边界。
  • 计数问题:计算岛屿周长的算法实际上是一个计数问题。

3. 组合方法计算岛屿周长

  • 组合方法:利用组合数学原理来简化问题的解决。
  • 相邻关系:识别并计算岛屿内部的相邻单元格,以减少重复计数。
  • 算法优化:通过优化算法减少不必要的计算,提高效率。

http://www.ppmy.cn/ops/115514.html

相关文章

Unity 百度AI实现无绿幕拍照抠像功能(详解版)

目录 一、前言 1.抠像效果 2.去哪找百度ai抠图 3.基础流程跳过 二、获取AccessToken 1.什么是Token 2.为什么要获取Token 3.如何获取token 4.解析json 5.完整代码 三、抠像 1.准备地址 2.建立链接&#xff0c;和基本配置 3.图片格式转换 4.开始上传 5.获取回复…

SpringAop

SprinAOP的底层实现基于动态代理&#xff08;JDK CGLIB&#xff09;。 AOP主要应⽤于⽇志记录&#xff0c;性能统计&#xff0c;安全控制&#xff0c;事务处理等⽅⾯&#xff0c;实现公共功能性的重复使⽤。 JDK动态代理 注&#xff1a;要求目标对象有接口实现 通过Proxy类…

stm32单片机个人学习笔记5(OLED调试工具)

前言 本篇文章属于stm32单片机&#xff08;以下简称单片机&#xff09;的学习笔记&#xff0c;来源于B站教学视频。下面是这位up主的视频链接。本文为个人学习笔记&#xff0c;只能做参考&#xff0c;细节方面建议观看视频&#xff0c;肯定受益匪浅。 STM32入门教程-2023版 细…

AI大模型之旅--milvus向量库安装

milvus-向量索引库 milvus的官方文档中看到最新版本的部署方式 :https://milvus.io/docs/install_standalone-docker.md 部署 curl -sfL https://raw.githubusercontent.com/milvus-io/milvus/master/scripts/standalone_embed.sh -o standalone_embed.sh 如果下载不下来&a…

深入理解主键回显:提升数据操作效率与准确性

在软件开发的世界中&#xff0c;主键回显是一个常常被提及但又容易被忽视其重要性的概念。今天&#xff0c;我们就来深入探讨一下主键回显的奥秘。 一、什么是主键回显&#xff1f; 在数据库设计中&#xff0c;主键是用于唯一标识表中每一行记录的字段。而主键回显&#xff0…

1.分页查询(后端)—— Vue3 + SpringCloud 5 + MyBatisPlus + MySQL 项目系列(基于 Zulu 11)

本手册是基于 Vue3 SpringCloud5 MyBatisPlus MySQL 的项目结构和代码实现&#xff0c;旨在作为一个教学案例进行讲解。为了使案例更具普适性&#xff0c;文档中的公司名称、实体类、表名以及字段名称等敏感信息均已脱敏。 项目结构概述 项目采用标准的分层架构&#xff0…

PHP智慧教育新篇章优校管理系统小程序源码

智慧教育新篇章 —— 优校管理系统 &#x1f680;【开篇启航&#xff1a;智慧教育的浪潮已至】 在这个日新月异的时代&#xff0c;教育也在悄然发生着变革。随着科技的飞速发展&#xff0c;智慧教育已成为教育领域的新风尚。而“优校管理系统”&#xff0c;正是这股浪潮中的佼…

RTE大会报名丨 重塑语音交互:音频技术和 Voice AI,RTE2024 技术专场第一弹!

Voice AI 实现 human-like 的最后一步是什么&#xff1f; AI 视频爆炸增长&#xff0c;新一代编解码技术将面临何种挑战&#xff1f; 当大模型进化到实时多模态&#xff0c;又将诞生什么样的新场景和玩法&#xff1f; 所有 AI Infra 都在探寻规格和性能的最佳平衡&#xff0…