分析

可以发现第一列和最后一列永远不会被删除,于是我们可以想到维护前后缀最小生成树,但是直接维护的话显然时间空间两爆炸。(通过上网找题解)可以发现我们关心的只是最左边和最右边两列,而不关心内部的连边情况。所以我们可以仅维护这两列的节点在最小生成树上形成的虚树,边权是对应链上最大的边权,合并时对两棵虚树上的所有边再跑一遍最小生成树就好了。

由于虚树的大小是\(O(n)\)级别的,所以该算法的时间复杂度为\(O(n(m+q) \log n)\)。

(这代码写起来有点恶心。)

代码

#include <bits/stdc++.h>

#define rin(i,a,b) for(int i=(a);i<=(b);++i)
#define irin(i,a,b) for(int i=(a);i>=(b);--i)
#define trav(i,a) for(int i=head[a];i;i=e[i].nxt)
#define Size(a) (int)a.size()
#define pb push_back
#define mkpr std::make_pair
#define fi first
#define se second
#define lowbit(a) ((a)&(-(a)))
typedef long long LL; using std::cerr;
using std::endl; inline int read(){
int x=0,f=1;char ch=getchar();
while(!isdigit(ch)){if(ch=='-')f=-1;ch=getchar();}
while(isdigit(ch)){x=x*10+ch-'0';ch=getchar();}
return x*f;
} const int MAXN=105;
const int MAXM=10005; int n,m,q;
unsigned int SA, SB, SC;int lim; int getweight() {
SA ^= SA << 16;
SA ^= SA >> 5;
SA ^= SA << 1;
unsigned int t = SA;
SA = SB;
SB = SC;
SC ^= t ^ SA;
return SC % lim + 1;
} struct mst_edge{
int u,v,w;
inline friend bool operator < (mst_edge x,mst_edge y){
return x.w<y.w;
}
}; struct mst{
LL sum;
std::vector<int> poi;
std::vector<mst_edge> edg;
inline void init(){sum=0;poi.clear();edg.clear();}
}; mst pre[MAXM],suf[MAXM];
std::vector<mst_edge> hng[MAXM],shu[MAXM],ext; inline int calc_id(int x,int y){
return (x-1)*m+y;
} int dsu[MAXN*MAXM];
int ecnt,head[MAXN*MAXM];
bool mark[MAXN*MAXM],havem[MAXN*MAXM];
std::vector<int> po,retp;
std::vector<mst_edge> ee,rete; struct Edge{
int to,nxt,w;
}e[MAXN*MAXM*4]; int getf(int x){
return dsu[x]==x?x:dsu[x]=getf(dsu[x]);
} inline bool merge_dsu(int x,int y){
x=getf(x),y=getf(y);
if(x!=y){
dsu[y]=x;
return true;
}
return false;
} inline void add_edge(int bg,int ed,int val){
++ecnt;
e[ecnt].to=ed;
e[ecnt].nxt=head[bg];
e[ecnt].w=val;
head[bg]=ecnt;
} void dfs1(int x,int pre){
havem[x]|=mark[x];
trav(i,x){
int ver=e[i].to;
if(ver==pre)continue;
dfs1(ver,x);
havem[x]|=havem[ver];
}
} void dfs2(int x,int pre,int las,int maxw){
int flag=0;
trav(i,x){
int ver=e[i].to;
if(ver==pre)continue;
if(havem[ver])++flag;
}
if(flag>1)mark[x]=true;
if(mark[x]){
retp.pb(x);
if(las)rete.pb((mst_edge){las,x,maxw});
}
trav(i,x){
int ver=e[i].to;
if(ver==pre)continue;
if(mark[x])dfs2(ver,x,x,e[i].w);
else dfs2(ver,x,las,std::max(maxw,e[i].w));
}
} mst merge(mst L,mst R,int bg){
mst ret;ret.init();
ret.sum=L.sum+R.sum;
po.clear();ee.clear();
retp.clear();rete.clear();ecnt=0;
rin(i,0,Size(L.poi)-1)po.pb(L.poi[i]);
rin(i,0,Size(R.poi)-1)po.pb(R.poi[i]);
rin(i,0,Size(L.edg)-1){
ret.sum-=L.edg[i].w;
ee.pb(L.edg[i]);
}
rin(i,0,Size(R.edg)-1){
ret.sum-=R.edg[i].w;
ee.pb(R.edg[i]);
}
rin(i,0,Size(ext)-1)ee.pb(ext[i]);
rin(i,0,Size(po)-1){
dsu[po[i]]=po[i];
head[po[i]]=0;
mark[po[i]]=havem[po[i]]=false;
}
std::sort(ee.begin(),ee.end());
int temp=0;
rin(i,0,Size(ee)-1){
if(merge_dsu(ee[i].u,ee[i].v)){
++temp;
ret.sum+=ee[i].w;
add_edge(ee[i].u,ee[i].v,ee[i].w);
add_edge(ee[i].v,ee[i].u,ee[i].w);
if(temp==Size(po)-1)break;
}
}
rin(i,1,n)mark[calc_id(i,bg)]=true;
if(bg==1)rin(i,0,Size(R.poi)-1)mark[R.poi[i]]=true;
else rin(i,0,Size(L.poi)-1)mark[L.poi[i]]=true;
dfs1(po[0],0);dfs2(po[0],0,0,0);
ret.poi=retp,ret.edg=rete;
return ret;
} int main(){
n=read(),m=read(),SA=read(),SB=read(),SC=read(),lim=read();
rin(i,1,n)rin(j,1,m){
int w=getweight();
if(j<m)hng[j].pb((mst_edge){calc_id(i,j),calc_id(i,j+1),w});
else hng[j].pb((mst_edge){calc_id(i,j),calc_id(i,1),w});
}
rin(i,1,n-1)rin(j,1,m){
int w=getweight();
shu[j].pb((mst_edge){calc_id(i,j),calc_id(i+1,j),w});
}
rin(i,1,n)pre[1].poi.pb(calc_id(i,1));
rin(i,0,Size(shu[1])-1){
pre[1].sum+=shu[1][i].w;
pre[1].edg.pb(shu[1][i]);
}
rin(i,2,m){
ext.clear();
rin(j,1,n)pre[i].poi.pb(calc_id(j,i));
rin(j,0,Size(shu[i])-1){
pre[i].sum+=shu[i][j].w;
pre[i].edg.pb(shu[i][j]);
}
rin(j,0,Size(hng[i-1])-1)ext.pb(hng[i-1][j]);
pre[i]=merge(pre[i-1],pre[i],1);
}
rin(i,1,n)suf[m].poi.pb(calc_id(i,m));
rin(i,0,Size(shu[m])-1){
suf[m].sum+=shu[m][i].w;
suf[m].edg.pb(shu[m][i]);
}
irin(i,m-1,1){
ext.clear();
rin(j,1,n)suf[i].poi.pb(calc_id(j,i));
rin(j,0,Size(shu[i])-1){
suf[i].sum+=shu[i][j].w;
suf[i].edg.pb(shu[i][j]);
}
rin(j,0,Size(hng[i])-1)ext.pb(hng[i][j]);
suf[i]=merge(suf[i],suf[i+1],m);
}
ext.clear();
rin(i,0,Size(hng[m])-1)ext.pb(hng[m][i]);
q=read();
while(q--){
int l=read(),r=read();
printf("%lld\n",merge(pre[l-1],suf[r+1],1).sum);
}
return 0;
}

[VIJOS2053][SDOI2019]世界地图:最小生成树+虚树的更多相关文章

  1. CF891C Envy 最小生成树/虚树

    正解:最小生成树/虚树 解题报告: 传送门! sd如我就只想到了最暴力的想法,一点儿优化都麻油想到,,,真的菜到爆炸了QAQ 然后就分别港下两个正解QAQ 法一,最小生成树 这个主要是要想到关于最小生 ...

  2. [SDOI2019]世界地图(kruskal重构树+虚树)

    通过子任务1.3十分显然,子任务4实际上就是线段树,和我下午写的[SDOI2015]道路修建一模一样,堪称“我抄我自己”,不会的可以先做一下这个题. 然后考虑正解,参考了zhoushuyu的博客,首先 ...

  3. 【题解】Luogu P5360 [SDOI2019]世界地图

    原题传送门 每次查询的实际就是将地图的一个前缀和一个后缀合并后的图的最小生成树边权和 我们要预处理每个前缀和后缀的最小生成树 实际求前缀和(后缀和)的过程珂以理解为上一个前缀和这一列的最小生成树进行合 ...

  4. 洛谷 P6199 - [EER1]河童重工(点分治+虚树)

    洛谷题面传送门 神仙题. 首先看到这样两棵树的题目,我们肯定会往动态树分治的方向考虑.考虑每次找出 \(T_2\) 的重心进行点分治.然后考虑跨过分治中心的点对之间的连边情况.由于连边边权与两棵树都有 ...

  5. BZOJ 2286 消耗战 (虚树+树形DP)

    给出一个n节点的无向树,每条边都有一个边权,给出m个询问,每个询问询问ki个点,问切掉一些边后使得这些顶点无法与顶点1连接.最少的边权和是多少.(n<=250000,sigma(ki)<= ...

  6. 【BZOJ-3572】世界树 虚树 + 树形DP

    3572: [Hnoi2014]世界树 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 1084  Solved: 611[Submit][Status ...

  7. 【BZOJ-2286】消耗战 虚树 + 树形DP

    2286: [Sdoi2011消耗战 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 2120  Solved: 752[Submit][Status] ...

  8. BZOJ 2286 树链剖分+DFS序+虚树+树形DP

    第一次学习虚树,就是把无关的点去掉.S里维护一条链即可. #include <iostream> #include <cstring> #include <cstdio& ...

  9. 青云的机房组网方案(简单+普通+困难)(虚树+树形DP+容斥)

    题目链接 1.对于简单的版本n<=500, ai<=50 直接暴力枚举两个点x,y,dfs求x与y的距离. 2.对于普通难度n<=10000,ai<=500 普通难度解法挺多 ...

随机推荐

  1. 数据库索引 B+树

    问题1.数据库为什么要设计索引?索引类似书本目录,用于提升数据库查找速度.问题2.哈希(hash)比树(tree)更快,索引结构为什么要设计成树型?加快查找速度的数据结构,常见的有两类:(1)哈希,例 ...

  2. Robot Framework(三)项目实践出现的问题以及解决方法

    导航: 1.元素定位失败 2.系统自带的确认弹窗 3.ElementNotVisibleException: Message: element not visible 1.元素定位失败(使用frame ...

  3. redis 学习(二)-- 通用命令

    redis 学习(二)-- 通用命令 1. keys pattern 含义:查找所有符合给定模式(pattern)的key 命令 含义 keys * 遍历所有 key keys he[h-l]* 遍历 ...

  4. wpf GridSplitter左右托不了或者拖拽异常

    对于水平分割线,需要将verticalAlignment属性设置为Center 对于垂直分割线,需要将horizontalAlignment属性设置为center 切记切记,不然很苦逼....

  5. kubernetes基本了解

    初识Kubernetes----k8s以及功能 kubernetes是由google公司开发的容器集群管理系统.采用go语言开发.也称为k8s,原因为k后面直到s这中间有8个字母,所以叫k8s.它主要 ...

  6. springboot(十八)-session共享

    前言 在传统的单服务架构中,一般来说,只有一个服务器,那么不存在 Session 共享问题,但是在分布式/集群项目中,Session 共享则是一个必须面对的问题,先看一个简单的架构图: 在这样的架构中 ...

  7. javaScript基本使用api

    基本方法 isArray() 判断数组 isArray() 方法用于判断是否是数组(有兼容性) 语法:Array.isArray(arr) 返回值:是数组,返回true.不是数组,返回false. i ...

  8. PHP--API

    PHP所有能力都是函数,内置1000多个函数,不是每一个函数都默认直接可以使用,有一些需要安装或者启用额外的“插件”扩展. 1,获取字符串长度 <?php $str='hello'; echo ...

  9. C语言字符串函数总结

    原文链接 函数名: stpcpy 功 能: 拷贝一个字符串到另一个 用 法: char *stpcpy(char *destin, char *source); 程序例: #include <s ...

  10. python cv2截取不规则区域图片

    知识掌握 cv2.threshold()函数: 设置固定级别的阈值应用于多通道矩阵,将灰度图像变换二值图像,或去除指定级别的噪声,或过滤掉过小或者过大的像素点. Python: cv2.thresho ...