Leetcode 675 为高尔夫比赛砍树

文章目录

  • [1. 题目描述](#1. 题目描述)
  • [2. 我的尝试](#2. 我的尝试)
  • [3. 题解](#3. 题解)
    • [1. BFS](#1. BFS)

1. 题目描述

Leetcode 675 为高尔夫比赛砍树

2. 我的尝试

cpp 复制代码
typedef priority_queue<int, vector<int>, greater<int>> heap;

class Solution {
public:
    int m;
    int n;

    int bfs(vector<vector<int>>& forest, vector<int> &h) {
        queue<pair<int, int>> q;
        int res = 0;
        int i = 0;
        int j = 0;
        int di[] = {0, -1, 0, 1};
        int dj[] = {-1, 0, 1, 0};
        const int N = 55;
        int d[N][N];
        bool st[N][N];

        q.push({0, 0});

        for (auto &t : h) {
            memset(d, 0x3f, sizeof d);
            memset(st, 0, sizeof st);
            d[i][j] = 0;

            while (q.size()) {
                i = q.front().first;
                j = q.front().second;
                q.pop();
//                 st[i][j] = true;

                if (forest[i][j] == t) {
                    res += d[i][j];
                    q = queue<pair<int, int>>();
                    q.push({i, j});
                    break;
                }

                for (int k = 0; k < 4; k ++) {
                    int x = i + di[k];
                    int y = j + dj[k];
                    if (0 <= x && x < m && 0 <= y && y < n && forest[x][y] && !st[x][y]) {
                        q.push({x, y});
                        st[x][y] = true;
                        d[x][y] = min(d[x][y], d[i][j] + 1);
                    }
                }
            }

            if (q.empty())
                return -1;
        }
        return res;
    }

    int cutOffTree(vector<vector<int>>& forest) {
        m = forest.size();
        n = forest[0].size();
//         priority_queue<int, vector<int>, greater<int>> h;
        vector<int> h;

        for (auto &v : forest)
            for (auto &x : v)
                if (x > 1)
                    h.push_back(x);

        sort(h.begin(), h.end());

        return bfs(forest, h);
    }
};

3. 题解

1. BFS

方法一:广度优先遍历[1](#1)

首先对矩阵中的树按照树的高度进行排序,我们依次求出相邻的树之间的最短距离。利用广度优先搜索,按照层次遍历,处理队列中的节点(网格位置)。visited 记录在某个时间点已经添加到队列中的节点,这些节点已被处理或在等待处理的队列中。对于下一个要处理的每个节点,查看他们的四个方向上相邻的点,如果相邻的点没有被遍历过且不是障碍,将其加入到队列中,直到找到终点为止,返回当前的步数即可。最终返回所有的步数之和即为最终结果

cpp 复制代码
class Solution {
public:
    int dirs[4][2] = {{-1, 0}, {1, 0}, {0, -1}, {0, 1}};

    int bfs(vector<vector<int>>& forest, int sx, int sy, int tx, int ty) {
        if (sx == tx && sy == ty) {
            return 0;
        }

        int row = forest.size();
        int col = forest[0].size();
        int step = 0;
        queue<pair<int, int>> qu;
        vector<vector<bool>> visited(row, vector<bool>(col, false));         
        qu.emplace(sx, sy);
        visited[sx][sy] = true;
        while (!qu.empty()) {
            step++;
            int sz = qu.size();
            for (int i = 0; i < sz; ++i) {
                auto [cx, cy] = qu.front();
                qu.pop();
                for (int j = 0; j < 4; ++j) {
                    int nx = cx + dirs[j][0];
                    int ny = cy + dirs[j][1];
                    if (nx >= 0 && nx < row && ny >= 0 && ny < col) {
                        if (!visited[nx][ny] && forest[nx][ny] > 0) {
                            if (nx == tx && ny == ty) {
                                return step;
                            }
                            qu.emplace(nx, ny);
                            visited[nx][ny] = true;
                        }
                    }
                }
            }
        }
        return -1;
    }

    int cutOffTree(vector<vector<int>>& forest) {
        vector<pair<int, int>> trees;
        int row = forest.size();
        int col = forest[0].size();
        for (int i = 0; i < row; ++i) {
            for (int j = 0; j < col; ++j) {
                if (forest[i][j] > 1) {
                    trees.emplace_back(i, j);
                }
            }
        }
        sort(trees.begin(), trees.end(), [&](const pair<int, int> & a, const pair<int, int> & b) {
            return forest[a.first][a.second] < forest[b.first][b.second];
        });

        int cx = 0;
        int cy = 0;
        int ans = 0;
        for (int i = 0; i < trees.size(); ++i) {
            int steps = bfs(forest, cx, cy, trees[i].first, trees[i].second);
            if (steps == -1) {
                return -1;
            }
            ans += steps;
            cx = trees[i].first;
            cy = trees[i].second;
        }
        return ans;
    }
};
  • 时间复杂度: O ( m 2 × n 2 ) O(m^2 \times n^2) O(m2×n2),其中 m m m 为矩阵的行数, n n n 为矩阵的列数。矩阵中最多有 m × n m \times n m×n 颗树,对树的高度进行排序,时间复杂度为 O ( m × n × log ⁡ ( m × n ) ) O(m \times n \times \log (m \times n)) O(m×n×log(m×n)),利用广度优先搜索两颗树之间的最短距离需要的时间为 O ( m × n ) O(m \times n) O(m×n),因此总的时间复杂为 O ( m × n × log ⁡ ( m × n ) + m 2 × n 2 ) = O ( m 2 × n 2 ) O(m \times n \times \log (m \times n) + m^2 \times n^2) = O(m^2 \times n^2) O(m×n×log(m×n)+m2×n2)=O(m2×n2)。

  • 空间复杂度: O ( m × n ) O(m \times n) O(m×n),其中 m m m 为矩阵的行数, n n n 为矩阵的列数。矩阵中最多有 m × n m \times n m×n 颗树,对树的高度进行排序,所需要的栈空间为 O ( log ⁡ ( m × n ) ) O(\log (m \times n)) O(log(m×n)),利用广度优先搜索队列中最多有 O ( m × n ) O(m \times n) O(m×n) 个元素,标记已遍历过的元素需要的空间为 O ( m × n ) O(m \times n) O(m×n),因此总的空间复杂度为 O ( m × n ) O(m \times n) O(m×n)。


  1. 官方题解 ↩︎
相关推荐
勇闯IT2 分钟前
有多少小于当前数字的数字
java·数据结构·算法
测试19982 分钟前
2025软件测试面试题汇总(接口测试篇)
自动化测试·软件测试·python·测试工具·面试·职场和发展·接口测试
liuqun031929 分钟前
开心灿烂go开发面试题
算法·leetcode·golang
liulilittle41 分钟前
通过高级处理器硬件指令集AES-NI实现AES-256-CFB算法并通过OPENSSL加密验证算法正确性。
linux·服务器·c++·算法·安全·加密·openssl
小皮侠43 分钟前
【算法篇】逐步理解动态规划模型6(回文串问题)
java·开发语言·算法·动态规划
IT猿手43 分钟前
动态多目标进化算法:基于迁移学习的动态多目标粒子群优化算法(TrMOPSO)求解IEEE CEC 2015,提供完整MATLAB代码
算法·matlab·迁移学习·动态多目标进化优化·动态多目标算法
এ᭄画画的北北1 小时前
力扣-279.完全平方数
数据结构·算法·leetcode
实习生小黄1 小时前
双三次贝塞尔曲面-canvas 实现4x4网格图片变化功能
前端·算法
jz_ddk1 小时前
[学习] FIR多项滤波器的数学原理详解:从多相分解到高效实现(完整仿真代码)
学习·算法·matlab
IT猿手1 小时前
动态多目标进化算法:基于迁移学习的动态多目标遗传算法Tr-NSGA-II求解CEC2015,提供完整MATLAB代码
人工智能·算法·机器学习·matlab·迁移学习·动态多目标进化算法·动态多目标优化算法