http://www.lydsy.com/JudgeOnline/problem.php?id=4016 (题目链接)

题意

  给出一张无向图,求出它的最小路径树,然后求最小路径树上节点数为${K}$的最长路径,并求出这样的路径有多少条。

Solution

  mdzz看错题了,以为求路径条数的时候对节点个数没有要求。。

  抠最小路径树有点恶心,还对字典序有要求,参见了DaD3zZ的方法,枚举边,将符合距离条件的连边,然后dfs,优先字典序小的。

  至于点分治,就是两个数组搞一搞,挺简单的一个统计。

细节

  不要乱用memset,不然复杂度就不对了。

  这种比较长的程序写在namespace或者结构体里面会比较清晰吧。

代码

// bzoj4016
#include<algorithm>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<cstdio>
#include<vector>
#include<cmath>
#include<queue>
#include<map>
#define LL long long
#define inf 1ll<<30
#define Pi acos(-1.0)
#define free(a) freopen(a".in","r",stdin),freopen(a".out","w",stdout);
using namespace std; const int maxn=30010,maxm=60010;
int vis[maxn],deep[maxn],f[maxn],size[maxn],head[maxn];
int n,m,K,cnt,sum,Dargen;
int ans1;LL ans2; struct edge {int from,to,next,w;}e[maxm<<1]; namespace Prepare {
int dis[maxn];
vector<int> v[maxn];
struct data {
int num,w;
friend bool operator < (const data a,const data b) {
return a.w>b.w;
}
}; void link(int u,int v,int w) {
e[++cnt]=(edge){u,v,head[u],w};head[u]=cnt;
e[++cnt]=(edge){v,u,head[v],w};head[v]=cnt;
}
void Dijkstra() {
priority_queue<data> q;
for (int i=1;i<=n;i++) dis[i]=inf;
q.push((data){1,0});dis[1]=0;
while (!q.empty()) {
data x=q.top();q.pop();
if (vis[x.num]) continue;
vis[x.num]=1;
for (int i=head[x.num];i;i=e[i].next)
if (!vis[e[i].to] && dis[e[i].to]>x.w+e[i].w) {
dis[e[i].to]=x.w+e[i].w;
q.push((data){e[i].to,dis[e[i].to]});
}
}
for (int i=1;i<=cnt;i++) {
int uu=e[i].from,vv=e[i].to,ww=e[i].w;
if (dis[uu]+ww==dis[vv]) v[uu].push_back(vv);
}
}
void build(int x) {
vis[x]=1;
sort(v[x].begin(),v[x].end());
int l=v[x].size();
for (int i=0;i<l;i++) if (!vis[v[x][i]]) {
link(x,v[x][i],dis[v[x][i]]-dis[x]);
build(v[x][i]);
}
}
} namespace NodeDivide {
int D[maxn],d[maxn],cntd[maxn],cntD[maxn]; void caldargen(int x,int fa) {
size[x]=1;f[x]=0;
for (int i=head[x];i;i=e[i].next) if (!vis[e[i].to] && e[i].to!=fa) {
caldargen(e[i].to,x);
size[x]+=size[e[i].to];
f[x]=max(f[x],size[e[i].to]);
}
f[x]=max(f[x],sum-size[x]);
if (f[x]<f[Dargen]) Dargen=x;
}
void caldeep(int x,int fa,int l) {
if (d[deep[x]]<l) d[deep[x]]=l,cntd[deep[x]]=1;
else if (d[deep[x]]==l) cntd[deep[x]]++;
for (int i=head[x];i;i=e[i].next) if (e[i].to!=fa && !vis[e[i].to]) {
deep[e[i].to]=deep[x]+1;
caldeep(e[i].to,x,l+e[i].w);
}
}
void work(int x) {
vis[x]=1;
for (int i=head[x];i;i=e[i].next) if (!vis[e[i].to]) {
deep[e[i].to]=1;
caldeep(e[i].to,0,e[i].w);
for (int j=1;d[j] && j<K;j++) {
if (ans1<D[K-1-j]+d[j]) {
ans1=D[K-1-j]+d[j];
ans2=(LL)cntD[K-1-j]*cntd[j];
}
else if (ans1==D[K-1-j]+d[j]) ans2+=(LL)cntD[K-1-j]*cntd[j];
}
for (int j=1;d[j];j++) {
if (D[j]<d[j]) D[j]=d[j],cntD[j]=cntd[j];
else if (D[j]==d[j]) cntD[j]+=cntd[j];
d[j]=cntd[j]=0;
}
}
for (int i=1;D[i];i++) D[i]=cntD[i]=0;
for (int i=head[x];i;i=e[i].next) if (!vis[e[i].to] && size[e[i].to]>=K) {
Dargen=0;sum=size[e[i].to];
caldargen(e[i].to,0);
work(Dargen);
}
}
void Init() {
memset(vis,0,sizeof(vis));
f[0]=inf;sum=n;cntD[0]=1;
Dargen=0;caldargen(1,0);
work(Dargen);
}
} int main() {
scanf("%d%d%d",&n,&m,&K);
for (int u,v,w,i=1;i<=m;i++) {
scanf("%d%d%d",&u,&v,&w);
Prepare::link(u,v,w);
}
Prepare::Dijkstra();
memset(head,0,sizeof(head));
memset(vis,0,sizeof(vis));
cnt=0;
Prepare::build(1);
NodeDivide::Init();
printf("%d %lld",ans1,ans2);
return 0;
}

【bzoj4016】 FJOI2014—最短路径树问题的更多相关文章

  1. [BZOJ4016][FJOI2014]最短路径树问题

    [BZOJ4016][FJOI2014]最短路径树问题 试题描述 给一个包含n个点,m条边的无向连通图.从顶点1出发,往其余所有点分别走一次并返回. 往某一个点走时,选择总长度最短的路径走.若有多条长 ...

  2. [BZOJ4016][FJOI2014]最短路径树问题(dijkstra+点分治)

    4016: [FJOI2014]最短路径树问题 Time Limit: 5 Sec  Memory Limit: 512 MBSubmit: 1796  Solved: 625[Submit][Sta ...

  3. 【BZOJ4016】[FJOI2014]最短路径树问题

    [BZOJ4016][FJOI2014]最短路径树问题 题面 bzoj 洛谷 题解 虽然调了蛮久,但是思路还是蛮简单的2333 把最短路径树构出来,然后点分治就好啦 ps:如果树构萎了,这组数据可以卡 ...

  4. 【BZOJ4016】[FJOI2014]最短路径树问题 最短路径树+点分治

    [BZOJ4016][FJOI2014]最短路径树问题 Description 给一个包含n个点,m条边的无向连通图.从顶点1出发,往其余所有点分别走一次并返回. 往某一个点走时,选择总长度最短的路径 ...

  5. 【BZOJ4016】[FJOI2014]最短路径树问题(点分治,最短路)

    [BZOJ4016][FJOI2014]最短路径树问题(点分治,最短路) 题面 BZOJ 洛谷 题解 首先把最短路径树给构建出来,然后直接点分治就行了. 这个东西似乎也可以长链剖分,然而没有必要. # ...

  6. bzoj 4016 [FJOI2014]最短路径树问题(最短路径树+树分治)

    4016: [FJOI2014]最短路径树问题 Time Limit: 5 Sec  Memory Limit: 512 MBSubmit: 426  Solved: 147[Submit][Stat ...

  7. BZOJ_4016_[FJOI2014]最短路径树问题_最短路+点分治

    BZOJ_4016_[FJOI2014]最短路径树问题_最短路+点分治 Description 给一个包含n个点,m条边的无向连通图.从顶点1出发,往其余所有点分别走一次并返回. 往某一个点走时,选择 ...

  8. 【BZOJ-4016】最短路径树问题 Dijkstra + 点分治

    4016: [FJOI2014]最短路径树问题 Time Limit: 5 Sec  Memory Limit: 512 MBSubmit: 1092  Solved: 383[Submit][Sta ...

  9. [FJOI2014]最短路径树问题 长链剖分

    [FJOI2014]最短路径树问题 LG传送门 B站传送门 长链剖分练手好题. 如果你还不会长链剖分的基本操作,可以看看我的总结. 这题本来出的很没水平,就是dijkstra(反正我是不用SPFA)的 ...

  10. 洛谷 [FJOI2014]最短路径树问题 解题报告

    [FJOI2014]最短路径树问题 题目描述 给一个包含\(n\)个点,\(m\)条边的无向连通图.从顶点\(1\)出发,往其余所有点分别走一次并返回. 往某一个点走时,选择总长度最短的路径走.若有多 ...

随机推荐

  1. appium 元素定位工具

    两种元素定位工具: 1.uiautomatorviewer是android-sdk自带的一个元素定位工具,目录D:\androidsdk\androidsdk\tools\bin . 双击启动uiau ...

  2. flask中的if __name__ == "__main__"

    在编写python文件时,一般会在入口文件中加入if __name__ == "__main__", 这样当这个脚本文件执行时就会执行这个语句下面的内容,而如果这个脚本文件被当作模 ...

  3. CHAPTER 25 The Greatest Show on Earth 第25章 地球上最壮观的演出

    CHAPTER 25 The Greatest Show on Earth 第25章 地球上最壮观的演出 Go for a walk in the countryside and you will f ...

  4. 【CentOS 7】scp示例

    1,从远端拷贝到本地 /tmp路径 root@raspberrypi:/download/api_weather# scp root@123.207.xxx.xxx:/xxx/* /tmp 2,从本地 ...

  5. PHP has encountered an Access Violation at 01F4A622解决方法

    php搭建的网站出现以下问题的解决方法分享: Z-blog,DedeCMS,Dsicuz!,PhpWind,PhpCMS,帝国CMS等都有可能出现php访问冲突问题. 今天访问网站发现出现了一个错误& ...

  6. 安装macOS Sierra后怎么找回“任何来源”选项

    安装macOS Sierra后,会发现系统偏好设置的“安全与隐私”中默认已经去除了允许“任何来源”App的选项,无法运行一些第三方应用(提示xx.app已经损坏).如果需要恢复允许“任何来源”的选项, ...

  7. DataRow对象的RowState和DataRowVersion属性特点

    DataRow对象有两个比较重要的属性,分别是行状态(RowState)和行版本(DataRowVersion),通过这两个属性能够有效的管理表中的行.下面简要的介绍一下行状态和行版本的特点和关系. ...

  8. psp报告

    1.读材料回答问题 (1)回想一下你曾经对计算机专业的畅想.当初你是如何做出选择计算机专业的决定的?你认为过去接触到的课程是否符合你对计算机专业的期待,为什么?你觉得计算机是你喜欢的领域吗,它是你擅长 ...

  9. 20180711-统计PDB中的蛋白质种类、膜蛋白文件个数及信息等

    20180710完成这份工作.简单,但是完成了还是很开心.在我尝试如何使用pickle保存数据后,尝试保存PDB文件中“HEADER”中的信息.文件均保存于实验室服务器(97.73.198.168)/ ...

  10. Daily Scrum (2015/10/28)

    昨天DEV们完成了一部分代码风格的修整.今晚在与其他组进行交流时我们发现我们的代码是需要在服务器上运行的,而且服务器是需要配置的,而且据说需要花一些时间.所以在编写代码之前PM提出我们应该先把服务器搭 ...