HDU1542 【线段树+扫描线】

#include<iostream>
#include<cstdio>
#include<algorithm>
using namespace std;
#define L(k) k<<1
#define R(k) k<<1|1
const int NO=205;
struct X
{
    int l,r;//左右短点对应的线段
    int s;//覆盖次数 全相同则不为-1  负责为-1
}p[NO<<2];
struct LINE
{
    double l,r;
    double h;
    int s;//覆盖次数
}line[NO<<1];
int n,m;
double x[NO<<1]={-1};
void creat(int k,int l,int r)
{
    p[k].l=l;
    p[k].r=r;
    p[k].s=0;
    if(l==r)
        return;
    int mid=(l+r)>>1;
    creat(L(k),l,mid);
    creat(R(k),mid+1,r);
}
double search_(int k)
{
    if(p[k].s!=-1)
        return (x[p[k].r+1]-x[p[k].l])*(p[k].s!=0);
    return search_(L(k))+search_(R(k));
}
void update(int k,double l,double r,int s)
{
    if(p[k].s!=-1&&x[p[k].l]==l&&x[p[k].r+1]==r)
    {
        p[k].s+=s;
        return;
    }
    if(p[k].s!=-1)
        p[L(k)].s=p[R(k)].s=p[k].s;
    int mid=(p[k].l+p[k].r)>>1;
    if(r<=x[mid+1])
        update(L(k),l,r,s);
    else if(x[mid+1]<=l)
        update(R(k),l,r,s);
    else
    {
        update(L(k),l,x[mid+1],s);
        update(R(k),x[mid+1],r,s);
    }
    p[k].s=p[L(k)].s==p[R(k)].s?p[L(k)].s:-1;
}
bool H(const LINE &a,const LINE &b){return a.h<b.h;}
int main()
{
    freopen("1.txt","r",stdin);
    int cas=1;
    while(scanf("%d",&m),m)
    {
        for(int i=1;i<=m;i++)
        {
            scanf("%lf%lf%lf%lf",&x[i],&line[i].h,&x[m+i],&line[m+i].h);
            line[i].l=line[m+i].l=x[i];
            line[i].r=line[m+i].r=x[m+i];
            line[i].s=1;
            line[m+i].s=-1;
        }
        sort(x+1,x+1+2*m);
        sort(line+1,line+1+2*m,H);
        n=0;HD
        for(int i=1;i<=2*m;i++)
            if(x[i]!=x[i-1])
                x[++n]=x[i];
        creat(1,1,n-1);
        double ans=0;
        line[0].h=line[1].h;
        for(int i=1;i<=2*m;i++)
        {
            ans+=(line[i].h-line[i-1].h)*search_(1);
            update(1,line[i].l,line[i].r,line[i].s);
        }
        printf("Test case #%d\nTotal explored area: %.2lf\n\n",cas++,ans);
    }
    return 0;
}

时间: 2024-10-12 11:50:48

HDU1542 【线段树+扫描线】的相关文章

hdu1542 线段树+扫描线+离散化

只想说题目给的欲实际不服     还是这类型的水题吧   建议看之前我写的那个 #include<stdio.h> #include<string.h> #include<algorithm> #include<iostream> using namespace std; #define LL(x) (x<<1) #define RR(x) ((x<<1)|1) int n; struct node { double y; int x

hdu1542 Atlantis (线段树+扫描线+离散化)

Atlantis Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submission(s): 9032    Accepted Submission(s): 3873 Problem Description There are several ancient Greek texts that contain descriptions of the fabled i

hdu 1255 覆盖的面积 线段树扫描线求重叠面积

稀里糊涂打完了没想到1A了,心情还是很舒畅的,c和c++的四舍五入还是四舍六入遇积进位遇偶舍,我感觉很混乱啊,这道题我输出的答案是7.62,也就是遇偶舍了,可是我就手动处理一下进位问题,发现0.005 系统自动进位0.01了,尼玛啊,我一下子就混乱了,不是遇偶舍么,0也是偶数啊,怎么就进位了呢.最后我就不手动处理进位问题了,直接0.2lf%,虽然我输出的结果是7.62,但是提交也过了 这道题和poj1151,hdu1542差不多,扫描线详细讲解http://blog.csdn.net/young

hdu 1542 Atlantis (线段树+扫描线)

Atlantis Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)Total Submission(s): 18559    Accepted Submission(s): 7523 Problem Description There are several ancient Greek texts that contain descriptions of the fabled i

【BZOJ】1382: [Baltic2001]Mars Maps (线段树+扫描线)

1382: [Baltic2001]Mars Maps Time Limit: 5 Sec  Memory Limit: 64 MB Description 给出N个矩形,N<=10000.其坐标不超过10^9.求其面积并 Input 先给出一个数字N,代表有N个矩形. 接下来N行,每行四个数,代表矩形的坐标. Output 输出面积并 Sample Input 2 10 10 20 20 15 15 25 30 Sample Output 225 本以为是傻逼题,没想到不容易啊- 线段树+扫描

BZOJ 4059 Cerc2012 Non-boring sequences 线段树+扫描线

题目大意:定义一个序列为[不无聊的]当且仅当这个序列的任意一个区间都存在一个数只出现过一次,给定一个序列,要求判断这个序列是否是[不无聊的] 定义lasti表示第i个元素上一次出现的位置(第一次出现则为0),nexti表示第i个元素下一次出现的位置(最后一次出现则为n+1),那么这个元素能成为某个区间仅出现一次的数,当且仅当这个区间的左端点在[lasti+1,i]之间,右端点在[i,nexti?1]之间 我们可以将区间的左右端点放在二维平面上,那么一个元素产生的贡献是一个矩形,我们要确定的是所有

HDU 4419 Colourful Rectangle --离散化+线段树扫描线

题意: 有三种颜色的矩形n个,不同颜色的矩形重叠会生成不同的颜色,总共有R,G,B,RG,RB,GB,RGB 7种颜色,问7种颜色每种颜色的面积. 解法: 很容易想到线段树扫描线求矩形面积并,但是如何维护每种颜色的长度着实让我伤透了脑筋.后来看了一位朋友的题解,才幡然醒悟. 开始想到了用二进制表示颜色,R用001表示,G用010表示,B用100表示.那么就可以用十进制1~7表示7种不同颜色了. 维护 cov[rt][1~3] 表示此区间内3种原色各有多少个, Len[rt][i]表示每种颜色的长

sgu316Kalevich Strikes Back(线段树+扫描线)

做法:总体想法是求出一个矩形的面积以及它所包含的矩形,然后用自己的面积减掉所包含的.主要问题是怎样求解它所包含的矩形. 因为是没有相交点的,可以利用扫描线的方法去做,类似染色,当前段如果是x色,也就是第x个矩形,那么再把他染成y颜色时,说明x包含y,而当扫到y的上边时,这一段又恢复到x色.标记一下被包含的矩形,记录所包含的矩形. 因为会有恢复染色操作,up需要时时更新,左儿子和右儿子一样颜色时就可以合并为一段. 1 ; 2 } 3 void down(int w) 4 { 5 if(s[w]!=

hdu1255 覆盖的面积 线段树-扫描线

矩形面积并 线段树-扫描线裸题 1 #include<stdio.h> 2 #include<string.h> 3 #include<algorithm> 4 #include<math.h> 5 using namespace std; 6 const int maxm=2e3+5; 7 const double eps=1e-5; 8 9 struct seg{ 10 double x,y1,y2; 11 int c; 12 bool operator

线段树+扫描线 HDOJ 5091 Beam Cannon

题目传送门 1 /* 2 题意:给出若干个点的坐标,用一个矩形去覆盖,问最多能覆盖几个点 3 线段树+扫描线:思路是先建一棵以[y, y + h]的树,左右儿子[x, x + w] 4 以这棵树为范围,从左到右扫描,更新点数,x的+1, x+w的-1(超过矩形范围) 5 ans = 每次更新时所覆盖点数的最大值 6 */ 7 #include <cstdio> 8 #include <algorithm> 9 #include <iostream> 10 #includ