【1615. 最大网络秩】

文章介绍了如何解决一个关于图论的问题,即在一个基础设施网络中找出最大网络秩。方法包括枚举所有城市对并计算度数之和,以及使用贪心算法只关注度数最高的两个城市集合来优化复杂度。两种方法的时间复杂度分别达到了O(n^2)和O(n+m),其中n是城市数量,m是道路数量。
摘要由CSDN通过智能技术生成

来源:力扣(LeetCode)

描述:

n 座城市和一些连接这些城市的道路 roads 共同组成一个基础设施网络。每个 roads[i] = [ai, bi] 都表示在城市 aibi 之间有一条双向道路。

两座不同城市构成的 城市对网络秩 定义为:与这两座城市 直接 相连的道路总数。如果存在一条道路直接连接这两座城市,则这条道路只计算 一次

整个基础设施网络的 最大网络秩 是所有不同城市对中的 最大网络秩

给你整数 n 和数组 roads,返回整个基础设施网络的 最大网络秩

示例 1:

1

输入:n = 4, roads = [[0,1],[0,3],[1,2],[1,3]]
输出:4
解释:城市 01 的网络秩是 4,因为共有 4 条道路与城市 01 相连。位于 01 之间的道路只计算一次。

示例 2:

2

输入:n = 5, roads = [[0,1],[0,3],[1,2],[1,3],[2,3],[2,4]]
输出:5
解释:共有 5 条道路与城市 12 相连。

示例 3:

输入:n = 8, roads = [[0,1],[1,2],[2,3],[2,4],[5,6],[5,7]]
输出:5
解释:25 的网络秩为 5,注意并非所有的城市都需要连接起来。

提示:

  • 2 <= n <= 100
  • 0 <= roads.length <= n * (n - 1) / 2
  • roads[i].length == 2
  • 0 <= ai, bi <= n-1
  • ai != bi
  • 每对城市之间 最多只有一条 道路相连

方法一:枚举

思路与算法

根据题意可知,两座不同城市构成的城市对的网络秩定义为:与这两座城市直接相连的道路总数,这两座城市之间的道路只计算一次。假设城市 x 的度数为 degree[x],则此时我们可以知道城市对 (i, j) 的网络秩为如下:

  • 如果 i 与 j 之间没有道路连接,则此时 (i, j) 的网络秩为 degree[i] + degree[j];
  • 如果 i 与 j 之间存在道路连接,则此时 (i, j) 的网络秩为 degree[i] + degree[j] − 1;

根据以上求网络秩的方法,我们首先求出所有城市在图中的度数,然后枚举所有可能的城市对 (i, j),求出城市对 (i, j) 的网络秩,即可找到最大的网络秩。

代码:

class Solution {
public:
    int maximalNetworkRank(int n, vector<vector<int>>& roads) {
        vector<vector<bool>> connect(n, vector<bool>(n, false));
        vector<int> degree(n, 0);
        for (auto v : roads) {
            connect[v[0]][v[1]] = true;
            connect[v[1]][v[0]] = true;
            degree[v[0]]++;
            degree[v[1]]++;
        }

        int maxRank = 0;
        for (int i = 0; i < n; i++) {
            for (int j = i + 1; j < n; j++) {
                int rank = degree[i] + degree[j] - (connect[i][j] ? 1 : 0);
                maxRank = max(maxRank, rank);
            }
        }
        return maxRank;
    }
};

执行用时:60 ms, 在所有 C++ 提交中击败了66.76%的用户
内存消耗:30.6 MB, 在所有 C++ 提交中击败了38.27%的用户
复杂度分析
时间复杂度:O(n2),其中 n 表示给城市的数目。我们需要枚举所有可能的城市对,最多有 n2个城市对。
空间复杂度:O(n2)。需要记录图中所有的城市之间的连通关系,需要的空间为 O(n2)。如果用邻接表存储连通关系,空间复杂度可以优化到 O(n+m),其中 m 表示 roads 的长度。

方法二:贪心

思路与算法

我们可以对解法一中的方法继续优化。设 first 表示所有节点中度数的最大值,second 表示所有节点中度数的次大值,实际我们只需要考虑度数为最大值与次大值的城市即可,其余即可城市可以无须考虑,原因如下:

  • 已知最大值 first 与次大值 second,则此时可以知道当前最差的情况下,假设这两城市存在连接,则最大的网络秩为 first + second − 1;
  • 假设存在度数比 second 的城市 x,则此时 degree[x] < second,此时含有 x 构成的城市对的最大网络秩不超过 degree[x] + first,此时一定满足 degree[x] + first ≤ second + first;

综上可以得出结论选择最大或者次大度数的城市一定是最优的。我们可以求出度数为 first 的城市集合 firstArr,同时求出度数为 second 的城市集合 secondArr。设城市的总数量为 n,道路的总数量为 m,集合 firstArr 的数量为 x,则此时该集合可以构造的城市对数量为 x ( x − 1 ) 2 \frac{x(x-1)}{2} 2x(x1) ,分以下几种情况来讨论:

  • 如果 x = 1,此时我们必须选择 firstArr 中唯一的城市,另一个城市只能在 secondArr 中选择,枚举 secondArr 中的每个城市,找到最大的网络秩即可,此时需要的时间复杂度为 O(n);
  • 如果 x>1 时,分类讨论如下:
    • 如果满足 x 2 \frac{x}{2} 2x > m 时,此时集合 firstArr 一定存在一对城市,他们之间没有道路连接,此时最大的网络秩即为 2 × first;
    • 如果满足 x 2 \frac{x}{2} 2x ≤ m 时,此时枚举集合 firstArr 中所有不同的城市对即可,此时不需要再考虑次大的城市集合 secondArr,因为此时一定满足 2 × first − 1 ≥ first + second > 2 × second ,此时时间复杂度不超过 O(m);

因此通过以上分析,上述解法的时间复杂度为 O(n+m)。

代码:

class Solution {
public:
    int maximalNetworkRank(int n, vector<vector<int>>& roads) {
        vector<vector<bool>> connect(n, vector<bool>(n, false));
        vector<int> degree(n);
        for (auto road : roads) {
            int x = road[0], y = road[1];
            connect[x][y] = true;
            connect[y][x] = true;
            degree[x]++;
            degree[y]++;
        }

        int first = -1, second = -2;
        vector<int> firstArr, secondArr;
        for (int i = 0; i < n; ++i) {
            if (degree[i] > first) {
                second = first;
                secondArr = firstArr;
                first = degree[i];
                firstArr.clear();
                firstArr.emplace_back(i);
            } else if (degree[i] == first) {
                firstArr.emplace_back(i);
            } else if (degree[i] > second){
                secondArr.clear();
                second = degree[i];
                secondArr.emplace_back(i);
            } else if (degree[i] == second) {
                secondArr.emplace_back(i);
            }
        }
        if (firstArr.size() == 1) {
            int u = firstArr[0];
            for (int v : secondArr) {
                if (!connect[u][v]) {
                    return first + second;
                }
            }
            return first + second - 1;
        } else {
            int m = roads.size();
            if (firstArr.size() * (firstArr.size() - 1) / 2 > m) {
                return first * 2;
            }
            for (int u: firstArr) {
                for (int v: firstArr) {
                    if (u != v && !connect[u][v]) {
                        return first * 2;
                    }
                }
            }
            return first * 2 - 1;
        }
    }
};

执行用时:60 ms, 在所有 C++ 提交中击败了66.76%的用户
内存消耗:30.7 MB, 在所有 C++ 提交中击败了36.59%的用户
复杂度分析
时间复杂度:O(n+m),其中 n 表示给定的数字 n,m 表示城市之间的道路总数。计算城市的度数需要的时间为 O(m),找到城市中最大度数和次大度数城市集合需要的时间为 O(n),计算城市对中最大的网络秩需要的时间为 O(m),因此总的时间复杂度为 O(m+n)。
空间复杂度:O(n2)。需要记录图中所有的城市之间的联通关系,需要的空间为 O(n2),记录所有节点的度需要的空间为 O(n),记录最大度数与次大度数的城市集合需要的空间为 O(n),因此总的空间复杂度为 O(n2)。如果用邻接表存储连通关系,空间复杂度可以优化到 O(n+m),其中 m 表示 roads 的长度。
author:LeetCode-Solution

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

千北@

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值