对于每个祭坛,算出每条线段阻碍它的角度区间,然后排序求并看看是否有空位即可,时间复杂度$O(n^2\log n)$。

这题在Main上官方时限是0.2S,因此需要几个常数优化:

$1.$为了避免用atan2(y,x)算角度,改成算斜率,所以需要分$4$个方向讨论。

$2.$对区间排序时,不要直接对结构体排序,而是对其指针排序。

$3.$在计算某个祭坛时,除了它本身需要特别计算之外,其它祭坛可以直接看成没有缺口的矩形,可以减少$n$个事件。

$4.$在计算的过程中,不断收缩左右边界,当左右边界重合时即可直接判定为不可能。

#include<cstdio>
#include<algorithm>
using namespace std;
const int N=1005,BUF=50000;
const double eps=1e-9,inf=1e9;
int n,m,cnt,i,X,k,ce,flag;char dir[N],Buf[BUF],*buf=Buf;double L,R;
inline void read(int&a){for(a=0;*buf<48;buf++);while(*buf>47)a=a*10+*buf++-48;}
inline void read(char&a){while(*buf<'A')buf++;a=*buf++;}
struct P{int x,y;P(){}P(int _x,int _y){x=_x,y=_y;}}a[N];
struct seg{
P x,y;
seg(){}
seg(P _x,P _y){
x=_x,y=_y;
if(x.x>y.x||x.y>y.y)swap(x,y);
}
}b[N*5],c[N*4],f[N];
struct E{
double l,r;
void set(double _l,double _r){
l=_l,r=_r;
if(l<L+eps)L=max(L,r);
if(r>R-eps)R=min(R,l);
if(L+eps>R)k=1;
}
}e[N*4],*q[N*4];
inline bool cmp(E*x,E*y){return x->l<y->l;}
inline bool cal(int x,int y,double&t){
if(x<=0)return 0;
t=1.0*y/x;
return 1;
}
inline double fix(double x){return x>0?inf:-inf;}
void initS(int x,int y){
cal(-f[X].x.y+y,f[X].x.x-x,L);
cal(-f[X].y.y+y,f[X].y.x-x,R);
if(L>R)swap(L,R);
for(int i=1;i<=m;i++)if((i+4)/5==X){
seg&j=b[i];double l,r;
if(!cal(-j.x.y+y,j.x.x-x,l))continue;
if(!cal(-j.y.y+y,j.y.x-x,r))r=fix(l);
if(l>r)swap(l,r);
if(r<L-eps||l>R+eps)continue;
e[ce++].set(l,r);
}
for(int i=1;i<=cnt&&!k;i++)if((i+3)/4!=X){
seg&j=c[i];double l,r;
if(!cal(-j.x.y+y,j.x.x-x,l))continue;
if(!cal(-j.y.y+y,j.y.x-x,r))r=fix(l);
if(l>r)swap(l,r);
if(r<L-eps||l>R+eps)continue;
e[ce++].set(l,r);
}
}
void initN(int x,int y){
cal(f[X].y.y-y,f[X].y.x-x,L);
cal(f[X].x.y-y,f[X].x.x-x,R);
if(L>R)swap(L,R);
for(int i=1;i<=m;i++)if((i+4)/5==X){
seg&j=b[i];double l,r;
if(!cal(j.y.y-y,j.y.x-x,r))continue;
if(!cal(j.x.y-y,j.x.x-x,l))l=fix(r);
if(l>r)swap(l,r);
if(r<L-eps||l>R+eps)continue;
e[ce++].set(l,r);
}
for(int i=1;i<=cnt&&!k;i++)if((i+3)/4!=X){
seg&j=c[i];double l,r;
if(!cal(j.y.y-y,j.y.x-x,r))continue;
if(!cal(j.x.y-y,j.x.x-x,l))l=fix(r);
if(l>r)swap(l,r);
if(r<L-eps||l>R+eps)continue;
e[ce++].set(l,r);
}
}
void initW(int x,int y){
cal(-f[X].x.x+x,f[X].x.y-y,L);
cal(-f[X].y.x+x,f[X].y.y-y,R);
if(L>R)swap(L,R);
for(int i=1;i<=m;i++)if((i+4)/5==X){
seg&j=b[i];double l,r;
if(!cal(-j.x.x+x,j.x.y-y,l))continue;
if(!cal(-j.y.x+x,j.y.y-y,r))r=fix(l);
if(l>r)swap(l,r);
if(r<L-eps||l>R+eps)continue;
e[ce++].set(l,r);
}
for(int i=1;i<=cnt&&!k;i++)if((i+3)/4!=X){
seg&j=c[i];double l,r;
if(!cal(-j.x.x+x,j.x.y-y,l))continue;
if(!cal(-j.y.x+x,j.y.y-y,r))r=fix(l);
if(l>r)swap(l,r);
if(r<L-eps||l>R+eps)continue;
e[ce++].set(l,r);
}
}
void initE(int x,int y){
cal(f[X].y.x-x,f[X].y.y-y,L);
cal(f[X].x.x-x,f[X].x.y-y,R);
if(L>R)swap(L,R);
for(int i=1;i<=m;i++)if((i+4)/5==X){
seg&j=b[i];double l,r;
if(!cal(j.y.x-x,j.y.y-y,r))continue;
if(!cal(j.x.x-x,j.x.y-y,l))l=fix(r);
if(l>r)swap(l,r);
if(r<L-eps||l>R+eps)continue;
e[ce++].set(l,r);
}
for(int i=1;i<=cnt&&!k;i++)if((i+3)/4!=X){
seg&j=c[i];double l,r;
if(!cal(j.y.x-x,j.y.y-y,r))continue;
if(!cal(j.x.x-x,j.x.y-y,l))l=fix(r);
if(l>r)swap(l,r);
if(r<L-eps||l>R+eps)continue;
e[ce++].set(l,r);
}
}
bool check(){
if(k)return 0;
for(int i=0;i<ce;i++)q[i]=e+i;
sort(q,q+ce,cmp);
for(int i=0;i<ce;i++){
if(q[i]->l>L+eps)return 1;
if(L<q[i]->r)L=q[i]->r;
}
return 0;
}
int main(){
fread(Buf,1,BUF,stdin);read(n);
for(i=1;i<=n;i++){
int x1,y1,x2,y2,o;char d;
read(x1),read(y1),read(x2),read(y2),read(d);
x1<<=2,y1<<=2,x2<<=2,y2<<=2;
if(x1>x2)swap(x1,x2);
if(y1>y2)swap(y1,y2);
a[i]=P((x1+x2)/2,(y1+y2)/2);
dir[i]=d;
if(d!='S')b[++m]=seg(P(x1,y1),P(x2,y1));
else{
o=(x1+x2)/2;
b[++m]=seg(P(x1,y1),P((x1+o)/2,y1));
b[++m]=seg(P(x2,y1),P((x2+o)/2,y1));
f[i]=seg(P((x1+o)/2,y1),P((x2+o)/2,y1));
}
if(d!='N')b[++m]=seg(P(x1,y2),P(x2,y2));
else{
o=(x1+x2)/2;
b[++m]=seg(P(x1,y2),P((x1+o)/2,y2));
b[++m]=seg(P(x2,y2),P((x2+o)/2,y2));
f[i]=seg(P((x1+o)/2,y2),P((x2+o)/2,y2));
}
if(d!='W')b[++m]=seg(P(x1,y1),P(x1,y2));
else{
o=(y1+y2)/2;
b[++m]=seg(P(x1,y1),P(x1,(y1+o)/2));
b[++m]=seg(P(x1,y2),P(x1,(y2+o)/2));
f[i]=seg(P(x1,(y1+o)/2),P(x1,(y2+o)/2));
}
if(d!='E')b[++m]=seg(P(x2,y1),P(x2,y2));
else{
o=(y1+y2)/2;
b[++m]=seg(P(x2,y1),P(x2,(y1+o)/2));
b[++m]=seg(P(x2,y2),P(x2,(y2+o)/2));
f[i]=seg(P(x2,(y1+o)/2),P(x2,(y2+o)/2));
}
c[++cnt]=seg(P(x1,y1),P(x2,y1));
c[++cnt]=seg(P(x1,y2),P(x2,y2));
c[++cnt]=seg(P(x1,y1),P(x1,y2));
c[++cnt]=seg(P(x2,y1),P(x2,y2));
}
for(X=1;X<=n;X++){
ce=k=0;
if(dir[X]=='S')initS(a[X].x,a[X].y);
if(dir[X]=='N')initN(a[X].x,a[X].y);
if(dir[X]=='W')initW(a[X].x,a[X].y);
if(dir[X]=='E')initE(a[X].x,a[X].y);
if(check())flag=1,printf("%d\n",X);
}
if(!flag)puts("BRAK");
return 0;
}

  

BZOJ2934 : [Poi1999]祭坛问题的更多相关文章

  1. bzoj AC倒序

    Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...

  2. BZOJ 2933([Poi1999]地图-区间Dp)

    2933: [Poi1999]地图 Time Limit: 1 Sec   Memory Limit: 128 MB Submit: 7   Solved: 7 [ Submit][ Status] ...

  3. 2929: [Poi1999]洞穴攀行

    2929: [Poi1999]洞穴攀行 Time Limit: 1 Sec  Memory Limit: 128 MBSubmit: 80  Solved: 41[Submit][Status][Di ...

  4. BZOJ 2929: [Poi1999]洞穴攀行

    2929: [Poi1999]洞穴攀行 Time Limit: 1 Sec  Memory Limit: 128 MBSubmit: 351  Solved: 195[Submit][Status][ ...

  5. BZOJ2935: [Poi1999]原始生物(欧拉回路)

    2935: [Poi1999]原始生物 Time Limit: 3 Sec  Memory Limit: 128 MBSubmit: 150  Solved: 71[Submit][Status][D ...

  6. bzoj 2929 [Poi1999]洞穴攀行 网络流

    2929: [Poi1999]洞穴攀行 Time Limit: 1 Sec  Memory Limit: 128 MBSubmit: 499  Solved: 295[Submit][Status][ ...

  7. BZOJ2936 Codevs3634 POI1999 积水 【并查集】*

    BZOJ2936 Codevs3634 POI1999 积水 题目描述 有这样一块土地,它可以被划分成N×M个正方形小块,每块面积是一平方英寸,第i行第j列的小块可以表示成P(i,j).这块土地高低不 ...

  8. 【BZOJ2927】[Poi1999]多边形之战 博弈

    [BZOJ2927][Poi1999]多边形之战 Description 多边形之战是一个双人游戏.游戏在一个有n个顶点的凸多边形上进行,这个凸多边形的n-3条对角线将多边形分成n-2个三角形,这n- ...

  9. luoguP3415 祭坛

    https://www.luogu.org/problemnew/show/P3415 考虑二分结界层数,将 n 个点按 x 大小依次加入答案,一行一行的做,用树状数组维护当前这一行中[0, x - ...

随机推荐

  1. iOS系类教程之用instruments来检验你的app

    比较了好多关于instruments 还是发现老外写的比较牛逼.于是果断翻译过来.有能力的的可以去看英文原版,鼓励大家看原版资料远离二手教程.这里是原文   入门 为了节省大家的时间,提供一个演示的D ...

  2. java.util.ConcurrentModificationException

    遍历 List 的时候将 item remove 掉会抛出此异常

  3. LeetCode : 223. Rectangle Area

    aaarticlea/png;base64,iVBORw0KGgoAAAANSUhEUgAABRQAAAQ0CAYAAAAPPZBqAAAMFGlDQ1BJQ0MgUHJvZmlsZQAASImVlw

  4. C#学习笔记-----C#枚举中的位运算权限分配

    一.基础知识 什么是位运算? 用二进制来计算,1&2:这就是位运算,其实它是将0001与0010做位预算   得到的结果是 0011,也就是3  2.位预算有多少种?(我们就将几种我们权限中会 ...

  5. mysql TIME_WAIT

    http://www.th7.cn/system/lin/201404/53762.shtml http://kerry.blog.51cto.com/172631/105233/ http://ww ...

  6. 【JAVA之泛型】

    一.引例. 1.引例. 假设现在有一个ArrayList的容器,如果不使用泛型约束,则可以向容器中加入各种类型的对象,但是如果取出来的时候只是用一种类型的转换则肯定会抛出ClassCastExcept ...

  7. 在SQL里如何写条件逻辑?

    主要涉及CASE,WHEN之类.. 不同的服务器上实现if...else...是不一样的. 建议用CASE ,WHEN,因为它们是SQL国标呢. mysql> SELECT -> SUM( ...

  8. [Win32命令行] 更改提示符字符串(PS1)

    当进入的目录比较深时, cmd的提示符几乎会占据整行, 很烦, 于是Google之... 参考: A better PROMPT for CMD.EXE ... 更改方式:        1. pro ...

  9. .NET Expression Tree

    Expression Tree 第一个简单的例子. [TestMethod] public void GodTest() { Expression<Func<int, int, int&g ...

  10. 使用html5的离线缓存技术

    突然想用html5的离线缓存,但是一直没有成功,在各种群里问发现很多人都没什么经验,最终终于在各种论坛找到解决方案了.下面就简单记录一下相关情况. 一.离线缓存的优点 我们都知道离线缓存主要是用来减少 ...