【八十二】【算法分析与设计】2421. 好路径的数目,928. 尽量减少恶意软件的传播 II,并查集的应用,元素信息绑定下标一起排序,元素通过下标进行绑定

2421. 好路径的数目

给你一棵 n 个节点的树(连通无向无环的图),节点编号从 0n - 1 且恰好有 n - 1 条边。

给你一个长度为 n 下标从 0 开始的整数数组 vals ,分别表示每个节点的值。同时给你一个二维整数数组 edges ,其中 edges[i] = [a(i), b(i)] 表示节点 a(i)b(i)( )之间有一条 无向 边。

一条 好路径 需要满足以下条件:

  1. 开始节点和结束节点的值 相同

  2. 开始节点和结束节点中间的所有节点值都 小于等于 开始节点的值(也就是说开始节点的值应该是路径上所有节点的最大值)。

请你返回不同好路径的数目。

注意,一条路径和它反向的路径算作 同一 路径。比方说, 0 -> 11 -> 0 视为同一条路径。单个节点也视为一条合法路径。

示例 1:

输入:vals = [1,3,2,1,3], edges = [[0,1],[0,2],[2,3],[2,4]] 输出:6 解释:总共有 5 条单个节点的好路径。 还有 1 条好路径:1 -> 0 -> 2 -> 4 。 (反方向的路径 4 -> 2 -> 0 -> 1 视为跟 1 -> 0 -> 2 -> 4 一样的路径) 注意 0 -> 2 -> 3 不是一条好路径,因为 vals[2] > vals[0] 。

示例 2:

输入:vals = [1,1,2,2,3], edges = [[0,1],[1,2],[2,3],[2,4]] 输出:7 解释:总共有 5 条单个节点的好路径。 还有 2 条好路径:0 -> 1 和 2 -> 3 。

示例 3:

输入:vals = [1], edges = [] 输出:1 解释:这棵树只有一个节点,所以只有一条好路径。

提示:

  • n == vals.length

  • 1 <= n <= 3 * 10(4)

  • 0 <= vals[i] <= 10(5)

  • edges.length == n - 1

  • edges[i].length == 2

  • 0 <= a(i), b(i) < n

  • a(i) != b(i)

  • edges 表示一棵合法的树。

1.

对于所有的边元素,我们可以得到他max的vals的值,并且可以构造一个max数组一一对应.

max进行排序,从小到大排序,并且是绑定下标进行排序.

这样我们就可以按照元素的max信息顺序进行访问元素本身.

2.

对于每一个边,如果两个点的最大值是不一样的,那么肯定不是好路径.

如果两个点的最大值是一样的,那么好路径的个数是两个点所在集合元素个数相乘,然后这两个结合合并.

 
class Solution {
public:vector<int> vals;//每一个点的值vector<vector<int>> edges;//每一个边信息int ret;//存储结果using p = pair<int, int>;//定义pairvector<p> max_edge;//为了绑定max和下标一起排序int n;//点的个数struct node {//存储所有元素对于的其他的信息int max;//存储集合最大值信息int size;//存储集合最大值个数信息int father;//并查集核心部分};vector<node> vec;//所有元素对应的其他的信息vector<int> st;//用vector模拟栈int _find(int i) {//并查集中找i下标对应集合的代表元素的下标while (!(i == vec[i].father)) {//当前节点是i,出口时i=vec[i].fatherst.push_back(i);//用栈存储走过的路i = vec[i].father;//进入下一节点的操作}while (st.size()) {//将所有走过的路的father直接设置为集合代表元素的下标//扁平化处理vec[st.back()].father = i;st.pop_back();}return i;}bool isSameSet(int x, int y) { return _find(x) == _find(y); }//判断是否是同一个集合void _union(int x, int y) {//将两个集合进行合并if (_find(x) == _find(y))return;else {int rx = _find(x);int ry = _find(y);int maxx = vec[rx].max;int maxy = vec[ry].max;if (maxx == maxy)//如果最大值是一样的,那么最大值的个数相加vec[ry].size = vec[rx].size + vec[ry].size;else if (maxx > maxy)//如果最大值不是一样的,只需要考虑这一种情况,修改ry的sizevec[ry].size = vec[rx].size;vec[ry].max = max(vec[rx].max, vec[ry].max);vec[rx].father = vec[ry].father;}}void init() {//正式解题操作之前的初始化操作n = edges.size();for (int i = 0; i < n; i++) {max_edge.push_back({max(vals[edges[i][0]], vals[edges[i][1]]), i});//vector中绑定max值和下标,一起排序}sort(max_edge.begin(), max_edge.end(),[](const auto& a, const auto& b) { return a.first < b.first; });//sort的lambda写法//第三个参数直接以函数的信息写出来了//关注于第三个参数,直接使用auto,用于判断a是否位于b的前面//pair中first存储的是max信息vec.clear(), vec.resize(vals.size());for (int i = 0; i < vals.size(); i++) {//初始化vec信息vec[i].father = i;vec[i].max = vals[i];vec[i].size = 1;}}void solve() {//正式开始解题步骤for (int i = 0; i < max_edge.size(); i++) {int id = max_edge[i].second;//找到边的下标int id1, id2;id1 = edges[id][0];//找边的两个点id2 = edges[id][1];int max_id1 = vec[_find(id1)].max;int max_id2 = vec[_find(id2)].max;if (max_id1 == max_id2) {//如果最大值是一样的,维护retret += vec[_find(id1)].size * vec[_find(id2)].size;_union(id1, id2);} else {//如果不是一样的_union(id1, id2);}}ret += vals.size();//最后不要忘记了ret需要加入没一个独立的点}int numberOfGoodPaths(vector<int>& _vals, vector<vector<int>>& _edges) {vals = _vals;edges = _edges;init();solve();return ret;}
};

928. 尽量减少恶意软件的传播 II

给定一个由 n 个节点组成的网络,用 n x n 个邻接矩阵 graph 表示。在节点网络中,只有当 graph[i][j] = 1 时,节点 i 能够直接连接到另一个节点 j

一些节点 initial 最初被恶意软件感染。只要两个节点直接连接,且其中至少一个节点受到恶意软件的感染,那么两个节点都将被恶意软件感染。这种恶意软件的传播将继续,直到没有更多的节点可以被这种方式感染。

假设 M(initial) 是在恶意软件停止传播之后,整个网络中感染恶意软件的最终节点数。

我们可以从 initial删除一个节点并完全移除该节点以及从该节点到任何其他节点的任何连接。

请返回移除后能够使 M(initial) 最小化的节点。如果有多个节点满足条件,返回索引 最小的节点

示例 1:

输入:graph = [[1,1,0],[1,1,0],[0,0,1]], initial = [0,1] 输出:0

示例 2:

输入:graph = [[1,1,0],[1,1,1],[0,1,1]], initial = [0,1] 输出:1

示例 3:

输入:graph = [[1,1,0,0],[1,1,1,0],[0,1,1,1],[0,0,1,1]], initial = [0,1] 输出:1

提示:

  • n == graph.length

  • n == graph[i].length

  • 2 <= n <= 300

  • graph[i][j]01.

  • graph[i][j] == graph[j][i]

  • graph[i][i] == 1

  • 1 <= initial.length < n

  • 0 <= initial[i] <= n - 1

  • initial 中每个整数都不同

1.

解题的思路,首先有许多个点,有毒点有正常点.

把所有的正常点相互连接的部分合并成一个集合.

遍历所有的毒点,访问所有的与毒点相连的集合,给这个集合打上标签,这个标签是这个集合与之相连的毒点.

因此这个标签实际上可能有多个,也可有可能只有一个,或者没有.

最后搞一个count数组对应所有的毒点,遍历所有的集合,如果该集合的标签只有一个,那么对应的count加上该集合的所有元素.

遍历count找到最大的count值并且找到下标,通过下标找到对应毒点的下标.

2.

需要注意的一点是,如果count值相等,需要返回较小值的毒点下标.

因此我们提前将it数组进行排序.

 
class Solution {
public:vector<vector<int>> g;//图的临界矩阵的存储方式vector<int> it;//毒点的下标,位置int ret;//最终结果存储遍历int n;//邻接矩阵的端点个数,也就是有多少个点struct node {//给每一个点,元素绑定一些信息,通过下标绑定所有的元素和元素对应的信息int father;//每一个元素的father下标,并查集的核心set<int> point_yuan;//每一个元素的毒点信息,实际上只需要关心集合头元素的该信息即可int size;//集合的元素个数};map<int, int> point_yuan_index;//将元素和下标进行映射,这样我们就可以通过元素找到对应的下标进而找到对应的所有的信息vector<node> v;//v表示所有元素对应的信息vector<int> st;//用vector模拟栈的实现,用于对并查集进行扁平化处理set<int> setnum;//set容器进行去重操作,存储所有的集合代表元素,代表元素的下标vector<int> count;//对应所有毒点的count,表示每一个毒点如果删去可以拯救的点的个数int _find(int i) {//并查集中查找的方法while (!(i == v[i].father)) {//当前节点是i位置,出口时i==v[i].fatherst.push_back(i);//对于每一个访问的下标进行存储,用栈存储走过的路i = v[i].father;//进入下一节点的操作}while (!st.empty()) {//对于所有走过的路,修改其father直接改为i位置,此时i是集合的代表元素的下标v[st.back()].father = i;st.pop_back();}return i;}void _union(int x, int y) {//集合的合并操作int rx = _find(x), ry = _find(y);//首先计算x下标,y下标所在集合的代表元素下标if (rx == ry)//如果所在集合一样就不需要合并return;else {//如果所在集合不一样就需要合并v[ry].size += v[rx].size;//对于每一个集合需要维护的信息是father和size信息//首先维护size信息,合并之后集合代表元素的下标是ry//所以需要改变ry的size信息v[rx].father = ry;}}void init() {//初始化函数,解题之前的必要操作sort(it.begin(), it.end());//首先给it进行排序n = g.size();//计算点的个数v.clear(), v.resize(n);//对元素信息进行分配空间for (int i = 0; i < n; i++) {v[i].father = i;//初始化fatherv[i].size = 1;//初始化size}for (int i = 0; i < it.size(); i++) {point_yuan_index[it[i]] = i;//毒点与下标建立映射}count.clear(), count.resize(it.size());ret = it[0];//一开始ret存储it第一个毒点的位置}void solve() {//正式开始解题过程for (int i = 0; i < n; i++) {//首先合并所有正常点for (int j = i + 1; j < n; j++) {if (g[i][j] == 1 && !point_yuan_index.count(i) &&!point_yuan_index.count(j))_union(i, j);//如果i,j都是正常点,并且ij之间有边,那么就合并在一起}}for (int i = 0; i < it.size(); i++) {//给所有的集合打标签int id = it[i];//找到毒点对于的下标for (int j = 0; j < n; j++) {if (id == j)continue;if (g[id][j] == 1) {v[_find(j)].point_yuan.insert(id);//set容器进行去重操作}}}for (int i = 0; i < n; i++) {//用set容器存储所有的集合的代表元素下标if (!point_yuan_index.count(i))//不考虑毒点setnum.insert(_find(i));}for (auto& x : setnum) {//遍历所有的集合点if (v[x].point_yuan.size() > 1)continue;else if (v[x].point_yuan.size() == 1) {//如果毒点连接只有一个,那么对于毒点的count值加上该集合的元素个数int id = point_yuan_index[*v[x].point_yuan.begin()];count[id] += v[x].size;}}int maxnum = 0;for (int i = 0; i < count.size(); i++) {//最后遍历count数组,找到最大值对应的毒点位置if (count[i] > maxnum) {maxnum = count[i];ret = it[i];}}}int minMalwareSpread(vector<vector<int>>& _graph, vector<int>& _initial) {g = _graph, it = _initial;init();solve();return ret;}
};

结尾

最后,感谢您阅读我的文章,希望这些内容能够对您有所启发和帮助。如果您有任何问题或想要分享您的观点,请随时在评论区留言。

同时,不要忘记订阅我的博客以获取更多有趣的内容。在未来的文章中,我将继续探讨这个话题的不同方面,为您呈现更多深度和见解。

谢谢您的支持,期待与您在下一篇文章中再次相遇!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/web/8154.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【Vue】Vue packages version mismatch(vue 和 vue-template-compiler)

报错&#xff1a;Vue packages version mismatch 原因&#xff1a;vue和vue-template-compiler版本不一样解决&#xff1a;如上vue版本为 2.6.14&#xff0c;vue-template-comiler版本为2.7.16。将vue-template-comiler版本设置为和vue版本一致即可。 npm install vue-templat…

【稳定检索|EI会议】✅2024年食品科学与生物医学国际会议(ICFSBS 2024)✅

2024 International Conference on Food Science and Biomedical Sciences 一、大会信息 会议名称&#xff1a;2024年食品科学与生物医学国际会议会议简称&#xff1a;ICFSBS 2024收录检索&#xff1a;提交Ei Compendex,CPCI,CNKI,Google Scholar等会议官网&#xff1a;http:/…

Windows安装RabbitMQ教程(附安装包)

需要两个安装包 Erlang 安装包: https://download.csdn.net/download/Brevity6/89274663 (自己从官网下载也可以) RabbitMQ Windows 安装包&#xff1a; https://download.csdn.net/download/Brevity6/89274667 (自己从官网下载也可以) Erlang安装 Erlang安装傻瓜式下一…

武汉星起航:亚马逊:跨境电商领军平台,中国卖家全球拓展的首选

2015年&#xff0c;亚马逊全球开店业务正式进入中国&#xff0c;为中国卖家带来了全新的跨境电商机遇。如今&#xff0c;亚马逊已在全球拥有包括美国、加拿大、墨西哥、英国、法国、德国等在内的17大海外站点&#xff0c;为中国卖家提供了广阔的销售市场。武汉星起航将详细探讨…

查看window电脑的凭据密码(netpass)

软件地址 根据自身的windows版本下载对应的版本&#xff0c;解压密码&#xff1a;ntps5291#

光伏远动通讯屏的组成

光伏远动通讯屏的组成 远动通讯屏主要用于电力系统数据采集与转发&#xff0c;远动通讯屏能够采集站内的各种数据&#xff0c;如模拟量、开关量和数字量等&#xff0c;并通过远动通讯规约将必要的数据上传至集控站或调度系统。这包括但不限于主变和输电线路的功率、电流、电压等…

温暖家居新风尚,能率壁挂炉——设计新风尚,体验再升级

随着家居品质要求的提升&#xff0c;现代人对家居的舒适性和设计感有了更高的追求。壁挂炉&#xff0c;作为现代家居中不可或缺的一部分&#xff0c;其重要性日益凸显。中国国际供热通风空调、卫浴及舒适家居系统展览会&#xff08;ISH China & CIHE&#xff09;将于2024年…

轻松搞定!png格式图片怎么弄?详细方法一网打尽

在数字图像处理的日常应用中&#xff0c;PNG格式因其高质量的图像和支持透明背景的特性而备受青睐。然而&#xff0c;对于一些刚刚接触这一格式的用户&#xff0c;或者在处理PNG图片时遇到一些挑战的用户来说&#xff0c;如何轻松而有效地处理这种格式的图像可能是个问题。png格…

汽车灯罩使用聚碳酸酯(PC)和PMMA(亚克力)哪个更好?汽车车灯的灯罩如果破损破裂破洞了要怎么修复?

汽车灯罩使用聚碳酸酯&#xff08;PC&#xff09;和PMMA&#xff08;亚克力&#xff09;哪个更好&#xff1f; 聚碳酸酯&#xff08;PC&#xff09;和PMMA&#xff08;亚克力&#xff09;都是汽车灯罩常见的材质&#xff0c;它们各自具有独特的优点和特性&#xff0c;因此选择…

Seata之TCC 模式的使用

系列文章目录 文章目录 系列文章目录前言前言 前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站,这篇文章男女通用,看懂了就去分享给你的码吧。 Seata 是一款开源的分布式事务解决方案,致力于在微服务架构下提供高性能…

如何保护数据安全?迅软DSE加密系统给信息撑把保护伞!

信息安全当然需要保护&#xff0c;不然企业的信息可以发给任何人&#xff0c;普通信息还好&#xff0c;如果是重要机密呢&#xff0c;企业重要信息被发出去后可能会造成一些麻烦&#xff0c;所以可以使用加密系统&#xff0c;对数据进行安全保护&#xff0c;防止泄密问题&#…

集成学习案例-幸福感预测

集成学习案例一 &#xff08;幸福感预测&#xff09; 背景介绍 此案例是一个数据挖掘类型的比赛——幸福感预测的baseline。比赛的数据使用的是官方的《中国综合社会调查&#xff08;CGSS&#xff09;》文件中的调查结果中的数据&#xff0c;其共包含有139个维度的特征&#xf…

C++ | Leetcode C++题解之第74题搜索二维矩阵

题目&#xff1a; 题解&#xff1a; class Solution { public:bool searchMatrix(vector<vector<int>>& matrix, int target) {int m matrix.size(), n matrix[0].size();int low 0, high m * n - 1;while (low < high) {int mid (high - low) / 2 l…

用一个故事告诉你协程到底是什么

神秘使者 “久闻Java语言跨越平台&#xff0c;框架众多&#xff0c;不过二十年功夫&#xff0c;就已晋升天下第一编程语言&#xff0c;今日一见&#xff0c;果然名不虚传呐&#xff01;” “使者先生您过奖了&#xff0c;咱们快些走&#xff0c;国王陛下已经等候多时了” 今…

ansible——INVENTORY主机清单

一、Inventory主机清单 Inventory支持对主机进行分组&#xff0c;每个组内可以定义多个主机&#xff0c;每个主机都可以定义在任何一个或多个主机组内 二、Inventory主机清单部署 2.1 前期准备 systemctl stop firewalld setenforce 0 yum install epel-release -y yum install…

【知识碎片】2024_05_07

今天记录了两个代码和C的几个破碎知识。 第一段代码是基础型的&#xff0c;关于数组。第二段代码是二分的&#xff0c;一开始没通过全部案例&#xff0c;值得再看。 每日代码 1.记负均正 输入一个数组&#xff0c;输出负数的个数&#xff0c;整数的平均值&#xff08;0都不参…

【JavaEE初阶系列】——Servlet运行原理以及Servlet API详解

目录 &#x1f6a9;Servlet运行原理 &#x1f6a9;Servlet API 详解 &#x1f393;HttpServlet核心方法 &#x1f393;HttpServletRequest核心方法 &#x1f388;核心方法的使用 &#x1f534;获取请求中的参数 &#x1f4bb;query string &#x1f4bb;直接通过form表…

Matlab图像中加入脉冲噪声、高斯噪声并用均值滤波、中值滤波进行滤波处理

一、脉冲噪声和高斯噪声简介 脉冲噪声和高斯噪声是两种常见的信号干扰类型&#xff0c;它们的特性和影响各不相同&#xff1a; 脉冲噪声&#xff08;Impulse Noise&#xff09;&#xff1a; 在图像中&#xff0c;脉冲噪声表现为随机出现的亮点或暗点&#xff0c;这些噪声点通常…

vector介绍与使用【C++】

C vector 前言一、vector的介绍c文档介绍简介 二、vector的定义和使用vector的定义vector代码演示 vector的使用vector iterator 的使用vector 空间增长问题vector 增删查改vector 迭代器失效问题引起底层空间改变eraseg与vs检测比较string迭代器失效 vector 在OJ中的使用只出现…

pandas 预处理

文章目录 第1关&#xff1a;数据读取与合并第2关&#xff1a;数据清洗第3关&#xff1a;数据转换 第1关&#xff1a;数据读取与合并 任务描述 本关任务&#xff1a;加载 csv 数据集&#xff0c;实现 DataFrame 合并。 知识讲解 Pandas 模块导入 import pandas as pd 读取 cs…