【leetcode100-051到054】【图论】四题合集

这篇具有很好参考价值的文章主要介绍了【leetcode100-051到054】【图论】四题合集。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

【岛屿数量】

给你一个由 '1'(陆地)和 '0'(水)组成的的二维网格,请你计算网格中岛屿的数量。

岛屿总是被水包围,并且每座岛屿只能由水平方向和/或竖直方向上相邻的陆地连接形成。

此外,你可以假设该网格的四条边均被水包围。

思路:

很经典也很基础的图搜题,bfs或者dfs都行,这题先用dfs写一下。

每次开启函数(而不是被递归调用),会将当前起点能接触到的所有陆地都访问一次再退出,记录函数开启的次数即可。

对每个格子,我们向上下左右四个方向拓展,对其中位置合法的、是陆地的、还没被访问过的格子进行递归调用,直到所有能访问的格子都访问完毕。

代码其实跟树的dfs也大同小异,区别只在出口的判断条件,以及可能递归的方向从两颗子树变成了四个方向(网格的情况,普通图的话就是变成所有neighbors)。

class Solution {
public:
    //合法性判断
    bool inArea(vector<vector<char>>& grid, int r, int c) {
        return 0 <= r && r < grid.size() && 0 <= c && c < grid[0].size();
    }
    // dfs
    void dfs(vector<vector<char>>& grid, int r, int c) {
        //不合法
        if (!inArea(grid, r, c))
            return;
        //来过了
        if (grid[r][c] != '1')
            return;
        //标记并扩展
        grid[r][c] = '2';
        dfs(grid, r - 1, c);
        dfs(grid, r + 1, c);
        dfs(grid, r, c - 1);
        dfs(grid, r, c + 1);
    }
    int numIslands(vector<vector<char>>& grid) {
        int ans = 0;

        for (int r = 0; r < grid.size(); r++) {
            for (int c = 0; c < grid[0].size(); c++) {
                if (grid[r][c] == '1') {
                    dfs(grid, r, c);
                    ans++;
                }
            }
        }
        return ans;
    }
};

【腐烂的橘子】

在给定的 m x n 网格 grid 中,每个单元格可以有以下三个值之一:

  • 值 0 代表空单元格;
  • 值 1 代表新鲜橘子;
  • 值 2 代表腐烂的橘子。

每分钟,腐烂的橘子 周围 4 个方向上相邻 的新鲜橘子都会腐烂。

返回 直到单元格中没有新鲜橘子为止所必须经过的最小分钟数。如果不可能,返回 -1 。

思路:

看到最短时间,想必也知道这题很可能是要选bfs了。

理论上来说应该是要重新开辟空间来记录橘子的腐烂时间的,但我发现这题把原数组处理一下就可以直接使用而且不用担心信息丢失的问题。

剩下思路直接看注释,这题的注释我写得可认真啦。

class Solution {
public:
    int orangesRotting(vector<vector<int>>& grid) {
        //建立记录腐烂时间的数组,或改造原数组
        // vector<vector<int> > visited(m, vector<int>(n, -1));
        //顺便把初始腐烂的橘子入队
        queue<pair<int, int>> q;
        for (int i = 0; i < grid.size(); i++) {
            for (int j = 0; j < grid[0].size(); j++) {
                grid[i][j] -= 2;
                if (grid[i][j] == 0)
                    q.push(make_pair(i, j));
            }
        }
        //取一个烂橘子,把四周还新鲜的橘子污染,记录污染时间,并把新的烂橘子入队,直到没有待处理的烂橘子
        int curTime = 0, r = 0, c = 0;
        int rMax = grid.size(), cMax = grid[0].size();
        while (!q.empty()) {
            //取
            auto cur = q.front();
            q.pop();
            r = cur.first;
            c = cur.second;
            curTime = grid[r][c];
            //污染
            if (0 <= r - 1 && grid[r - 1][c] == -1) {
                q.push(make_pair(r - 1, c));
                grid[r - 1][c] = curTime + 1;
            }
            if (r + 1 < rMax && grid[r + 1][c] == -1) {
                q.push(make_pair(r + 1, c));
                grid[r + 1][c] = curTime + 1;
            }
            if (0 <= c - 1 && grid[r][c - 1] == -1) {
                q.push(make_pair(r, c - 1));
                grid[r][c - 1] = curTime + 1;
            }
            if (c + 1 < cMax && grid[r][c + 1] == -1) {
                q.push(make_pair(r, c + 1));
                grid[r][c + 1] = curTime + 1;
            }
        }
        //检查有无新鲜橘子剩下,顺便找最大时间
        int maxTime = 0;
        for (int i = 0; i < grid.size(); i++) {
            for (int j = 0; j < grid[0].size(); j++) {
                if (grid[i][j] == -1)
                    return -1;
                maxTime = max(maxTime, grid[i][j]);
            }
        }
        return maxTime;
    }
};

【课程表】

你这个学期必须选修 numCourses 门课程,记为 0 到 numCourses - 1 。

在选修某些课程之前需要一些先修课程。 先修课程按数组 prerequisites 给出,其中 prerequisites[i] = [ai, bi] ,表示如果要学习课程 ai 则 必须 先学习课程  bi 。

  • 例如,先修课程对 [0, 1] 表示:想要学习课程 0 ,你需要先完成课程 1 。

请你判断是否可能完成所有课程的学习?如果可以,返回 true ;否则,返回 false 。

思路:

从这题开始离开了美好的网格,变成了普通的图,好在这是一个超经典拓扑序列题,我们捋一下拓扑序列的思路就好了。

  • 建立入度表,出边表,存储当前可上课程的队列,已上课的数量;
  • 入度为零的是当前可以上的课,把它入队;
  • 每次从队列中取一节可以上的课,把它上了,已上课数量++;
  • 把这门课出边表里的邻居节点的入度都减一,如果减到零那么该邻居也可以上了,把它入队;
  • 当队伍空了,说明当前没有可以上的课了,检查已上课数量和应上课数量,就可以知道答案。
class Solution {
public:
    bool canFinish(int numCourses, vector<vector<int>>& prerequisites) {
        //建立入度表
        vector<int> inDegree(numCourses, 0);
        //建立出边表
        unordered_map<int, vector<int>> outTable;
        //记录数据
        for (int i = 0; i < prerequisites.size(); i++) {
            inDegree[prerequisites[i][0]]++;
            outTable[prerequisites[i][1]].push_back(prerequisites[i][0]);
        }
        //建立可上课队列
        queue<int> q;
        //把当前能上的课入队
        for (int i = 0; i < numCourses; i++) {
            if (inDegree[i] == 0)
                q.push(i);
        }
        //计数
        int cnt = 0;
        //队列不空(还有课能上)时循环
        while (!q.empty()) {
            //取出一门,上,计数,出边表里的课全都减入度,减到0的入队
            int cur = q.front();
            q.pop();
            cnt++;
            for (int i = 0; i < outTable[cur].size(); i++) {
                if (--inDegree[outTable[cur][i]] == 0)
                    q.push(outTable[cur][i]);
            }
        }
        //对比计数,返回结果
        return cnt == numCourses;
    }
};

【前缀树】

Trie(发音类似 "try")或者说 前缀树 是一种树形数据结构,用于高效地存储和检索字符串数据集中的键。这一数据结构有相当多的应用情景,例如自动补完和拼写检查。

请你实现 Trie 类:

  • Trie() 初始化前缀树对象。
  • void insert(String word) 向前缀树中插入字符串 word 。
  • boolean search(String word) 如果字符串 word 在前缀树中,返回 true(即,在检索之前已经插入);否则,返回 false 。
  • boolean startsWith(String prefix) 如果之前已经插入的字符串 word 的前缀之一为 prefix ,返回 true ;否则,返回 false 。

思路:

数据结构类的题好像没什么思路可言,倒是对熟练c++面向对象思想挺有帮助的。本题精髓在于26格的next数组,直接看代码吧。文章来源地址https://www.toymoban.com/news/detail-817544.html

class Trie {
private:
    bool isEnd;
    Trie* next[26];

public:
    Trie() {
        isEnd = false;
        for (int i = 0; i < 26; i++) {
            next[i] = nullptr;
        }
    }

    void insert(string word) {
        Trie* cur = this;
        for (char c : word) {
            if (cur->next[c - 'a'] == nullptr) {
                cur->next[c - 'a'] = new Trie();
            }
            cur = cur->next[c - 'a'];
        }
        cur->isEnd = true;
    }

    bool search(string word) {
        Trie* cur = this;
        for (char c : word) {
            if (cur->next[c - 'a'] == nullptr)
                return false;
            cur = cur->next[c - 'a'];
        }
        return cur->isEnd;
    }

    bool startsWith(string prefix) {
        Trie* cur = this;
        for (char c : prefix) {
            if (cur->next[c - 'a'] == nullptr)
                return false;
            cur = cur->next[c - 'a'];
        }
        return true;
    }
};

到了这里,关于【leetcode100-051到054】【图论】四题合集的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • LeetCode热题 100整理

    35. 搜索插入位置 给定一个排序数组和一个目标值,在数组中找到目标值,并返回其索引。如果目标值不存在于数组中,返回它将会被按顺序插入的位置。 请必须使用时间复杂度为 O(log n) 的算法。 示例 1: 输入: nums = [1,3,5,6], target = 5 输出: 2 示例 2: 输入: nums = [1,3,5,6], target

    2024年02月13日
    浏览(30)
  • 【leetcode100-086到090】【动态规划】一维五题合集2

    【单词拆分】 给你一个字符串  s  和一个字符串列表  wordDict  作为字典。如果可以利用字典中出现的一个或多个单词拼接出  s  则返回  true 。 注意: 不要求字典中出现的单词全部都使用,并且字典中的单词可以重复使用。 思路: 首先,我们依次考虑s的前i个字母能否被

    2024年02月19日
    浏览(27)
  • 【LeetCode 热题 100】图论 专题(bfs,拓扑排序,Trie树 字典树)

    from: https://leetcode.cn/studyplan/top-100-liked/ bfs 具有 边权为1 的最短路性质 拓扑排序,入度 Trie树, 高效存储 字符串【见鬼,不知道为什么写错,需要掌握熟练度】 dfs 写法,比较简洁 bfs 写法,有最短路性质 bfs 具有 边权为1 的最短路性质 拓扑排序 模板题 数组写法,简洁,需

    2024年02月13日
    浏览(36)
  • Leetcode hot100题 个人整理版

    给你两个非空的链表,表示两个非负的整数。它们每位数字都是按照 逆序 的方式存储的,并且每个节点只能存储 一位 数字。 请你将两个数相加,并以相同形式返回一个表示和的链表。 你可以假设除了数字 0 之外,这两个数都不会以 0 开头。 给定两个大小分别为 m m m 和 n

    2024年02月11日
    浏览(34)
  • 每天一道leetcode:797. 所有可能的路径(图论&中等&深度优先遍历)

    给你一个有 n 个节点的 有向无环图(DAG) ,请你找出所有从节点 0 到节点 n-1 的路径并输出( 不要求按特定顺序 ) graph[i] 是一个从节点 i 可以访问的所有节点的列表(即从节点 i 到节点 graph[i][j] 存在一条有向边)。 n == graph.length 2 = n = 15 0 = graph[i][j] n graph[i][j] != i (即不存

    2024年02月12日
    浏览(44)
  • 人工智能深度学习100种网络模型,精心整理,全网最全,PyTorch框架逐一搭建

    大家好,我是微学AI,今天给大家介绍一下人工智能深度学习100种网络模型,这些模型可以用PyTorch深度学习框架搭建。模型按照个人学习顺序进行排序: 深度学习模型 ANN (Artificial Neural Network) - 人工神经网络:基本的神经网络结构,包括输入层、隐藏层和输出层。 学习点击地

    2024年02月14日
    浏览(32)
  • 【LeetCode75】第三十四题 叶子相似的树

    目录 题目: 示例: 分析: 代码:  给我们两棵二叉树,让我们判断这两棵二叉树的从左到右的叶子节点组成的叶子序列是否一致,即从左到右的叶子节点的数值包括顺序是否一致。 那么首先我们知道,叶子节点是左右子树都为空的节点,那么我们可以先把两棵树的叶子节

    2024年02月12日
    浏览(33)
  • 每天一道leetcode:剑指 Offer 34. 二叉树中和为某一值的路径(中等&图论&深度优先遍历&递归)

    给你二叉树的根节点 root 和一个整数目标和 targetSum ,找出所有 从根节点到叶子节点 路径总和等于给定目标和的路径。 叶子节点 是指没有子节点的节点。 树中节点总数在范围 [0, 5000] 内 -1000 = Node.val = 1000 -1000 = targetSum = 1000 使用递归深度优先遍历,使用前序遍历,在遍历途

    2024年02月12日
    浏览(40)
  • 【代码随想录-Leetcode第四题:150. 逆波兰表达式求值】

    给你一个字符串数组 tokens ,表示一个根据 逆波兰表示法 表示的算术表达式。 请你计算该表达式。返回一个表示表达式值的整数。 注意: 有效的算符为 ‘+’、‘-’、‘*’ 和 ‘/’ 。 每个操作数(运算对象)都可以是一个整数或者另一个表达式。 两个整数之间的除法总

    2024年02月13日
    浏览(31)
  • MATLAB——图论合集

    本帖总结一些经典的图论问题,通过MATLAB如何计算答案。近期在复习考研,以此来巩固一下相关知识——虽然考研肯定不能用MATLAB代码哈哈,不过在实际应用中解决问题还是很不错的,比C++易上手得多~ 此外,本帖图论中非常重要的知识点——最小生成树。作为数据结构的理

    2024年02月08日
    浏览(27)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包