给你一个无向图(原始图),图中有 n
个节点,编号从 0
到 n - 1
。你决定将图中的每条边 细分 为一条节点链,每条边之间的新节点数各不相同。
图用由边组成的二维数组 edges
表示,其中 edges[i] = [ui, vi, cnti]
表示原始图中节点 ui
和 vi
之间存在一条边,cnti
是将边 细分 后的新节点总数。注意,cnti == 0
表示边不可细分。
要 细分 边 [ui, vi]
,需要将其替换为 (cnti + 1)
条新边,和 cnti
个新节点。新节点为 x1
, x2
, ..., xcnti
,新边为 [ui, x1]
, [x1, x2]
, [x2, x3]
, ..., [xcnti+1, xcnti]
, [xcnti, vi]
。
现在得到一个 新的细分图 ,请你计算从节点 0
出发,可以到达多少个节点?如果节点间距离是 maxMoves
或更少,则视为 可以到达 。
给你原始图和 maxMoves
,返回 新的细分图中从节点 0
出发 可到达的节点数 。
示例 1:
输入:edges = [[0,1,10],[0,2,1],[1,2,2]], maxMoves = 6, n = 3 输出:13 解释:边的细分情况如上图所示。 可以到达的节点已经用黄色标注出来。
示例 2:
输入:edges = [[0,1,4],[1,2,6],[0,2,8],[1,3,1]], maxMoves = 10, n = 4 输出:23
示例 3:
输入:edges = [[1,2,4],[1,4,5],[1,3,1],[2,3,4],[3,4,5]], maxMoves = 17, n = 5 输出:1 解释:节点 0 与图的其余部分没有连通,所以只有节点 0 可以到达。
提示:
0 <= edges.length <= min(n * (n - 1) / 2, 104)
edges[i].length == 3
0 <= ui < vi < n
0 <= cnti <= 104
0 <= maxMoves <= 109
1 <= n <= 3000
分析:题目意思从源点0出发,最大距离能走maxMoves,然后求出可到达的节点数。题目给出了edges数组,这个就是边和权值,我们有了边和权值,就能最初源点到其他点的最短路径。这里的在初始化权值的时候要注意,这个不能要加1,因为自己就是一个点,然后加上自己就是要加上1,然后用dijkstra求出最短路径。得到最短距离dist数组,计算原始图中所有可达节点的数量,加入结果中。
然后再考虑细分边上的节点。我们考虑某一条边上的两个节点。让这两个节点分别沿着这条边继续往对方节点走,看看有多少个细分节点是可达的。如果它们相遇了,也就是他们走过的节点数量大于这条边上的细分节点数量,那么说明这条边上的所有细分节点都是可达的。否则,将他们走过的节点数量加入到结果中。
AC代码:
- class Solution {
- int res=0 ;
- int [][] map ;
- int dist[];
-
-
- public int reachableNodes(int[][] edges, int maxMoves, int n) {
- map= new int[n][n] ;
- dist = new int[n];
-
- //进行初始化
- for (int[] ints : map) {
-
- Arrays.fill(ints,Integer.MAX_VALUE);
- }
- Arrays.fill(dist,Integer.MAX_VALUE);
- for (int[] edge : edges) {
- //这里所有的距离都加1,加上自己的那个点
- map[edge[1]][edge[0]] = edge[2]+1 ;
- map[edge[0]][edge[1]] = edge[2]+1 ;
- }
- dijkstra(0,map,dist);
-
- for(int d : dist){
-
- if(d<=maxMoves){
- res++ ;
- }
- }
-
- for (int[] edge : edges) {
- int a = edge[0] , b =edge[1] , cnta = 0 ,cntb= 0 ;
-
- if (dist[a]
- //看看还能多走几个细分节点
- cnta = Math.min(edge[2],maxMoves-dist[a]);
- }
- if (dist[b]
- cntb= Math.min(edge[2],maxMoves-dist[b]);
- }
-
- res+= Math.min(edge[2],cntb+cnta) ;
-
- }
-
-
- return res ;
-
- }
-
- public void dijkstra(int v , int [][]a ,int dist[]){
-
- int n = dist.length-1 ;
- if (v<0||v>=n ){
- return;
- }
-
- //标记是否访问
- boolean [] isVisited = new boolean[n+1] ;
- for (int i =0 ;i<=n ;i++){
- dist[i] = a[v][i] ;
- isVisited[i] =false ;
- }
-
- //自己到自己为0
- dist[v] = 0 ;
- isVisited[v] =true ;
-
- //遍历所有能通的点
- for (int i = 0 ;i<= n;i++){
- int temp = Integer.MAX_VALUE ;
- int u =v ;
- for (int j =0 ;j<=n;j++){
- //如果没有访问,而且距离小于temp,最小值就交换
- if (!isVisited[j]&&temp>dist[j]){
- u=j ;
- temp=dist[j] ;
- }
- }
-
- //标记u点访问过了
- isVisited[u] = true ;
-
- //在从u点找到u到其他点最小值
- for (int j = 0;j<=n;j++){
- if (!isVisited[j]&&a[u][j]
- int newDist = dist[u] + a[u][j] ;
- if (newDist
- dist[j] = newDist;
-
- }
- }
- }
- }
-
- }
- }
-
相关阅读:
校正叠加(calibrated stacking)方法—技术
Mybatis动态SQL踩坑记
苹果搜索广告投放(ASA)因元数据被拒有哪些情况呢(下)
css中元素长度单位
前端 JS 经典:Math 常用方法汇总
JSON数据获取指南!
数据结构与算法-队列
Android 基础知识3-1项目目录结构
墨者学院——登录密码重置漏洞分析溯源
JavaScript【实例、静态方法与属性、原型链、instanceof 运算符、Object 对象的相关方法、对象的继承、多重继承、严格模式】(十九)
-
原文地址:https://blog.csdn.net/weixin_54046648/article/details/128049448