题意:给你一张带权无向图,先求出这张图从点1出发的最短路树,再求在树上经过k个节点最长的路径值,以及个数.

分析:首先求最短路树,跑一遍最短路之后dfs一遍即可建出最短路树.

第二个问题,树分治解决.

对于以root为根的树,所求的路径只会有两种情况.

  1. 存在于root的子树中,不经过root;
  2. 经过root,路径的两端在root的两棵子树中.

    第一种情况,我们交给分治去解决,

    第二种情况,需要知道所有子树中走过j步能到达的最远距离,以及其方案数.通过dfs可以得到这些信息.用这些信息,再去和其他子树的信息结合,去更新答案.
#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
const int INF = 1<<30;
const int MAXN = 1e5+5;
struct Edge{
int v,w,next;
}E[MAXN<<2];
int head[MAXN],tot , son[MAXN], Max[MAXN], siz[MAXN], dep[MAXN] ,now[MAXN];
LL cnt[MAXN], Maxcnt[MAXN] , ansnum;
int maxdep , clk, ansdep, minson;
bool vis[MAXN];
int root,N,M,k; void init()
{
memset(vis,0,sizeof(vis));
memset(head,-1,sizeof(head));
tot = ansnum = clk = 0;
ansdep = 0;
} void Add(int u,int v,int w){
E[tot] =(Edge){v,w,head[u]};
head[u] = tot++;
}
//求子树的重心
void getsize(int u, int fa)
{
siz[u] = 1;
for (int i = head[u]; i != -1; i = E[i].next)
{
int v = E[i].v;
if (v == fa || vis[v])
continue;
getsize(v, u);
siz[u] += siz[v];
}
}
void getroot(int u, int fa, int s)
{
int max1 = 0;
for (int i = head[u]; i != -1; i = E[i].next)
{
int v = E[i].v;
if (v == fa || vis[v])
continue;
getroot(v, u, s);
max1 = max(max1, siz[v]);
}
max1 = max(max1, s - siz[u]);
if (minson > max1)
{
minson = max1;
root = u;
}
}
void getMaxdep(int depp, int u, int fa)
{
maxdep = max(maxdep, depp);
for (int i = head[u]; i != -1; i = E[i].next)
{
int v = E[i].v;
if (v == fa || vis[v])
continue;
getMaxdep(depp + 1, v, u);
}
}
void getdep(int depp, int len, int u, int fa)
{
if (dep[depp] < len)
{
dep[depp] = len;
cnt[depp] = 1;
}
else if (dep[depp] == len)
cnt[depp]++;
if (depp >= k)
return;
for (int i = head[u]; i != -1; i = E[i].next)
{
int v = E[i].v;
if (v == fa || vis[v])
continue;
getdep(depp + 1, len + E[i].w, v, u);
}
}
void getans(int u)
{
vis[u] = 1;
for (int i = head[u]; i != -1; i = E[i].next){
int v = E[i].v;
if (vis[v]) continue;
minson = INF;
getsize(v, -1);
getroot(v, -1, siz[v]);
getans(root);
}
/*
求经过k个节点的最长路径,以及其方案数
答案可能有两种情况, 一是存在于u的子树中,这种情况交给分治处理
二是该路径经过了重心本身,在以下代码中处理
*/
clk++;
now[0] = clk;
dep[0] = 0 ,cnt[0] = 1;
Maxcnt[0] = 1, Max[0] = 0;
for (int i = head[u]; i != -1; i = E[i].next){
int v = E[i].v;
if (vis[v]) continue;
maxdep = -1;
//获取这棵子树的最大深度
getMaxdep(1, v, u);
for (int j = 0; j <= maxdep && j <= k; j++) dep[j] = -1;
//获取这棵子树中经过i个节点,所能走的最长距离以及方案数,
//分别记录在dep[i], 和cnt[i]中
getdep(1, E[i].w, v, u); //根据当前信息和这棵子树的信息更新答案
for (int j = 0; j <= maxdep && j < k; j++){
int tmp = k - j - 1;
if (now[tmp] != clk)
continue;
if (ansdep < Max[tmp] + dep[j]){
ansdep = Max[tmp] + dep[j];
ansnum = Maxcnt[tmp] * cnt[j];
}
else if (ansdep == Max[tmp] + dep[j]){
ansnum += Maxcnt[tmp] * cnt[j];
}
}
//用这个子树的信息更新当前的信息
//Max[i]记录之前已经访问过的子树中,经过i个节点能走过的最长距离
//Maxcnt[i] 记录其方案数
for (int j = 1; j <= maxdep && j < k; j++){
if (now[j] != clk || Max[j] < dep[j]){
Max[j] = dep[j];
Maxcnt[j] = cnt[j];
now[j] = clk;
}
else if (now[j] == clk && Max[j] == dep[j]){
Maxcnt[j] += cnt[j];
}
}
}
vis[u] = 0;
} ///////////最短路树
struct Dij{
struct Edge{
int v, w;
bool operator < (const Edge & rhs) const{
return v<rhs.v;
}
};
vector<Edge> G[MAXN];
int N,d[MAXN];
bool vis[MAXN]; struct HeapNode{
int u,val;
bool operator < (const HeapNode & rhs) const{
return val>rhs.val;
}
};
void init(int N){
this -> N = N;
memset(vis,0,sizeof(vis));
for(int i=0;i<=N;++i) G[i].clear();
} void AddEdge(int u,int v,int w){
G[u].push_back((Edge){v,w});
} void dijkstra(int s){
for(int i=1;i<=N;++i){
sort(G[i].begin(),G[i].end());
} for(int i=0;i<=N;++i) d[i] = INF;
d[s] = 0;
priority_queue< HeapNode > Q;
Q.push((HeapNode){s,0});
while(!Q.empty()){
HeapNode x = Q.top(); Q.pop();
if(vis[x.u]) continue;
vis[x.u] = 1;
int u = x.u, sz = G[u].size();
for(auto & e : G[u]){
int v = e.v;
if(d[v]> d[u]+e.w){
d[v] = d[u] + e.w;
Q.push((HeapNode){v,d[v]});
}
}
}
} void dfs(int u,int fa){
vis[u] = true;
for(auto & e: G[u]){
int v= e.v;
if(v==fa || vis[v]) continue;
if(d[v]== d[u]+e.w){
Add(u,v,e.w);
Add(v,u,e.w);
dfs(v,u);
}
}
}
}G; ////////////////// main
int main()
{
#ifndef ONLINE_JUDGE
freopen("in.txt","r",stdin);
freopen("out.txt","w",stdout);
#endif
int T; scanf("%d",&T);
while(T--){
scanf("%d %d %d",&N, &M, &k);
G.init(N);
int u,v,w;
while(M--){
scanf("%d %d %d",&u, &v, &w);
G.AddEdge(u,v,w);
G.AddEdge(v,u,w);
}
G.dijkstra(1);
init(); //树的初始化
memset(G.vis,0,sizeof(G.vis));
G.dfs(1,-1); //建最短路径树 memset(now,-1,sizeof(now));
root = -1;
minson = INF;
getroot(1,-1,N);
getans(root);
printf("%d %lld\n",ansdep,ansnum);
}
return 0;
}

HDU - 4871 Shortest-path tree (最短路径树+ 树分治)的更多相关文章

  1. hdu 5274 Dylans loves tree(LCA + 线段树)

    Dylans loves tree Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Othe ...

  2. 【CF938G】Shortest Path Queries(线段树分治,并查集,线性基)

    [CF938G]Shortest Path Queries(线段树分治,并查集,线性基) 题面 CF 洛谷 题解 吼题啊. 对于每个边,我们用一个\(map\)维护它出现的时间, 发现询问单点,边的出 ...

  3. hdu 3631 Shortest Path(Floyd)

    题目链接:pid=3631" style="font-size:18px">http://acm.hdu.edu.cn/showproblem.php?pid=36 ...

  4. ZOJ 2760 How Many Shortest Path(最短路径+最大流)

    Description Given a weighted directed graph, we define the shortest path as the path who has the sma ...

  5. HDU 5636 Shortest Path 暴力

    Shortest Path 题目连接: http://acm.hdu.edu.cn/showproblem.php?pid=5636 Description There is a path graph ...

  6. HDU 5636 Shortest Path(Floyed,枚举)

    Shortest Path Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others) Tot ...

  7. HDU - 3631 Shortest Path(Floyd最短路)

    Shortest Path Time Limit: 1000MS Memory Limit: 32768KB 64bit IO Format: %I64d & %I64u SubmitStat ...

  8. HDU - 4725_The Shortest Path in Nya Graph

    The Shortest Path in Nya Graph Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (J ...

  9. hdu 3631 Shortest Path

    floyd算法好像很奇妙的样子.可以做到每次加入一个点再以这个点为中间点去更新最短路,效率是n*n. #include<cstdio> #include<cstring> #i ...

  10. HDU 5636 Shortest Path

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=5636 题解: 1.暴力枚举: #include<cmath> #include<c ...

随机推荐

  1. 详谈JavaScript 匿名函数及闭包

    1.匿名函数函数是JavaScript中最灵活的一种对象,这里只是讲解其匿名函数的用途.匿名函数:就是没有函数名的函数. 1.1 函数的定义,首先简单介绍一下函数的定义,大致可分为三种方式 第一种:这 ...

  2. php 将一个字符串分割为组成它的字符

    问: php里如何将一个字符串分割为组成它的字符? 比如hello  -> [h, e, l, l, o]   以下有三种方法: 这是需要被分割的字符串:  $str = 'Hello小样'; ...

  3. 推荐一个 HTML5在线的流程图工具——ProcessOn

    一直想找个简单好用的UML建模工具,无意在茫茫百度中看见了网友推荐的N多工具,从中找了一个叫 ProcessOn 的工具,可以说非常棒.如果我是WEB开发人员,我肯定去深入研究HTML5啦,太令人着迷 ...

  4. JSP指令与动作元素

    include指令 语法:<%@ include file="URL" %> 其中,URL表示一个要包含的页面. include动作(是一个动作标签) 语法:<j ...

  5. IDEA破解后无法启动

    在网上找了破解IDEA的方法 原文:https://blog.csdn.net/qq_38637558/article/details/78914772 ①到这个地方下载 IntelliJ IDEA ...

  6. 使用隧道技术进行C&C通信

    一.C&C通信 这里的C&C服务器指的是Command & Control Server--命令和控制服务器,说白了就是被控主机的遥控端.一般C&C节点分为两种,C&a ...

  7. python3-requests库的使用

    同步请求库requests用来做测试和简单爬虫其实非常好用的,今天来讲一讲,毕竟不熟悉就用,吃了很大亏啊,文档一定要好好看 http://docs.python-requests.org/zh_CN/ ...

  8. sql语句中left join、right join 以及inner join之间的使用与区别

    sql语句中left join.right join 以及innerjoin之间的使用与区别 left join(左联接) 返回包括左表中的所有记录和右表中联结字段相等的记录  right join( ...

  9. CH5402 选课【树形DP】【背包】

    5402 选课 0x50「动态规划」例题 描述 学校实行学分制.每门的必修课都有固定的学分,同时还必须获得相应的选修课程学分.学校开设了 N(N≤300) 门的选修课程,每个学生可选课程的数量 M 是 ...

  10. PowerDesigner概念模型与物理模型相互转换及导出数据字典

    最近公司项目竣工,验收完成后,把整体平台的所有文档都写清楚,找包发给甲方,由于本人是维护数据库工作,依上面要求,必须编写<数据库设计说明书>里面格式包含三个部分:概念模型.物理模型.数据字 ...