【BZOJ-4016】最短路径树问题 Dijkstra + 点分治
4016: [FJOI2014]最短路径树问题
Time Limit: 5 Sec Memory Limit: 512 MB
Submit: 1092 Solved: 383
[Submit][Status][Discuss]
Description
Input
Output
Sample Input
1 2 1
2 3 1
3 4 1
2 5 1
3 6 1
5 6 1
Sample Output
HINT
Source
Solution
这道题还是比较好搞的。
首先按照题目的意思搞出 最短路径树 来,这里利用vector排序搞了一下字典序的问题。
然后就可以用 点分治 处理答案了。
令$dp[dep][0/1]$表示经过点数为$dep$的最长路径的长度和方案数,然后DFS一遍就可以了。
Code
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<vector>
#include<queue>
using namespace std;
inline int read()
{
int x=0,f=1; char ch=getchar();
while (ch<'0' || ch>'9') {if (ch=='-') f=-1; ch=getchar();}
while (ch>='0' && ch<='9') {x=x*10+ch-'0'; ch=getchar();}
return x*f;
}
#define MAXN 100010
#define LL long long int N,M,K;
LL Dist,Num; vector<int> G[MAXN]; namespace Graph{ struct EdgeNode{
int next,to,dis,from;
}edge[MAXN<<1];
int head[MAXN],cnt; inline void AddEdge(int u,int v,int w) {cnt++; edge[cnt].next=head[u]; head[u]=cnt; edge[cnt].to=v; edge[cnt].from=u; edge[cnt].dis=w;}
inline void InsertEdge(int u,int v,int w) {AddEdge(u,v,w); AddEdge(v,u,w);} #define Pa pair<int,int>
#define MP make_pair
#define INF 0x7fffffff
priority_queue<Pa,vector<Pa>,greater<Pa> >q;
int dist[MAXN];
inline void Dijkstra(int S=1)
{
for (int i=1; i<=N; i++) dist[i]=INF;
q.push(MP(0,S)); dist[S]=0;
while (!q.empty()) {
int dis=q.top().first;
int now=q.top().second;
q.pop();
if (dis>dist[now]) continue;
for (int i=head[now]; i;i=edge[i].next) {
if (dist[edge[i].to]>dis+edge[i].dis) {
dist[edge[i].to]=dis+edge[i].dis;
q.push(MP(dist[edge[i].to],edge[i].to));
}
}
} for (int i=1; i<=cnt; i++) {
int u=edge[i].from,v=edge[i].to,d=edge[i].dis;
if (dist[u]+d==dist[v]) G[u].push_back(v);
} // for (int i=1; i<=N; printf("Now=%d\n",i),i++)
// for (int j=0; j<G[i].size(); j++)
// printf("%d ",G[i][j]);
}
} namespace TreeDivide{ struct EdgeNode{
int next,to,dis;
}edge[MAXN<<1];
int head[MAXN],cnt; inline void AddEdge(int u,int v,int w) {cnt++; edge[cnt].next=head[u]; head[u]=cnt; edge[cnt].to=v; edge[cnt].dis=w;}
inline void InsertEdge(int u,int v,int w) {/*printf("<%d %d>\n",u,v,w);*/ AddEdge(u,v,w); AddEdge(v,u,w);} int size[MAXN],mx[MAXN],root,Sz;
bool visit[MAXN];
inline void Getroot(int now,int last)
{
size[now]=1; mx[now]=0;
for (int i=head[now]; i; i=edge[i].next)
if (edge[i].to!=last && !visit[edge[i].to]) {
Getroot(edge[i].to,now);
size[now]+=size[edge[i].to];
mx[now]=max(mx[now],size[edge[i].to]);
}
mx[now]=max(mx[now],Sz-size[now]);
if (mx[now]<mx[root]) root=now;
} int f[MAXN][2],g[MAXN][2];
inline void DFS(int now,int last,int dep,int dis)
{
if (dep>K) return;
if (dis>f[dep][0])
f[dep][0]=dis,f[dep][1]=1;
else
if (dis==f[dep][0])
f[dep][1]++; for (int i=head[now]; i; i=edge[i].next)
if (edge[i].to!=last && !visit[edge[i].to]) {
DFS(edge[i].to,now,dep+1,dis+edge[i].dis);
}
} inline void Divide(int now)
{
visit[now]=1;
for (int i=1; i<=K; i++) g[i][0]=g[i][1]=0;
g[0][0]=0; g[0][1]=1; K--;
for (int i=head[now]; i; i=edge[i].next)
if (!visit[edge[i].to]) { for (int j=1; j<=K; j++) f[j][0]=f[j][1]=0; DFS(edge[i].to,now,1,edge[i].dis); for (int j=1; j<=K; j++) {
if (Dist<g[K-j][0]+f[j][0])
Dist=g[K-j][0]+f[j][0],Num=(LL)g[K-j][1]*f[j][1];
else
if (Dist==g[K-j][0]+f[j][0])
Num+=(LL)g[K-j][1]*f[j][1];
} for (int j=1; j<=K; j++) {
if (g[j][0]<f[j][0])
g[j][0]=f[j][0],g[j][1]=f[j][1];
else
if (g[j][0]==f[j][0])
g[j][1]+=f[j][1];
}
}
K++; for (int i=head[now]; i; i=edge[i].next)
if (!visit[edge[i].to]) {
root=0;
Sz=size[edge[i].to];
Getroot(edge[i].to,now);
Divide(root);
}
} bool mark[MAXN];
inline void BuildTree(int now)
{
mark[now]=1;
sort(G[now].begin(),G[now].end());
for (int i=0; i<G[now].size(); i++)
if (!mark[G[now][i]]) {
BuildTree(G[now][i]);
InsertEdge(now,G[now][i],Graph::dist[G[now][i]]-Graph::dist[now]);
}
}
}using namespace TreeDivide; int main()
{
N=read(),M=read(),K=read();
for (int i=1,x,y,z; i<=M; i++) {
x=read(),y=read(),z=read();
Graph::InsertEdge(x,y,z);
}
Graph::Dijkstra();
TreeDivide::BuildTree(1); Sz=mx[root=0]=N;
TreeDivide::Getroot(1,0);
TreeDivide::Divide(root); printf("%lld %lld\n",Dist,Num);
return 0;
}
【BZOJ-4016】最短路径树问题 Dijkstra + 点分治的更多相关文章
- BZOJ 4016 最短路径树问题 最短路径树构造+点分治
题目: BZOJ4016最短路径树问题 分析: 大家都说这是一道强行拼出来的题,属于是两种算法的模板题. 我们用dijkstra算法算出1为源点的最短路数组,然后遍历一下建出最短路树. 之后就是裸的点 ...
- [BZOJ4016][FJOI2014]最短路径树问题(dijkstra+点分治)
4016: [FJOI2014]最短路径树问题 Time Limit: 5 Sec Memory Limit: 512 MBSubmit: 1796 Solved: 625[Submit][Sta ...
- luogu 2993 [FJOI2014]最短路径树问题 Dijkstra+点分治
挺简单的,但是给人一种把两个问题强行弄到一起的感觉. 十分不好写. Code: #include <queue> #include <cstdio> #include < ...
- 【BZOJ4016】[FJOI2014]最短路径树问题(点分治,最短路)
[BZOJ4016][FJOI2014]最短路径树问题(点分治,最短路) 题面 BZOJ 洛谷 题解 首先把最短路径树给构建出来,然后直接点分治就行了. 这个东西似乎也可以长链剖分,然而没有必要. # ...
- BZOJ 4016 [FJOI2014]最短路径树问题 (贪心+点分治)
题目大意:略 传送门 硬是把两个题拼到了一起= = $dijkstra$搜出单源最短路,然后$dfs$建树,如果$dis_{v}=dis_{u}+e.val$,说明这条边在最短路图内,然后像$NOIP ...
- HYSBZ - 4016 最短路径树问题 点分治 + 最短路径最小字典序
题目传送门 题解:首先对于给定的图,需要找到那些从1好点出发然后到x号点的最短路, 如果有多条最短路就要找到字典序最小的路,这样扣完这些边之后就会有一棵树.然后再就是很普通的点分治了. 对于扣边这个问 ...
- 【bzoj4016】[FJOI2014]最短路径树问题 堆优化Dijkstra+DFS树+树的点分治
题目描述 给一个包含n个点,m条边的无向连通图.从顶点1出发,往其余所有点分别走一次并返回. 往某一个点走时,选择总长度最短的路径走.若有多条长度最短的路径,则选择经过的顶点序列字典序最小的那条路径( ...
- bzoj 4016 [FJOI2014]最短路径树问题(最短路径树+树分治)
4016: [FJOI2014]最短路径树问题 Time Limit: 5 Sec Memory Limit: 512 MBSubmit: 426 Solved: 147[Submit][Stat ...
- bzoj 4016: [FJOI2014]最短路径树问题
bzoj4016 最短路路径问题 Time Limit: 5 Sec Memory Limit: 512 MB Description 给一个包含n个点,m条边的无向连通图.从顶点1出发,往其余所有点 ...
随机推荐
- Spark记录-Scala记录(基础程序例子)
import scala.util.control._ object learnning { def main(args:Array[String]):Unit={ val n:Int=10 prin ...
- Oozie 生成JMS消息并向 JMS Provider发送消息过程分析
一,涉及到的工程 从官网下载源码,mvn 编译成 Eclipse工程文件:
- mysql统计一个字段的多种状态
假如我有下面的表:ID Item status updatetime author1 a 1 2014-01-0 ...
- 转:我是否该放弃VB.Net?
我是否该放弃VB.Net呢?这个问题一次次的出现在我的脑海里,而且这种想法越来越强烈.放弃VB.Net至少能让我的生活变得轻松些.如果你是个C#程序员,那拷贝粘贴代码会很容易,因为可以找到的例子代码如 ...
- Docker01 CentOS配置Docker
Docker 是一个开源的应用容器引擎,让开发者可以打包他们的应用以及依赖包到一个可移植的容器中,然后发布到任何流行的 Linux 机器上,也可以实现虚拟化.容器是完全使用沙箱机制,相互之间不会有任何 ...
- mysql.user细节三问
一.如何拒绝用户从某个精确ip访问数据库假如在mysql.user表中存在用户'mydba'@'192.168.85.%',现在想拒绝此用户从某个精确ip访问数据库 # 创建精确ip用户,分配不同的密 ...
- Coins in a Line I & II
Coins in a Line I There are n coins in a line. Two players take turns to take one or two coins from ...
- 【Unity_UWP】Unity 工程发布win10 UWP 时的本地文件读取 (上篇)
Universal Windows Platform(UWP)是微软Windows10专用的通用应用平台,其目的在于在统一操作系统下控制所有智能电子设备. 自从Unity 5.2之后,配合VS 201 ...
- 【oracle】入门学习(一)
一直想学oracle但都没有下定决心.这次借了书,一定要学好oracle. 目前学习 <Oracle从入门到精通> 明日科技 的Oracle 11g 版本 关系型数据库的基本理论 数据模型 ...
- Java登陆拦截器
package com.beidou.warehouseerp.interceptor; import com.alibaba.fastjson.JSON; import com.beidou.war ...