[2013 ACM/ICPC Asia Regional Nanjing Online C][hdu 4750]Count The Pairs(kruskal + 二分)

时间:2021-11-12 07:59:29

http://acm.hdu.edu.cn/showproblem.php?pid=4750

题意:

定义f(u,v)为u到v每条路径上的最大边的最小值..现在有一些询问..问f(u,v)>=t的点对有所少对,注意(1,2)和(2,1)是不同的点对

分析:

原来最小生成树有一个很鬼畜的结论,那就是一个图的最小生成树中任意两个点的路径中的最大边一定最小。(妈蛋,完全不知道这个)

然后此题就变得很明朗了,用kruskal算法,加边的时候此边连接的两个集合的路径中的最大边就是这个边,存储下来,询问的时候二分查找即可。

[2013 ACM/ICPC Asia Regional Nanjing Online C][hdu 4750]Count The Pairs(kruskal + 二分)[2013 ACM/ICPC Asia Regional Nanjing Online C][hdu 4750]Count The Pairs(kruskal + 二分)
 + ;
  + ;
 
  
 Edge eg[E];
  
  ll ans[E];
 
            }
       }
      ; i <= n; ++i)
         fa[i] = i, cnt[i] = ;
 
     id[] = -;
     ; i < m; ++i) {
         scanf(         ++eg[i].u, ++eg[i].v;
         id[i + ] = eg[i].len;
     }
 
     std::sort(id, id + m + );
     ; i < m; ++i)
         eg[i].len = std::lower_bound(id, id + m + , eg[i].len) - id;
 
     memset(ans, ,  
     std::sort(eg, eg + m, cmp);
          ; i < m; ++i) {
         x = Find(eg[i].u), y = Find(eg[i].v);
                      ans[eg[i].len] = ans[eg[i].len] + cnt[x] * cnt[y];
             cnt[x] += cnt[y];
             fa[y] = x;
         }
     }
 
     ; --i)
         ans[i] = ans[i] + ans[i + ];
 
          scanf(     ) {
         scanf(         l = -, r = m + ;
         ) {
             mid = (l + r) >> ;
                              r = mid;
                              l = mid;
         }
         printf();
     }
 }
    == scanf(         work();
     }
     ;
 }