C++算法:有向图访问计数的原理及实现

简介: C++算法:有向图访问计数的原理及实现

题目

现有一个有向图,其中包含 n 个节点,节点编号从 0 到 n - 1 。此外,该图还包含了 n 条有向边。

给你一个下标从 0 开始的数组 edges ,其中 edges[i] 表示存在一条从节点 i 到节点 edges[i] 的边。

想象在图上发生以下过程:

你从节点 x 开始,通过边访问其他节点,直到你在 此过程 中再次访问到之前已经访问过的节点。

返回数组 answer 作为答案,其中 answer[i] 表示如果从节点 i 开始执行该过程,你可以访问到的不同节点数。

2 <= n <= 100000

无自环。

原理分析

如果只有一个连通区域,则有且只有一环。反证法:假定没有环,除源点外,还可以到达n个端点,共n+1个端点,与共有n个端点重复。假定有x个环,则不重复端点数为:1+n-x。当且仅当x为1是,不重复端点数为n。

当有多个连通区域时,任何一个连通区域都有且只有一个环。下面分两步来证明:一,此连通区域必定有环。二,此区域不存在两个或更多的环。

假定此区域的一条边为i0->edges[i0],edges[i0]简称为i1。如果没有环, 则edges[i1](简称为i2)也在此连通区域,edges[i2](简称i3)也在此连通区域,i4.... 。此连通区域的点数无限,和端点数小于等于n矛盾。

由于出度为1,所以进入环后,无法离开环。自然没第二个环。

编码思路

根据拓扑排序,发现那些点在环上。

根据并集查找获取各连通区域。

统计各连通区域在环上的点数。

求环上各点可以到达的点数,就是此环长度(端点数)。

DFS非环上各点可以到达的点数。就是到环的距离+此环的长度。

拓扑排序和并集查找已经封装,可以直接使用。

核心源码

class CTestTS : public CTopSort
{
public:
    // 通过 CTopSort 继承
    virtual void OnDo(int pre, int cur) override
    {
        m_vCycle[cur] = false;
    }
    vector<int> m_vCycle;
};
class Solution {
public:
    vector<int> countVisitedNodes(vector<int>& edges) {
        m_c = edges.size();
        vector<vector<int>> vNeiB(m_c);
        CUnionFind uf(m_c);
        for (int i = 0; i < edges.size(); i++)
        {
            vNeiB[i].emplace_back(edges[i]);
            uf.Union(i, edges[i]);
        }
        m_ts.m_vCycle.assign(m_c, true);        
        m_ts.Init(vNeiB);
        m_vDis.resize(m_c, -1);
        //环可能处于不同的联通区域
        std::unordered_map<int, int> mRegionNode;//各联通区域环的端点数
        for (int i = 0; i < m_c; i++)
        {
            if (m_ts.m_vCycle[i])
            {
                mRegionNode[uf.GetConnectRegionIndex(i)]++;
            }
        }
        for (int i = 0; i < m_c; i++)
        {
            if (m_ts.m_vCycle[i])
            {
                m_vDis[i] = mRegionNode[uf.GetConnectRegionIndex(i)];
            }
        }
        for (int i = 0; i < m_c; i++)
        {
            dfs(i, edges);
        }
        return m_vDis;
    }
    int dfs(int cur,const vector<int>& edges)
    {
        if (-1 != m_vDis[cur])
        {
            return  m_vDis[cur];
        }
        return m_vDis[cur] = dfs(edges[cur], edges) + 1;
    }
    vector<int> m_vDis;
    CTestTS m_ts;
    int m_c;
};

测试用代码

int main()
{
    vector<int> edges = { 1,2,3,4,0 };
    //vector<int> edges = { 1,2,0,0 };
    Solution slu;
    auto res = slu.countVisitedNodes(edges);
}

测试环境

Win10 +VS2022 + C++17

相关下载

源码:可直接运行

https://download.csdn.net/download/he_zhidan/88388717


其它

视频课程

如果你觉得复杂,想从简单的算法开始,可以学习我的视频课程。

https://edu.csdn.net/course/detail/38771

我的其它课程

https://edu.csdn.net/lecturer/6176

测试环境

win7 VS2019 C++17 或Win10 VS2022 Ck++17

相关下载

算法精讲《闻缺陷则喜算法册》doc版

https://download.csdn.net/download/he_zhidan/88348653

相关文章
|
7月前
|
存储 监控 算法
防止员工泄密软件中文件访问日志管理的 Go 语言 B + 树算法
B+树凭借高效范围查询与稳定插入删除性能,为防止员工泄密软件提供高响应、可追溯的日志管理方案,显著提升海量文件操作日志的存储与检索效率。
210 2
|
12月前
|
存储 监控 算法
基于 C++ 哈希表算法实现局域网监控电脑屏幕的数据加速机制研究
企业网络安全与办公管理需求日益复杂的学术语境下,局域网监控电脑屏幕作为保障信息安全、规范员工操作的重要手段,已然成为网络安全领域的关键研究对象。其作用类似网络空间中的 “电子眼”,实时捕获每台电脑屏幕上的操作动态。然而,面对海量监控数据,实现高效数据存储与快速检索,已成为提升监控系统性能的核心挑战。本文聚焦于 C++ 语言中的哈希表算法,深入探究其如何成为局域网监控电脑屏幕数据处理的 “加速引擎”,并通过详尽的代码示例,展现其强大功能与应用价值。
233 2
|
存储 算法 C++
Windows共享文件:探秘C++实现的B树索引算法奇境
在数字化时代,Windows共享文件的高效管理至关重要。B树算法以其自平衡多路搜索特性,在文件索引与存储优化中表现出色。本文探讨B树在Windows共享文件中的应用,通过C++实现具体代码,展示其构建文件索引、优化数据存储的能力,提升文件检索效率。B树通过减少磁盘I/O操作,确保查询高效,为企业和个人提供流畅的文件共享体验。
|
7月前
|
存储 缓存 算法
如何管理员工上网:基于 Go 语言实现的布隆过滤器访问拦截算法应用
布隆过滤器以空间换时间,通过多哈希函数实现黑名单的高效存储与毫秒级检索,解决传统方案内存占用大、响应慢等问题,助力企业低成本、高效率管理员工上网行为。
300 3
|
12月前
|
监控 算法 数据处理
基于 C++ 的 KD 树算法在监控局域网屏幕中的理论剖析与工程实践研究
本文探讨了KD树在局域网屏幕监控中的应用,通过C++实现其构建与查询功能,显著提升多维数据处理效率。KD树作为一种二叉空间划分结构,适用于屏幕图像特征匹配、异常画面检测及数据压缩传输优化等场景。相比传统方法,基于KD树的方案检索效率提升2-3个数量级,但高维数据退化和动态更新等问题仍需进一步研究。未来可通过融合其他数据结构、引入深度学习及开发增量式更新算法等方式优化性能。
288 17
|
11月前
|
存储 机器学习/深度学习 算法
基于 C++ 的局域网访问控制列表(ACL)实现及局域网限制上网软件算法研究
本文探讨局域网限制上网软件中访问控制列表(ACL)的应用,分析其通过规则匹配管理网络资源访问的核心机制。基于C++实现ACL算法原型,展示其灵活性与安全性。文中强调ACL在企业与教育场景下的重要作用,并提出性能优化及结合机器学习等未来研究方向。
275 4
|
10月前
|
存储 监控 算法
基于跳表数据结构的企业局域网监控异常连接实时检测 C++ 算法研究
跳表(Skip List)是一种基于概率的数据结构,适用于企业局域网监控中海量连接记录的高效处理。其通过多层索引机制实现快速查找、插入和删除操作,时间复杂度为 $O(\log n)$,优于链表和平衡树。跳表在异常连接识别、黑名单管理和历史记录溯源等场景中表现出色,具备实现简单、支持范围查询等优势,是企业网络监控中动态数据管理的理想选择。
254 0
|
11月前
|
机器学习/深度学习 存储 算法
基于 C++ 布隆过滤器算法的局域网上网行为控制:URL 访问过滤的高效实现研究
本文探讨了一种基于布隆过滤器的局域网上网行为控制方法,旨在解决传统黑白名单机制在处理海量URL数据时存储与查询效率低的问题。通过C++实现URL访问过滤功能,实验表明该方法可将内存占用降至传统方案的八分之一,查询速度提升约40%,假阳性率可控。研究为优化企业网络管理提供了新思路,并提出结合机器学习、改进哈希函数及分布式协同等未来优化方向。
307 0
|
存储 监控 算法
基于 C++ 哈希表算法的局域网如何监控电脑技术解析
当代数字化办公与生活环境中,局域网的广泛应用极大地提升了信息交互的效率与便捷性。然而,出于网络安全管理、资源合理分配以及合规性要求等多方面的考量,对局域网内计算机进行有效监控成为一项至关重要的任务。实现局域网内计算机监控,涉及多种数据结构与算法的运用。本文聚焦于 C++ 编程语言中的哈希表算法,深入探讨其在局域网计算机监控场景中的应用,并通过详尽的代码示例进行阐释。
263 4
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。