Description

有一天,由于某种穿越现象作用,你来到了传说中的小人国。小人国的布局非常奇特,整个国家的交通系统可以被看成是一个2行C列的矩形网格,网格上的每个点代表一个城市,相邻的城市之间有一条道路,所以总共有2C个城市和3C-2条道路。 小人国的交通状况非常槽糕。有的时候由于交通堵塞,两座城市之间的道路会变得不连通,直到拥堵解决,道路才会恢复畅通。初来咋到的你决心毛遂自荐到交通部某份差事,部长听说你来自一个科技高度发达的世界,喜出望外地要求你编写一个查询应答系统,以挽救已经病入膏肓的小人国交通系统。 小人国的交通部将提供一些交通信息给你,你的任务是根据当前的交通情况回答查询的问题。交通信息可以分为以下几种格式: Close r1 c1 r2 c2:相邻的两座城市(r1,c1)和(r2,c2)之间的道路被堵塞了; Open r1 c1 r2 c2:相邻的两座城市(r1,c1)和(r2,c2)之间的道路被疏通了; Ask r1 c1 r2 c2:询问城市(r1,c1)和(r2,c2)是否连通。如果存在一条路径使得这两条城市连通,则返回Y,否则返回N;

Input

第一行只有一个整数C,表示网格的列数。接下来若干行,每行为一条交通信息,以单独的一行“Exit”作为结束。我们假设在一开始所有的道路都是堵塞的。 对30%测试数据,我们保证C小于等于1000,信息条数小于等于1000; 对100%测试数据,我们保证 C小于等于100000,信息条数小于等于100000。

Output

对于每个查询,输出一个“Y”或“N”。

Sample Input

2
Open 1 1 1 2
Open 1 2 2 2
Ask 1 1 2 2
Ask 2 1 2 2
Exit

Sample Output

Y
N

HINT

 

Source

一道很恶心的数据结构题,但是数据结构却很简单,就是一颗裸的线段树。然而,用线段树维护连通性却很恶心(因为它只有相邻的两个点之间有连边)。我的线段树维护了8个域——左右,上下,对角,中间。慢慢写,反正我是写醉了。

奉劝:(代码仅限对拍,莫看莫看)

 #include<iostream>
#include<cstdio>
#include<cstdlib>
using namespace std; #define maxn 100010
int n;
struct node
{
int lc,rc,l,r;
bool u[]; //左上-左下 右上-右下
bool v[]; //左上-右上 左下-右下
bool w[]; //左下-右上 左上-右下
bool t[]; //中间的两个
};
struct SEG
{
int cnt; node tree[maxn*]; inline void updata(int now,bool flag)
{
tree[now].w[] = tree[now].w[] = false;
if (!flag) if (tree[now].l != tree[now].r) tree[now].u[] = tree[now].u[] = false,tree[now].v[] = tree[now].v[] = false;
int lc = tree[now].lc,rc = tree[now].rc; bool sign = false;
do
{
sign = false;
if (!tree[now].u[])
{
if (tree[now].v[]&&tree[now].w[])
tree[now].u[] = true,sign = true;
else if (tree[now].v[]&&tree[now].w[])
tree[now].u[] = true,sign = true;
else if (tree[now].u[]&&tree[now].v[]&&tree[now].v[])
tree[now].u[] = true,sign = true;
else if (tree[lc].u[]) tree[now].u[] = true,sign = true;
}
if (!tree[now].u[])
{
if (tree[now].v[]&&tree[now].w[])
tree[now].u[] = true,sign = true;
else if (tree[now].v[]&&tree[now].w[])
tree[now].u[] = true,sign = true;
else if (tree[now].u[]&&tree[now].v[]&&tree[now].v[])
tree[now].u[] = true,sign = true;
else if (tree[rc].u[]) tree[now].u[] = true,sign = true;
}
if (!tree[now].v[])
{
if (tree[now].r - tree[now].l == && tree[now].t[])
tree[now].v[] = true,sign = true;
else if (tree[now].u[]&&tree[now].w[])
tree[now].v[] = true,sign = true;
else if (tree[now].u[]&&tree[now].w[])
tree[now].v[] = true,sign = true;
else if (tree[lc].v[]&&tree[rc].v[]&&tree[now].t[])
tree[now].v[] = true,sign = true;
else if (tree[lc].w[]&&tree[now].t[]&&tree[rc].w[])
tree[now].v[] = true,sign = true;
}
if (!tree[now].v[])
{
if (tree[now].r - tree[now].l == && tree[now].t[])
tree[now].v[] = true,sign = true;
if (tree[now].u[]&&tree[now].w[])
tree[now].v[] = true,sign = true;
else if (tree[now].u[]&&tree[now].w[])
tree[now].v[] = true,sign = true;
else if (tree[lc].v[]&&tree[rc].v[]&&tree[now].t[])
tree[now].v[] = true,sign = true;
else if (tree[lc].w[]&&tree[now].t[]&&tree[rc].w[])
tree[now].v[] = true,sign = true;
}
if (!tree[now].w[])
{
if (tree[now].u[]&&tree[now].v[])
tree[now].w[] = true,sign = true;
else if (tree[now].u[]&&tree[now].v[])
tree[now].w[] = true,sign = true;
else if (tree[lc].w[]&&tree[now].t[]&&tree[rc].v[])
tree[now].w[] = true,sign = true;
else if (tree[lc].v[]&&tree[now].t[]&&tree[now].w[])
tree[now].w[] = true,sign = true;
}
if (!tree[now].w[])
{
if (tree[now].u[]&&tree[now].v[])
tree[now].w[] = true,sign = true;
else if (tree[now].u[]&&tree[now].v[])
tree[now].w[] = true,sign = true;
else if (tree[lc].w[]&&tree[now].t[]&&tree[rc].v[])
tree[now].w[] = true,sign = true;
else if (tree[lc].v[]&&tree[now].t[]&&tree[rc].w[])
tree[now].w[] = true,sign = true;
}
}
while (sign);
if (tree[now].l == tree[now].r)
tree[now].w[] = tree[now].w[] = false;
} inline int build(int l,int r)
{
int now = ++cnt,mid = (l + r) >> ;
tree[now].l = l; tree[now].r = r;
if (l == r)
{
tree[now].v[] = tree[now].v[] = true;
updata(now,);
return now;
}
tree[now].lc = build(l,mid);
tree[now].rc = build(mid+,r);
updata(now,);
return now;
} inline bool ask(int a1,int b1,int a2,int b2,int now)
{
if (a1 == a2 && b1 == b2) return true;
int l = tree[now].l,r = tree[now].r,lc = tree[now].lc,rc = tree[now].rc,mid = (l + r) >> ;
if (b1 == l&&b2 == r)
{
if (a1 == a2 && tree[now].v[a1]) return true;
if (a1 != a2)
{
if (a1 == && tree[now].w[]) return true;
if (a1 == && tree[now].w[]) return true;
if (l == r && tree[now].u[]) return true;
}
}
else if (b2 <= mid && ask(a1,b1,a2,b2,lc)) return true;
else if (b1 > mid && ask(a1,b1,a2,b2,rc)) return true;
else if (mid >= b1 && mid < b2 && tree[now].t[]&&ask(a1,b1,,mid,lc)&&ask(,mid+,a2,b2,rc)) return true;
else if (mid >= b1 && mid < b2 && tree[now].t[]&&ask(a1,b1,,mid,lc)&&ask(,mid+,a2,b2,rc)) return true;
else if (a1 != a2 && tree[now].t[] && tree[now].t[])
{
if (b2 <= mid) return tree[rc].u[]&&ask(a1,b1,a1,mid,lc)&&ask(a2,b2,a2,mid,lc);
else if (b1 > mid) return tree[lc].u[]&&ask(a1,mid+,a1,b1,rc)&&ask(a2,mid+,a2,b2,rc);
}
else if (a1 == a2 && tree[now].t[]&&tree[now].t[])
{
if (b2 <= mid)
return tree[rc].u[]&&ask(a1,b1,((a1-)^)+,b1,lc)&&ask(((a1-)^)+,b1,((a1-)^)+,mid,lc)&&ask(a2,b2,a2,mid,lc);
else if (b1 > mid)
return tree[lc].u[]&&ask(a1,b2,((a1-)^)+,b2,rc)&&ask(((a1-)^)+,mid+,((a1-)^)+,b2,rc)&&ask(a2,mid+,a2,b1,rc);
}
return false;
} inline void modify(int a1,int b1,int a2,int b2,int now,bool sign)
{
int l = tree[now].l,r = tree[now].r,lc = tree[now].lc,rc = tree[now].rc,mid = (l + r) >> ;
bool flag = false;
if (a1 == a2) //处在同一行上
{
if (b1 == mid && b2 == mid + )
tree[now].t[a1] = sign,flag = sign;
else if (b1 <= mid) modify(a1,b1,a2,b2,lc,sign);
else modify(a1,b1,a2,b2,rc,sign);
}
else //处在同一列上
{
if (l == r) tree[now].u[] = tree[now].u[] = sign,flag = true;
else if (b1 <= mid) modify(a1,b1,a2,b2,lc,sign);
else modify(a1,b1,a2,b2,rc,sign);
}
updata(now,flag|sign);
return;
}
}seg; inline int read()
{
int x=,f=; char ch=getchar();
while(ch<''||ch>'') { if(ch=='-')f=-; ch=getchar(); }
while(ch>=''&&ch<='') { x=x*+ch-''; ch=getchar(); }
return x * f;
} int main()
{
freopen("1018.in","r",stdin);
freopen("1018.out","w",stdout);
n = read();
seg.build(,n);
char opt[]; int a1,a2,b1,b2;
do
{
scanf("%s ",opt);
if (opt[] == 'E') break;
a1 = read(); b1 = read(); a2 = read(); b2 = read();
if (b1 > b2) swap(a1,a2),swap(b1,b2);
if (opt[] == 'A')
{
if (seg.ask(a1,b1,a2,b2,)) printf("Y\n");
else printf("N\n");
}
else if (opt[] == 'O') seg.modify(a1,b1,a2,b2,,true);
else seg.modify(a1,b1,a2,b2,,false);
}
while (true);
return ;
}

BZOJ 1018 堵塞的交通的更多相关文章

  1. [BZOJ]1018 堵塞的交通(SHOI2008)

    一道有点神的线段树. Description 有一天,由于某种穿越现象作用,你来到了传说中的小人国.小人国的布局非常奇特,整个国家的交通系统可以被看成是一个2行C列的矩形网格,网格上的每个点代表一个城 ...

  2. BZOJ 1018 堵塞的交通traffic(线段树)

    题目链接:http://61.187.179.132/JudgeOnline/problem.php?id=1018 题意:一个2*n的格子,相邻格子之间有一条道路.初始时道路是不通的. 三种操作:( ...

  3. 【BZOJ】【1018】【SHOI2008】堵塞的交通traffic

    线段树 这题的线段树+分类讨论蛮神奇的……我以前学的线段树简直就是渣渣QAQ 看了下ydc题解里的思想>_>用线段树维护连通性!那么就自己写吧……每个节点表示一段区间的连通性(我的叶子节点 ...

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

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

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

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

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

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

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

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

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

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

  9. 1018: [SHOI2008]堵塞的交通traffic

    1018: [SHOI2008]堵塞的交通traffic 链接 分析: 用线段树维护区间的四个端点的联通情况,然后查询的时候,把所有覆盖到的区间合并起来即可. 六种情况左上到右上(左边到右边的情况)… ...

随机推荐

  1. jQuery 实现上下,左右滑动

    前几天的任务:http://t.sina.com.cn/  的下滑效果. 渐变移动出足够的空白 -> 淡出最后一个 ->渐变移动出足够的空白 我们要做的是向左移动效果.这个效果用时需添加一 ...

  2. Android中的主题Theme

    系统自带的Theme: android以及为我们定义好了一些theme,需要是我们直接可以拿来使用. 常用的Theme通常如下:  android:theme="@android:style ...

  3. ReentrantLock与synchronized的差别

    总的来说,lock更加灵活. 主要同样点:Lock能完毕synchronized所实现的全部功能 不同: 1.ReentrantLock功能性方面更全面,比方时间锁等候,可中断锁等候,锁投票等,因此更 ...

  4. android 57 QQ登录

    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools=&q ...

  5. Java基础知识强化之集合框架笔记35:List练习之产生10个1~20之间的随机数(要求:随机数不能重复)

    1. 需求:获取10个1-20之间的随机数,要求不能重复 用数组实现,但是数组的长度是固定的,长度不好确定.所以我们使用集合实现. 分析: • 创建产生随机数的对象 • 创建一个存储随机数的集合. • ...

  6. 零基础学习云计算及大数据DBA集群架构师【预科2015年12月14日周一】

    1.第一天比较轻松,上午填表格,录指纹,拍照片,做自我介绍. 2.下午老师简单介绍了一下PC\交换机\路由器\塔式服务器\机架式服务器(1U\2U)\刀片服务器\磁带机 3.班主任陈老师朱老师,预科秦 ...

  7. 树形dp练习

    /*poj 1463 最小点覆盖 匈牙利*/ #include<iostream> #include<cstdio> #include<cstring> #defi ...

  8. PHP 通过随机数获得ASCII 值返回字符。

    for($i=0;$i<20;$i++){ $Matrix .= chr(65 + rand(0,26)); }

  9. Canvas模糊化处理图片、毛玻璃处理图片之stackblur.js

    Canvas实现毛玻璃效果解决方式1:使用stackblur.js 在Android系统中实现图片的毛玻璃效果比较好用的类库是:Android StackBlur 官方Git地址:https://gi ...

  10. HTML5 WebAudioAPI-实例(二)

    简单播放实例1: var url='../content/audio/海阔天空.mp3'; if (!window.AudioContext) { alert('您的浏览器不支持AudioContex ...