线段树的每个叶子节点存一列。

每个节点维护六个域,分别是左上左下、左上右上、左上右下、左下右上、左下右下、右上右下在区间内部的连通性,不考虑绕出去的情况。

初始每个叶子的左上左下、右上右下是连通的。

每次修改纵列时,直接去线段树里修改。

每次修改横行时,将其左右两列全都pushup一遍。

具体怎么维护,并不难,请自行思考。

每次询问时,提取出来[1,l-1],[l,r],[r+1,n]的连通性,然后讨论是否绕出来的情况。

不推荐使用循环维护,人工讨论反而更加方便。

#include<cstdio>
#include<algorithm>
using namespace std;
#define N 100001
int n;
bool heng[2][N];/*heng(i,j)±íʾ(i,j)ºÍ(i+1,j)Ö®¼äÊÇ·ñÓбß*/
bool zong[N];/*zong(i)±íʾ(0,i)ºÍ(1,i)Ö®¼äÊÇ·ñÓбß*/
struct Node{bool zszx,zsys,zsyx,zxys,zxyx,ysyx;}T[N<<2];
/*!!!ÐèÒªµ¥¶ÀÌÖÂÛµ¥¸ö×ÝÁÐºÍÆäËûµÄºÏ²¢µÄÇé¿ö!!!...ºÃ°ÉÆäʵ²»ÓÃ...³õʼʱÿ¸ö×ÝÁеÄzsys,zxyx=1*/
inline void pushup(Node &rt,const Node &ls,const Node &rs,const int &r1)
{
if(ls.zszx) rt.zszx=1;//zszx
else if(ls.zsys && rs.zszx && ls.zxyx && heng[0][r1] && heng[1][r1]) rt.zszx=1;
else rt.zszx=0;
if(rs.ysyx) rt.ysyx=1;//ysyx
else if(rs.zsys && ls.ysyx && rs.zxyx && heng[0][r1] && heng[1][r1]) rt.ysyx=1;
else rt.ysyx=0;
if(ls.zsys && rs.zsys && heng[0][r1]) rt.zsys=1;//zsys
else if(ls.zsyx && rs.zxys && heng[1][r1]) rt.zsys=1;
else rt.zsys=0;
if(ls.zsys && rs.zsyx && heng[0][r1]) rt.zsyx=1;//zsyx
else if(ls.zsyx && rs.zxyx && heng[1][r1]) rt.zsyx=1;
else rt.zsyx=0;
if(ls.zxys && rs.zsys && heng[0][r1]) rt.zxys=1;//zxys
else if(ls.zxyx && rs.zxys && heng[1][r1]) rt.zxys=1;
else rt.zxys=0;
if(ls.zxys && rs.zsyx && heng[0][r1]) rt.zxyx=1;//zxyx
else if(ls.zxyx && rs.zxyx && heng[1][r1]) rt.zxyx=1;
else rt.zxyx=0;
}
void update(int p,bool v,int rt,int l,int r)/*µ¥×ÝÁÐÐÞ¸Ä*/
{
if(l==r)
{
zong[p]=v;
T[rt].zszx=T[rt].zsyx=T[rt].zxys=T[rt].ysyx=v;
return;
}
int m=(l+r>>1);
if(p<=m) update(p,v,rt<<1,l,m);
else update(p,v,rt<<1|1,m+1,r);
pushup(T[rt],T[rt<<1],T[rt<<1|1],m);
}
void update(int p,int rt,int l,int r)/*ÓÉÓÚÅԱߵĺáÐнøÐÐÁËÐ޸Ķø½øÐеĵ¥×ÝÁиüÐÂ*/
{
if(l==r) return;
int m=(l+r>>1);
if(p<=m) update(p,rt<<1,l,m);
else update(p,rt<<1|1,m+1,r);
pushup(T[rt],T[rt<<1],T[rt<<1|1],m);
}
Node query(int ql,int qr,int rt,int l,int r)
{
if(ql<=l && r<=qr) return T[rt];
int m=(l+r>>1);
if(ql<=m && m<qr)
{
Node res;
pushup(res,query(ql,qr,rt<<1,l,m),query(ql,qr,rt<<1|1,m+1,r),m);
return res;
}
else if(ql<=m) return query(ql,qr,rt<<1,l,m);
else return query(ql,qr,rt<<1|1,m+1,r);
}
inline bool query(const bool &x1,const int &y1,const bool &x2,const int &y2)
{
Node M=query(y1,y2,1,1,n),L,R;
if(y1!=1) L=query(1,y1-1,1,1,n);
if(y2!=n) R=query(y2+1,n,1,1,n);
if(x1 && x2)
{
if(M.zxyx) return 1;
if(y1!=1) if(L.ysyx && M.zsyx && heng[0][y1-1] && heng[1][y1-1]) return 1;
if(y2!=n) if(R.zszx && M.zxys && heng[0][y2] && heng[1][y2]) return 1;
if(y1!=1 && y2!=n) if(L.ysyx && heng[0][y1-1] && heng[1][y1-1]
&& R.zszx && heng[0][y2] && heng[1][y2] && M.zsys)
return 1;
return 0;
}
else if(x1)
{
if(M.zxys) return 1;
if(y1!=1) if(L.ysyx && M.zsys && heng[0][y1-1] && heng[1][y1-1]) return 1;
if(y2!=n) if(R.zszx && M.zxyx && heng[0][y2] && heng[1][y2]) return 1;
if(y1!=1 && y2!=n) if(L.ysyx && heng[0][y1-1] && heng[1][y1-1]
&& R.zszx && heng[0][y2] && heng[1][y2] && M.zsyx)
return 1;
return 0;
}
else if(x2)
{
if(M.zsyx) return 1;
if(y1!=1) if(L.ysyx && M.zxyx && heng[0][y1-1] && heng[1][y1-1]) return 1;
if(y2!=n) if(R.zszx && M.zsys && heng[0][y2] && heng[1][y2]) return 1;
if(y1!=1 && y2!=n) if(L.ysyx && heng[0][y1-1] && heng[1][y1-1]
&& R.zszx && heng[0][y2] && heng[1][y2] && M.zxys)
return 1;
return 0;
}
else
{
if(M.zsys) return 1;
if(y1!=1) if(L.ysyx && M.zxys && heng[0][y1-1] && heng[1][y1-1]) return 1;
if(y2!=n) if(R.zszx && M.zsyx && heng[0][y2] && heng[1][y2]) return 1;
if(y1!=1 && y2!=n) if(L.ysyx && heng[0][y1-1] && heng[1][y1-1]
&& R.zszx && heng[0][y2] && heng[1][y2] && M.zxyx)
return 1;
return 0;
}
}
int main()
{
// freopen("bzoj1018.in","r",stdin);
char op[7];
int x1,x2,y1,y2;
scanf("%d",&n);
for(int i=1;i<=(n<<2);++i)
T[i].zsys=T[i].zxyx=1;
while(1)
{
scanf("%s",op);
if(op[0]=='E') break;
else if(op[0]=='O')
{
scanf("%d%d%d%d",&x1,&y1,&x2,&y2);
if(y1==y2) update(y1,1,1,1,n);
else
{
if(y1>y2) swap(y1,y2);
heng[x1-1][y1]=1;
update(y1,1,1,n);
update(y2,1,1,n);
}
}
else if(op[0]=='C')
{
scanf("%d%d%d%d",&x1,&y1,&x2,&y2);
if(y1==y2) update(y1,0,1,1,n);
else
{
if(y1>y2) swap(y1,y2);
heng[x1-1][y1]=0;
update(y1,1,1,n);
update(y2,1,1,n);
}
}
else
{
scanf("%d%d%d%d",&x1,&y1,&x2,&y2);
if(y1>y2)
{
swap(x1,x2);
swap(y1,y2);
}
puts(query(x1-1,y1,x2-1,y2) ? "Y" : "N");
}
}
return 0;
}

【线段树】bzoj1018 [SHOI2008]堵塞的交通traffic的更多相关文章

  1. [BZOJ1018][SHOI2008]堵塞的交通traffic 线段树维护连通性

    1018: [SHOI2008]堵塞的交通traffic Time Limit: 3 Sec  Memory Limit: 162 MB Submit: 3795  Solved: 1253 [Sub ...

  2. bzoj千题计划108:bzoj1018: [SHOI2008]堵塞的交通traffic

    http://www.lydsy.com/JudgeOnline/problem.php?id=1018 关键点在于只有两行 所以一个2*m矩形连通情况只有6种 编号即对应代码中的a数组 线段树维护 ...

  3. 【离线 撤销并查集 线段树分治】bzoj1018: [SHOI2008]堵塞的交通traffic

    本题可化成更一般的问题:离线动态图询问连通性 当然可以利用它的特殊性质,采用在线线段树维护一些标记的方法 Description 有一天,由于某种穿越现象作用,你来到了传说中的小人国.小人国的布局非常 ...

  4. Bzoj1018[SHOI2008]堵塞的交通traffic(线段树)

    这题需要维护连通性,看到有连接删除,很容易直接就想LCT了.然而这题点数20w操作10w,LCT卡常估计过不去.看到这个东西只有两行,考虑能否用魔改后的线性数据结构去维护.我想到了线段树. 考虑如果两 ...

  5. bzoj1018[SHOI2008]堵塞的交通traffic——线段树

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1018 巧妙的线段树.维护矩阵四个角的连通性. 考虑两个点连通的可能路径分成3部分:两点左边. ...

  6. [BZOJ1018][SHOI2008]堵塞的交通traffic 时间分治线段树

    题面 介绍一种比较慢的但是好想的做法. 网上漫天的线段树维护联通性,然后想起来费很大周折也很麻烦.我的做法也是要用线段树的,不过用法完全不同. 这个东西叫做时间分治线段树. 首先我们建一个\(1..m ...

  7. BZOJ1018 [SHOI2008]堵塞的交通traffic

    本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000作者博客:http://www.cnblogs.com/ljh2000-jump/转 ...

  8. [bzoj1018][SHOI2008]堵塞的交通traffic_线段树

    bzoj-1018 SHOI-2008 堵塞的交通traffic 参考博客:https://www.cnblogs.com/MashiroSky/p/5973686.html 题目大意:有一天,由于某 ...

  9. 【BZOJ1018】[SHOI2008]堵塞的交通traffic 线段树

    [BZOJ1018][SHOI2008]堵塞的交通traffic Description 有一天,由于某种穿越现象作用,你来到了传说中的小人国.小人国的布局非常奇特,整个国家的交通系统可以被看成是一个 ...

随机推荐

  1. Acadia Lab 203 + Lab 231

    在做完 Lab 6 之后,惊觉选做实验缺口很大,于是遍历了一遍夏任务,找到了一条最省力的路线. 做完 Lab 6 的连线不用拆,可以接下来做以下两个实验: Lab 203 网络时钟 核心代码如下: v ...

  2. scikit-learn实现ebay数据分析 的随笔

    注:只是随笔 import pandas as pd train = pd.read_csv()  读入scv格式的文件 train = train_set.drop(['EbayID','Quant ...

  3. LDAP客户端

    LDAP客户端通过与服务端关联起来,就可以使用服务端的系统账号登录系统,通过useradd 添加用户是在ldap里是没有显示的,ldap添加用户,在/etc/passwd里也是没有显示的,ldap添加 ...

  4. 怎样从altera下载软件与器件库

    首先要注册一个帐号,否则是不能下载的. step1:进入support->download 这是页面下方的显示,可以选择想要安装的Quartus版本以及该版本支持的器件.这里以16.0标准版为例 ...

  5. JQuery源码解析-- 对象的创建

    使用 $("a") 返回的对象就不再是一个简单的DOM对象了,而是一个复杂的JQuery对象. 那么JQuery是怎么创建对象的. 为了便于分析,我将JQuery中复杂的代码简化了 ...

  6. JavaWeb 学习005-4个页面,5条sql语句(添加、查看、修改、删除)

    今天完成了的事: = 1.班级模块,list页面的添加班级信息操作,中文乱码问题 尚待解决 这俩天做得工作是: 首先搭建好main页面:是由top.left.right三个部分构成的,拼接到一起. 流 ...

  7. Android 手机自动化测试工具有哪几种?

    1.Monkey是Android SDK自带的测试工具,在测试过程中会向系统发送伪随机的用户事件流,如按键输入.触摸屏输入.手势输入等),实现对正在开发的应用程序进行压力测试,也有日志输出.实际上该工 ...

  8. TFS 服务端默认端口更改

    由于服务商限制8080,为了外网能访问.如果可以做映射还好.如果不能那就修改默认端口 以下为网上资源 安装完Team Foundation Server 2005 后,默认的端口是8080.如果想要事 ...

  9. CyclicBarrier在多线程同步运行后相互访问的问题。

    CyclicBarrier的介绍 一个同步辅助类,它允许一组线程互相等待,直到到达某个公共屏障点 (common barrier point).在涉及一组固定大小的线程的程序中,这些线程必须不时地互相 ...

  10. python课程第三周重点记录

    1.关于set,set是一个无序且不重复的元素集合 s = set(["ss","dd","pp"]) #创建set s1 = set([& ...