【BZOJ4912】天才黑客(最短路,虚树)
【BZOJ4912】天才黑客(最短路,虚树)
题面
题解
\(Anson\)爷讲过的题目,然而我还是不会做
只有照着\(zsy\)的程序打我才会做。。。。果然太弱了。
这道题目显然是把边看成点,然后把原图中的每一个点的入边和出边之间相互连边,
边权是\(lcp\)的长度,也就是在\(Trie\)树上对应的点的\(LCA\)
那么,考虑如何优化,对于一个点,把它的入边和出现对应的按照\(dfs\)序排序
利用虚树的思想,此时只需要相邻的点求\(LCA\)
那么,对于一个\(LCA\),显然可以把建立一个虚点,把它的所有儿子全部连上来,
因为儿子可能在之前就连在某个虚点上了,这样子只需要把虚点连上来就好了。
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<set>
#include<map>
#include<vector>
#include<queue>
using namespace std;
#define ll long long
#define RG register
#define MAX 111111
#define MAXL 1111111
#define pi pair<int,int>
inline int read()
{
RG int x=0,t=1;RG char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=-1,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return x*t;
}
int n,m,k,V[MAXL],h1[MAX],h2[MAX],Cnt,pos[MAXL];
struct Line{int v,next;}e[MAX];
inline void Add(int *h,int u,int v){e[++Cnt]=(Line){v,h[u]};h[u]=Cnt;}
struct Trie
{
struct Line{int v,next;}e[MAX];
int h[MAX],cnt;
inline void Add(int u,int v){e[++cnt]=(Line){v,h[u]};h[u]=cnt;}
int dfn[MAX],st[18][MAX],lg[MAX],tim,dep[MAX];
void init(){memset(h,0,sizeof(h));tim=cnt=0;memset(st,0,sizeof(st));}
void dfs(int u)
{
st[0][dfn[u]=++tim]=dep[u];
for(int i=h[u];i;i=e[i].next)
dep[e[i].v]=dep[u]+1,dfs(e[i].v),st[0][++tim]=dep[u];
}
void pre()
{
dfs(1);
for(int i=2;i<=tim;++i)lg[i]=lg[i>>1]+1;
for(int j=1;j<=lg[tim];++j)
for(int i=1;i+(1<<j)-1<=tim;++i)
st[j][i]=min(st[j-1][i],st[j-1][i+(1<<(j-1))]);
}
int LCA(int u,int v)
{
u=dfn[u];v=dfn[v];if(u>v)swap(u,v);
int k=lg[v-u+1];
return min(st[k][u],st[k][v-(1<<k)+1]);
}
}T;
struct Graph
{
struct Line{int v,next,w;}E[MAXL];
int h[MAXL],cnt,tot;bool vis[MAXL];
inline void Add(int u,int v,int w){E[++cnt]=(Line){v,h[u],w};h[u]=cnt;}
void init(){memset(h,0,sizeof(h));memset(vis,0,sizeof(vis));cnt=0;}
int dis[MAXL];
void Dijkstra()
{
priority_queue<pi,vector<pi>,greater<pi> >Q;
memset(dis,127,sizeof(dis));
for(int i=h1[1];i;i=e[i].next)
Q.push(make_pair(dis[e[i].v]=V[e[i].v],e[i].v));
while(!Q.empty())
{
int u=Q.top().second;Q.pop();
if(vis[u])continue;vis[u]=true;
for(int i=h[u];i;i=E[i].next)
{
int v=E[i].v;
if(dis[v]>dis[u]+E[i].w+V[v])
dis[v]=dis[u]+E[i].w+V[v],Q.push(make_pair(dis[v],v));
}
}
}
}G;
int S[MAX],top,p1[MAX],p2[MAX],q1[MAX],q2[MAX];
bool cmp(int a,int b){return T.dfn[pos[abs(a)]]<T.dfn[pos[abs(b)]];}
void Link(int u)
{
top=0;
for(int i=h2[u];i;i=e[i].next)S[++top]=e[i].v;
for(int i=h1[u];i;i=e[i].next)S[++top]=-e[i].v;
sort(&S[1],&S[top+1],cmp);
for(int i=1;i<=top;++i)
{
p1[i]=++G.tot;p2[i]=++G.tot;q1[i]=++G.tot;q2[i]=++G.tot;
if(i>1)
{
G.Add(p1[i-1],p1[i],0);G.Add(q1[i-1],q1[i],0);
G.Add(p2[i],p2[i-1],0);G.Add(q2[i],q2[i-1],0);
}
if(S[i]>0)G.Add(S[i],p1[i],0),G.Add(S[i],p2[i],0);
else S[i]=-S[i],G.Add(q1[i],S[i],0),G.Add(q2[i],S[i],0);
}
for(int i=1;i<top;++i)
{
int u=T.LCA(pos[S[i]],pos[S[i+1]]);
G.Add(p1[i],q1[i+1],u);G.Add(p2[i+1],q2[i],u);
}
}
int main()
{
int Cases=read();
while(Cases--)
{
n=read();m=G.tot=read();k=read();Cnt=0;
G.init();T.init();memset(V,0,sizeof(V));
memset(h1,0,sizeof(h1));memset(h2,0,sizeof(h2));
for(int i=1;i<=m;++i)
{
int u=read(),v=read();V[i]=read();pos[i]=read();
Add(h1,u,i);Add(h2,v,i);
}
for(int i=1;i<k;++i){int u=read(),v=read(),w=read();T.Add(u,v);}
T.pre();
for(int i=2;i<=n;++i)Link(i);
G.Dijkstra();
for(int u=2;u<=n;++u)
{
int ans=2e9;
for(int i=h2[u];i;i=e[i].next)
ans=min(ans,G.dis[e[i].v]);
printf("%d\n",ans);
}
}
return 0;
}
【BZOJ4912】天才黑客(最短路,虚树)的更多相关文章
- BZOJ4912 SDOI2017天才黑客(最短路+虚树)
容易想到把边当成点重建图跑最短路.将每条边拆成入边和出边,作为新图中的两个点,由出边向入边连边权为原费用的边.对于原图中的每个点,考虑由其入边向出边连边.直接暴力两两连边当然会被卡掉,注意到其边权是t ...
- 洛谷3783 SDOI2017 天才黑客(最短路+虚树+边转点+线段树优化建图)
成功又一次自闭了 怕不是猪国杀之后最自闭的一次 一看到最短路径. 我们就能推测这应该是个最短路题 现在考虑怎么建图 根据题目的意思,我们可以发现,在本题中,边与边之间存在一些转换关系,但是点与点之间并 ...
- [SDOI2017]天才黑客[最短路、前缀优化建图]
题意 一个 \(n\) 点 \(m\) 边的有向图,还有一棵 \(k\) 个节点的 trie ,每条边上有一个字符串,可以用 trie 的根到某个节点的路径来表示.每经过一条边,当前携带的字符串就会变 ...
- [LOJ#2270][BZOJ4912][SDOI2017]天才黑客
[LOJ#2270][BZOJ4912][SDOI2017]天才黑客 试题描述 SD0062 号选手小 Q 同学为了偷到 SDOI7012 的试题,利用高超的黑客技术潜入了 SDOI 出题组的内联网的 ...
- 【SDOI2017】天才黑客(前后缀优化建图 & 最短路)
Description 给定一张有向图,\(n\) 个点,\(m\) 条边.第 \(i\) 条边上有一个边权 \(c_i\),以及一个字符串 \(s_i\). 其中字符串 \(s_1, s_2, \c ...
- 良心送分题(牛客挑战赛35E+虚树+最短路)
目录 题目链接 题意 思路 代码 题目链接 传送门 题意 给你一棵树,然后把这棵树复制\(k\)次,然后再添加\(m\)条边,然后给你起点和终点,问你起点到终点的最短路. 思路 由于将树复制\(k\) ...
- 2020牛客NOIP赛前集训营-提高组(第三场)C-牛半仙的妹子Tree【虚树,最短路】
正题 题目链接:https://ac.nowcoder.com/acm/contest/7609/C 题目大意 给出\(n\)个点的一棵树,\(m\)个时刻各有一个操作 标记一个点,每个点被标记后的每 ...
- [SDOI2017]天才黑客
题目大意 给一张有向图,再给一颗字典树,有向图上的每条边有一个非负边权还有一个字典树上的字符串,从一条边到另一条边的代价是那条边的边权和这两个字符串的最长公共前缀,问从1到其他点的最短路. 题解 一看 ...
- 【SDOI2017】天才黑客
[SDOI2017]天才黑客 这题太神了. 先模Claris 大神的题解. 首先我们要将边转换为点.如果暴力连边就会有\(m^2\)的边,于是我们考虑优化建图. 难点在于快速得到两个边的串的\(lcp ...
随机推荐
- HBase第二章 基本API
1.pom.xml <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www ...
- eclipse jetty debug
一. 1, Eeclipse中选择 Run --> External Tools --> External Tools Configurations 然后new一个Program项. ...
- Maven学习(二)-----Maven启用代理访问
Maven启用代理访问 如果你的公司正在建立一个防火墙,并使用HTTP代理服务器来阻止用户直接连接到互联网.如果您使用代理,Maven将无法下载任何依赖. 为了使它工作,你必须声明在 Maven 的配 ...
- 用Python实现多站点运维监控
在小型公司里如果产品线单一的话,比如就一个app, 一般1~2个运维就够用了.如果产品过于庞大,就需要多个运维人员. 但对于多产品线的公司来说,运维人员就要必须分多个人负责,因为超过200个站点让1个 ...
- 《Node.js核心技术教程》学习笔记
<Node.js核心技术教程>TOC \o "1-3" \h \z \u 1.章模块化编程 2019.2.19 13:30' PAGEREF _101 \h 1 08D ...
- 城市规模越大,工资、GDP、犯罪率越高:4.5星|《规模》
规模 信息浓度非常高的一本书.篇幅也不小,纸书有568页,致谢与注释只占7%. 全书讲各种复杂的东西中存在的普遍规律:哺乳动物体重每增加一倍,心率降低25%:城市人口每增加一倍,加油站只增加85%:城 ...
- EasyUI 效果还不错的数据处理等待效果
$("#form").form("submit",{ url:url, onSubmit: function(){ parent.$.messager.prog ...
- Live Archive 训练题
7091 Height Ordering Mrs. Chambers always has her class line up in height order (shortest at the fro ...
- linux awk,sort,uniq,wc,cut命令详解
1.awk awk是行处理器: 相比较屏幕处理的优点,在处理庞大文件时不会出现内存溢出或是处理缓慢的问题,通常用来格式化文本信息 $ 表示当前行 $ 表示第一列 NF 表示一共有多少列 $NF 表示最 ...
- Learn Docker(一)—软件安装与常规操作
一.安装Docker Windows平台 在Windows10 X64专业版上可以直接下载Docker原生应用进行安装,在控制面板的程序与功能里启用Hyper-v,之后就可以运行docker程序啦. ...