bzoj 1023: [SHOI2008]cactus仙人掌图 2125: 最短路 4728: 挪威的森林 静态仙人掌上路径长度的维护系列
%%% http://immortalco.blog.uoj.ac/blog/1955
一个通用的写法是建树,对每个环建一个新点,去掉环上的边,原先环上每个点到新点连边,边权为点到环根的最短/长路长度
1023 求仙人掌直径
树形dp,维护每个点向下的最长和次长路径长度,对原有的点直接更新答案,对新点可以把对应环上的点取出,倍长,破环成链,并用单调队列正反各扫一次
#include<cstdio>
char buf[],*ptr=buf-;
int _(){
int x=,c=*++ptr;
while(c<)c=*++ptr;
while(c>)x=x*+c-,c=*++ptr;
return x;
}
const int N=;
int es[N],enx[N],ev[N],e0[N],e1[N],ep=;
int dfn[N],low[N],tk=,ss[N],sp=,os[N],op,q[N],ql,qr;
int n,m,D,idp,ans=,f1[N],f2[N];
void ae(int*e,int a,int b,int c){
es[ep]=b;enx[ep]=e[a];ev[ep]=c;e[a]=ep++;
es[ep]=a;enx[ep]=e[b];ev[ep]=c;e[b]=ep++;
}
int min(int a,int b){return a<b?a:b;}
void maxs(int&a,int b){if(a<b)a=b;}
void maxs(int&a,int&b,int c){
if(a<=c)b=a,a=c;
else if(b<c)b=c;
}
void tj(int w){
dfn[w]=low[w]=++tk;
for(int i=e0[w];i;i=enx[i]){
int u=es[i];
if(!u)continue;
es[i^]=;
if(!dfn[u]){
ss[++sp]=u;
tj(u);
if(ss[sp]==u)--sp,ae(e1,w,u,);
}else if((low[w]=min(low[w],dfn[u]))==dfn[u]){
op=;
while(sp&&dfn[ss[sp]]>dfn[u])os[++op]=ss[sp--];
ae(e1,u,++idp,);
for(int j=;j<=op;++j)ae(e1,idp,os[j],min(j,op+-j));
}
}
}
void dfs(int w,int pa){
for(int i=e1[w];i;i=enx[i]){
int u=es[i];
if(u==pa)continue;
dfs(u,w);
maxs(f1[w],f2[w],f1[u]+ev[i]);
}
if(w<=n)maxs(ans,f1[w]+f2[w]);
else{
op=;
for(int i=e1[w];i;i=enx[i]){
int u=es[i];
if(u!=pa)os[++op]=u;
}
os[++op]=pa;
D=op>>;
ql=,qr=;
for(int i=;i<=D;++i)os[op+i]=os[i];
op+=D;
for(int i=;i<=op;++i){
while(ql<=qr&&q[ql]+D<i)++ql;
if(ql<=qr)maxs(ans,f1[os[q[ql]]]+f1[os[i]]-q[ql]+i);
while(ql<=qr&&f1[os[q[qr]]]-q[qr]<=f1[os[i]]-i)--qr;
q[++qr]=i;
}
ql=,qr=;
for(int i=op;i;--i){
while(ql<=qr&&q[ql]-D>i)++ql;
if(ql<=qr)maxs(ans,f1[os[q[ql]]]+f1[os[i]]+q[ql]-i);
while(ql<=qr&&f1[os[q[qr]]]+q[qr]<=f1[os[i]]+i)--qr;
q[++qr]=i;
}
}
}
int main(){
fread(buf,,sizeof(buf),stdin);
idp=n=_();m=_();
for(int i=,c,a,b;i<m;++i){
c=_();
a=_();
for(int j=;j<c;++j){
b=_();
ae(e0,a,b,);
a=b;
}
}
tj();
dfs(,);
printf("%d",ans);
return ;
}
2125 多次询问仙人掌上两点间最短路
任意两点a,b间距离分情况考虑,设c=lca(a,b),若c是原有的点,则距离为树上a,b的距离dis(a,b),否则设x,y分别为a,b到c的路径上与c最近的点,则距离为dis(a,x)+dis(b,y)+环上x,y间的距离
倍增或链剖求一下lca再用前缀和特判一下环上情况
#include<cstdio>
#include<algorithm>
char buf[],*ptr=buf-;
int _(){
int x=,c=*++ptr;
while(c<)c=*++ptr;
while(c>)x=x*+c-,c=*++ptr;
return x;
}
const int N=;
int n,m,q;
int es[N],enx[N],ev[N],e0[],e1[],ep=,ss[],sp=,idp,os[],op,d1[],d2[];
int dfn[],low[],tk=;
void ae(int*e,int a,int b,int c){
es[ep]=b;enx[ep]=e[a];ev[ep]=c;e[a]=ep++;
es[ep]=a;enx[ep]=e[b];ev[ep]=c;e[b]=ep++;
}
int min(int a,int b){return a<b?a:b;}
void f0(int w){
dfn[w]=low[w]=++tk;
for(int i=e0[w];i;i=enx[i]){
int u=es[i];
if(!u)continue;
if(!dfn[u]){
ss[++sp]=i;
es[i^]=;
f0(u);
low[w]=min(low[w],low[u]);
if(ss[sp]==i)--sp,ae(e1,w,u,ev[i]);
}else if((low[w]=min(low[w],dfn[u]))==dfn[u]){
++idp;op=;
ae(e1,idp,u,);
op=;
int s1=ev[i],s2=;
while(sp&&dfn[es[ss[sp]]]>dfn[u]){
int e=ss[sp--];
os[op++]=e;
s2+=ev[e];
}
for(int p=;p<op;++p){
int e=os[p];
ae(e1,idp,es[e],min(d1[es[e]]=s1,s2));
d2[es[e]]=s1+s2;
s1+=ev[e],s2-=ev[e];
}
}
}
}
int fa[][],dep[],Dep[];
void f1(int w,int pa){
fa[][w]=pa;
for(int i=e1[w];i;i=enx[i]){
int u=es[i];
if(u==pa)continue;
dep[u]=dep[w]+;
Dep[u]=Dep[w]+ev[i];
f1(u,w);
}
}
int main(){
fread(buf,,sizeof(buf),stdin);
n=_();m=_();q=_();
idp=n;
for(int i=,a,b,c;i<=m;++i){
a=_();b=_();c=_();
ae(e0,a,b,c);
}
f0();
f1(,);
for(int i=;i<;++i)for(int j=;j<=idp;++j)fa[i][j]=fa[i-][fa[i-][j]];
for(int i=,a,b,ans;i<q;++i){
a=_();b=_();
if(dep[a]<dep[b])std::swap(a,b);
ans=Dep[a]+Dep[b];
for(int d=,s=dep[a]-dep[b];d<;++d)if(s>>d&)a=fa[d][a];
if(a==b)ans-=Dep[a]*;
else{
for(int d=;~d;--d)if(fa[d][a]!=fa[d][b])a=fa[d][a],b=fa[d][b];
if(fa[][a]<=n)ans-=Dep[fa[][a]]*;
else{
ans-=Dep[a]+Dep[b];
int s=d1[a]-d1[b];
if(s<)s=-s;
ans+=min(s,d2[a]-s);
}
}
printf("%d\n",ans);
}
return ;
}
4728 带加点、加环操作维护仙人掌最长简单路径
对每条路径,路径上的点不会比两端更晚加入,所以可以离线处理
对新建的树点分治,分别考虑当前分治中心对应的子树(由于根改变,新边权要重新计算),若分治中心为环,则维护环上的前缀max/后缀max(由于环上两点间有两种路径,要分别维护),否则维护每个相邻点方向的最长路径
bzoj 1023: [SHOI2008]cactus仙人掌图 2125: 最短路 4728: 挪威的森林 静态仙人掌上路径长度的维护系列的更多相关文章
- bzoj 1023: [SHOI2008]cactus仙人掌图 tarjan缩环&&环上单调队列
1023: [SHOI2008]cactus仙人掌图 Time Limit: 1 Sec Memory Limit: 162 MBSubmit: 1141 Solved: 435[Submit][ ...
- 【刷题】BZOJ 1023 [SHOI2008]cactus仙人掌图
Description 如果某个无向连通图的任意一条边至多只出现在一条简单回路(simple cycle)里,我们就称这张图为仙人掌图(cactus).所谓简单回路就是指在图上不重复经过任何一个顶点的 ...
- BZOJ 1023: [SHOI2008]cactus仙人掌图 | 在仙人掌上跑DP
题目: 求仙人掌直径 http://www.lydsy.com/JudgeOnline/problem.php?id=1023 题解: 首先给出仙人掌的定义:满足所有的边至多在一个环上的无向联通图 我 ...
- bzoj 1023 [SHOI2008]cactus仙人掌图 ( poj 3567 Cactus Reloaded )——仙人掌直径模板
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1023 http://poj.org/problem?id=3567 因为lyd在讲课,所以有 ...
- bzoj 1023: [SHOI2008]cactus仙人掌图
这道题是我做的第一道仙人掌DP,小小纪念一下…… 仙人掌DP就是环上的点环状DP,树上的点树上DP.就是说,做一遍DFS,DFS的过程中处理出环,环上的点先不DP,先把这些换上的点的后继点都处理出来, ...
- BZOJ.1023.[SHOI2008]cactus仙人掌图(DP)
题目链接 类似求树的直径,可以用(类似)树形DP求每个点其子树(在仙人掌上就是诱导子图)最长链.次长链,用每个点子节点不同子树的 max{最长链}+max{次长链} 更新答案.(不需要存次长链,求解过 ...
- bzoj 1023: [SHOI2008]cactus仙人掌图【tarjan+dp+单调队列】
本来想先求出点双再一个一个处理结果写了很长发现太麻烦 设f[u]为u点向下的最长链 就是再tarjan的过程中,先照常处理,用最长儿子链和次长儿子链更新按ans,然后处理以这个点为根的环,也就是这个点 ...
- 1023: [SHOI2008]cactus仙人掌图 - BZOJ
Description如果某个无向连通图的任意一条边至多只出现在一条简单回路(simple cycle)里,我们就称这张图为仙人图(cactus).所谓简单回路就是指在图上不重复经过任何一个顶点的回路 ...
- 【BZOJ】1023: [SHOI2008]cactus仙人掌图 静态仙人掌(DFS树)
[题意]给定仙人掌图(每条边至多在一个简单环上),求直径(最长的点对最短路径).n<=50000,m<=10^7. [算法]DFS树处理仙人掌 [题解]参考:仙人掌相关问题的处理方法(未完 ...
随机推荐
- vue 可编辑表格组件
<template> <div class="table"> <table border="1px" v-dragform> ...
- 【LeetCode 100_二叉树_遍历】Same Tree
解法一:递归 bool isSameTree(TreeNode* p, TreeNode* q) { if (p == NULL && q == NULL) return true; ...
- 20165202 2017-2018-2 《Java程序设计》第6周学习总结
教材学习内容总结 Ch8 String类: 程序可以直接使用,String类不能有子类 - 构造String对象 使用String类声明并创建对象 String s = new String(&quo ...
- linux下c语言源码编译
一.源码编译过程 源码 ---> 预处理 ---> 编译 ---> 汇编 ---> 链接 --->执行 我们可以把它分为三部分来完成: ./configure ...
- Mysql双机热备--预备知识
1.双机热备 对于双机热备这一概念,我搜索了很多资料,最后,还是按照大多数资料所讲分成广义与狭义两种意义来说. 从广义上讲,就是对于重要的服务,使用两台服务器,互相备份,共同执行同一服务.当一台服务器 ...
- DataTable和实体类通过反射相互转换
using System.Runtime.Serialization; using System.Data; using System.Reflection; using System.Collect ...
- 基于AMBA总线的SPI协议IP核的设计与验证
https://wenku.baidu.com/view/9542213131126edb6f1a1048.html?mark_pay_doc=2&mark_rec_page=1&ma ...
- vue和react
1. 数据渲染 vue是使用template模板进行渲染,react使用的是jsx语法,对组件进行渲染 vue模板中使用{{ this.data }} 双括号包着变量,代表变量表示的值.外面那层表示需 ...
- EasyPusher/EasyDarwin支持H.265 RTSP/RTP直播推流与分发播放
前言描述 随着大屏时代和高清时代的到来,人们已经不再满足于VGA.CIF这种小分辨率了,取而代之的是720P.1080P.4K级的视频传输,虽然我们国家的基础带宽一直在上升,但普遍情况下,传输高清视频 ...
- Android中的sp和wp指针
经常会在android的framework代码中发现sp<xxx>和wp<xxx>这样的指针,平时看的时候都把他当成一个普通的指针封装过掉了,这几天终于忍不住了,想深入了解一下 ...