题目链接

只有两行,可能的路径数不多,考虑用线段树维护各种路径的连通性。

每个节点记录luru(left_up->right_up),lurd,ldru,ldrd,luld,rurd,表示这个区间对应的连通信息。

合并时 横向道路需要用两个数组up[],down[]记录连通性;纵向道路在修改时直接单点改,再PushUp。

询问时连通的可能性有多种,都需判断。

//5708kb	768ms
#include <cstdio>
#include <cctype>
#include <algorithm>
#define gc() getchar()
const int N=1e5+5; int n,tot;//root=0
bool road[2][N];
struct Node
{
int l,r,ls,rs;
bool luru,lurd,ldru,ldrd,luld,rurd;
}node[N<<1]; inline int read()
{
int now=0;register char c=gc();
for(;!isdigit(c);c=gc());
for(;isdigit(c);now=now*10+c-'0',c=gc());
return now;
}
#define now node[rt]
#define nls node[ls]
#define nrs node[rs]
void Build(int l,int r)
{
int p=tot++; node[p].l=l, node[p].r=r;
if(l==r) node[p].luru=node[p].ldrd=1;//这个初始化要有。
else{
int m=l+r>>1;
node[p].ls=tot, Build(l,m),
node[p].rs=tot, Build(m+1,r);
}
}
//void Print(Node rt){
// printf("%d~%d\nluru:%d lurd:%d ldru:%d ldrd:%d luld:%d rurd:%d\n",rt.l,rt.r,rt.luru,rt.lurd,rt.ldru,rt.ldrd,rt.luld,rt.rurd);
//}
inline void Update(int rt)
{
int ls=now.ls,rs=now.rs,p=node[ls].r;
now.luru=(nls.luru&&road[0][p]&&nrs.luru)||(nls.lurd&&road[1][p]&&nrs.ldru);
now.lurd=(nls.luru&&road[0][p]&&nrs.lurd)||(nls.lurd&&road[1][p]&&nrs.ldrd);
now.ldru=(nls.ldrd&&road[1][p]&&nrs.ldru)||(nls.ldru&&road[0][p]&&nrs.luru);
now.ldrd=(nls.ldrd&&road[1][p]&&nrs.ldrd)||(nls.ldru&&road[0][p]&&nrs.lurd);
now.luld=nls.luld||(nls.luru&&road[0][p]&&nrs.luld&&road[1][p]&&nls.ldrd);
now.rurd=nrs.rurd||(nrs.luru&&road[0][p]&&nls.rurd&&road[1][p]&&nrs.ldrd);
}
Node Merge(const Node &ls,const Node &rs)
{
Node tmp;
int p=ls.r; tmp.l=ls.l, tmp.r=rs.r;//左右端点要更新!(p)
tmp.luru=(ls.luru&&road[0][p]&&rs.luru)||(ls.lurd&&road[1][p]&&rs.ldru);
tmp.lurd=(ls.luru&&road[0][p]&&rs.lurd)||(ls.lurd&&road[1][p]&&rs.ldrd);
tmp.ldru=(ls.ldrd&&road[1][p]&&rs.ldru)||(ls.ldru&&road[0][p]&&rs.luru);
tmp.ldrd=(ls.ldrd&&road[1][p]&&rs.ldrd)||(ls.ldru&&road[0][p]&&rs.lurd);
tmp.luld=ls.luld||(ls.luru&&road[0][p]&&rs.luld&&road[1][p]&&ls.ldrd);
tmp.rurd=rs.rurd||(rs.luru&&road[0][p]&&ls.rurd&&road[1][p]&&rs.ldrd);
return tmp;
}
void Modify_Row(int rt,int p,bool is_down,bool v)
{
int m=now.l+now.r>>1;
if(m==p){//当前区间从m处与m+1断开,所以应该在这更新!
road[is_down][m]=v;
if(now.l!=now.r) Update(rt);
}
else{
if(p<=m) Modify_Row(now.ls,p,is_down,v);
else Modify_Row(now.rs,p,is_down,v);
Update(rt);
}
}
void Modify_Col(int rt,int p,bool v)
{
if(now.l==now.r)
/*now.luru=now.ldrd=*/now.lurd=now.ldru=now.luld=now.rurd=v;//更新列之间的就可以了?
else{
int m=now.l+now.r>>1;
if(p<=m) Modify_Col(now.ls,p,v);
else Modify_Col(now.rs,p,v);
Update(rt);
}
}
Node Query(int rt,int L,int R)
{
if(L<=now.l && now.r<=R) return now;
int m=now.l+now.r>>1;
if(L<=m)
if(m<R) return Merge(Query(now.ls,L,R),Query(now.rs,L,R));
else return Query(now.ls,L,R);
return Query(now.rs,L,R);
}
void Ask(int r1,int c1,int r2,int c2)
{
Node l=Query(0,1,c1),m=Query(0,c1,c2),r=Query(0,c2,n);//由这三个位置去判断
// Print(l), Print(m), Print(r);
if(r1==1 && r2==1)
{
if ((m.luru)||
(m.lurd&&r.luld)||
(m.ldru&&l.rurd)||
(l.rurd&&m.ldrd&&r.luld)) puts("Y");
else puts("N");
}
else if(r1==2 && r2==2)
{
if ((m.ldrd)||
(m.ldru&&r.luld)||
(m.lurd&&l.luld)||
(l.rurd&&m.luru&&r.luld)) puts("Y");
else puts("N");
}
else if(r1==1)//r2==2
{
if ((m.lurd)||
(m.luru&&r.luld)||
(m.ldrd&&l.rurd)||
(l.rurd&&m.ldru&&r.luld)) puts("Y");
else puts("N");
}
else if(r1==2)//r2==1
{
if ((m.ldru)||
(m.ldrd&&r.luld)||
(m.luru&&l.rurd)||
(l.rurd&&m.lurd&&r.luld)) puts("Y");
else puts("N");
}
}
//void Print_All(int rt){
// if(now.ls) Print_All(now.ls);
// Print(node[rt]);
// if(now.rs) Print_All(now.rs);
//} int main()
{
n=read(), Build(1,n);
char opt[3]; int r1,r2,c1,c2;
// Print_All(0);
while(scanf("%s",opt), opt[0]!='E')
{
r1=read(),c1=read(),r2=read(),c2=read();
if(c1>c2) std::swap(r1,r2),std::swap(c1,c2);//不是在r1>r2的时候换。
if(opt[0]=='C')
if(r1==r2) Modify_Row(0,c1,r1==2,0);
else Modify_Col(0,c1,0);
else if(opt[0]=='O')
if(r1==r2) Modify_Row(0,c1,r1==2,1);
else Modify_Col(0,c1,1);
else Ask(r1,c1,r2,c2);
// Print_All(0);
}
return 0;
}

BZOJ.1018.[SHOI2008]堵塞的交通(线段树维护连通性)的更多相关文章

  1. BZOJ 1018: [SHOI2008]堵塞的交通traffic [线段树 区间信息]

    1018: [SHOI2008]堵塞的交通traffic Time Limit: 3 Sec  Memory Limit: 162 MBSubmit: 3064  Solved: 1027[Submi ...

  2. 数据结构(线段树):BZOJ 1018: [SHOI2008]堵塞的交通traffic

    1018: [SHOI2008]堵塞的交通traffic Time Limit: 3 Sec  Memory Limit: 162 MBSubmit: 2638  Solved: 864 Descri ...

  3. BZOJ 1018 [SHOI2008]堵塞的交通traffic

    1018: [SHOI2008]堵塞的交通traffic Time Limit: 3 Sec  Memory Limit: 162 MBSubmit: 2247  Solved: 706[Submit ...

  4. BZOJ1018[SHOI2008]堵塞的交通——线段树

    题目描述 有一天,由于某种穿越现象作用,你来到了传说中的小人国.小人国的布局非常奇特,整个国家的交通系统可以被看成是一个2行C列的矩形网格,网格上的每个点代表一个城市,相邻的城市之间有一条道路,所以总 ...

  5. 洛谷 4246 BZOJ 1018 [SHOI2008]堵塞的交通

    [题解] 原来线段树还可以这么玩.. 我们用线段树维护连通性.对于一个矩形,我们用4个标记维护4个点的联通情况,再用两个标记维护右边两个点与它们右边的与它们在同一行的点的联通情况. 画图表示,就是 另 ...

  6. [BZOJ 3995] [SDOI2015] 道路修建 【线段树维护连通性】

    题目链接:BZOJ - 3995 题目分析 这道题..是我悲伤的回忆.. 线段树维护连通性,与 BZOJ-1018 类似,然而我省选之前并没有做过  1018,即使它在 ProblemSet 的第一页 ...

  7. [BZOJ 1018] [SHOI2008] 堵塞的交通traffic 【线段树维护联通性】

    题目链接:BZOJ - 1018 题目分析 这道题就说明了刷题少,比赛就容易跪..SDOI Round1 Day2 T3 就是与这道题类似的..然而我并没有做过这道题.. 这道题是线段树维护联通性的经 ...

  8. BZOJ 1018: [SHOI2008]堵塞的交通traffic(线段树)

    题目:http://www.lydsy.com/JudgeOnline/problem.php?id=1018 用线段树维护区间连通性,对于每一个区间记录6个域表示(左上,左下)(左上,右上)(右上, ...

  9. Luogu P4246 [SHOI2008]堵塞的交通(线段树+模拟)

    P4246 [SHOI2008]堵塞的交通 题意 题目描述 有一天,由于某种穿越现象作用,你来到了传说中的小人国.小人国的布局非常奇特,整个国家的交通系统可以被看成是一个\(2\)行\(C\)列的矩形 ...

随机推荐

  1. Spark记录-官网学习配置篇(一)

    参考http://spark.apache.org/docs/latest/configuration.html Spark提供三个位置来配置系统: Spark属性控制大多数应用程序参数,可以使用Sp ...

  2. 实现vue2.0响应式的基本思路

    最近看了vue2.0源码关于响应式的实现,以下博文将通过简单的代码还原vue2.0关于响应式的实现思路. 注意,这里只是实现思路的还原,对于里面各种细节的实现,比如说数组里面数据的操作的监听,以及对象 ...

  3. Git与GitHub学习笔记(三).gitignore文件忽略和删除本地以及远程文件

    一.Git提供了文件忽略功能.当对工作区某个目录或者某些文件设置了忽略后,git将不会对它们进行追踪 HELP:如何在IntelliJ IDEA中使用.ignore插件忽略不必要提交的文件 问题:最近 ...

  4. Kafka 温故(一):Kafka背景及架构介绍

    一.Kafka简介 Kafka是分布式发布-订阅消息系统.它最初由LinkedIn公司开发,使用Scala语言编写,之后成为Apache项目的一部分.Kafka是一个分布式的,可划分的,多订阅者,冗余 ...

  5. 基于JWT(Json Web Token)的ASP.NET Web API授权方式

    token应用流程 初次登录:用户初次登录,输入用户名密码 密码验证:服务器从数据库取出用户名和密码进行验证 生成JWT:服务器端验证通过,根据从数据库返回的信息,以及预设规则,生成JWT 返还JWT ...

  6. HTML5 中 Geolocation 获取地理位置的原理是什么?

    http://www.zhihu.com/question/20473051?sort=created geolocation的位置信息来源包括GPS.IP地址.RFID.WIFI和蓝牙的MAC地址. ...

  7. spm

    Spatial Pyramid Matching 看了很多关于SPM的介绍,但是网络上的资源大多都是对论文Beyond bags of features: Spatial pyramid matchi ...

  8. Python 入门基础5 --元组、字典、集合

    今日目录: 一.元组 二.字典 三.集合 四.后期添加内容 一.元组 1.定义 t1 = () # 参数为for可以循环的对象(可迭代对象) 思考: 如何定义一个只有一个值的元组? ("li ...

  9. j2ee组件简介

  10. imperva 更改agent的注册密码

    imperva agent 在注册到网关的时候显示账号密码错误 如下图 这是一个数据库审计的设备由于当初最开始实施的时候并不是我安装的,所以账号密码我也不清楚,客户留的账号密码也不确定.因此导致账号密 ...