1. 程式人生 > >POJ 2528 Mayor's posters 線段樹(成段更新+離散化)

POJ 2528 Mayor's posters 線段樹(成段更新+離散化)

題意:

給出N個海報,每個海報有一個長度區間(a,b).按順序貼在牆上。

問最後可以看到幾張海報。

思路:

一想到的就是線段樹,對每個區間進行染色,最後查詢一共有多少種顏色。

第一次寫玩沒看資料大小。MLE了。。仔細一看,海報長度1QW。

然後寫了個離散化的,300MS+。

又去看了別人的離散化。。神多了。。60MS。。

優化後的離散

#include <iostream>
#include <cstdio>
#include <algorithm>
#include <string>
#include <cmath>
#include <cstring>
#include <queue>
#include <set>
#include <vector>
#include <stack>
#include <map>
#include <iomanip>
#define PI acos(-1.0)
#define Max 20005
#define inf 1<<28
#define LL(x) (x<<1)
#define RR(x)(x<<1|1)
using namespace std;

struct kdq
{
    int l,r,flag;
} tree[Max*4];

struct kdq1
{
    int num,id;
} poster[Max];

int aa[Max][2];
void build_tree(int l,int r,int u)
{
    tree[u].l=l;
    tree[u].r=r;
    tree[u].flag=0;
    if(l==r)return ;
    int mid=(l+r)>>1;
    build_tree(l,mid,LL(u));
    build_tree(mid+1,r,RR(u));
}

void update(int l,int r,int u,int i)
{
    if(l>tree[u].r||r<tree[u].l)return ;
    if(l==tree[u].l&&r==tree[u].r)
    {
        tree[u].flag=i;
        return ;
    }
    if(tree[u].flag > 0 && tree[u].flag != i)
    {
        tree[LL(u)].flag = tree[u].flag;
        tree[RR(u)].flag= tree[u].flag;
        tree[u].flag = 0;
    }
    int mid=(tree[u].l+tree[u].r)>>1;
    if(r<=mid)
        update(l,r,LL(u),i);
    else if(l>mid)
        update(l,r,RR(u),i);
    else
    {
        update(l,mid,LL(u),i);
        update(mid+1,r,RR(u),i);
    }
    if(tree[LL(u)].flag==tree[RR(u)].flag)
        tree[u].flag=tree[LL(u)].flag;
    else
        tree[u].flag=0;
}

int ans;
bool visit1[20005];

void query(int l,int r,int u)
{
    if(tree[u].flag&&!visit1[tree[u].flag])
        return ;
    if(tree[u].flag)
    {
        ans+=visit1[tree[u].flag];
        visit1[tree[u].flag]=0;
        return ;
    }
    int mid=(l+r)>>1;
    query(l,mid,LL(u));
    query(mid+1,r,RR(u));
}
bool cmp(kdq1 a,kdq1 b)
{
    return a.num<b.num;
}
int main()
{
    int i,j,k,l,n,m,T;
    scanf("%d",&T);
    int a,b;
    while(T--)
    {
        memset(visit1,1,sizeof(visit1));
        scanf("%d",&n);
        for(i=0; i<n; i++)
        {
            scanf("%d%d",&aa[i][0],&aa[i][1]);
            poster[2*i].num=aa[i][0];
            poster[2*i].id=-(i+1);
            poster[2*i+1].num=aa[i][1];
            poster[2*i+1].id=i+1;
        }
        sort(poster,poster+2*n,cmp);
        int temp=poster[0].num;
        int tp=1;
        for(i=0; i<2*n; i++)
        {
            if(temp!=poster[i].num)
            {
                tp++;
                temp=poster[i].num;
            }
            if(poster[i].id<0)
            {
                aa[-poster[i].id-1][0]=tp;
            }
            else
            {
                aa[poster[i].id-1][1]=tp;
            }
        }
        build_tree(1,tp,1);
        for(i=0; i<n; i++)
            update(aa[i][0],aa[i][1],1,i+1);
        ans=0;
        query(1,tp,1);
        printf("%d\n",ans);
    }
    return 0;
}
第一次寫的離散
#include <iostream>
#include <cstdio>
#include <algorithm>
#include <string>
#include <cmath>
#include <cstring>
#include <queue>
#include <set>
#include <vector>
#include <stack>
#include <map>
#include <iomanip>
#define PI acos(-1.0)
#define Max 20005
#define inf 1<<28
#define LL(x) (x<<1)
#define RR(x)(x<<1|1)
using namespace std;

struct kdq
{
    int l,r,flag;
} tree[Max*4];

int aa[Max],bb[Max];

void build_tree(int l,int r,int u)
{
    tree[u].l=l;
    tree[u].r=r;
    tree[u].flag=0;
    if(l==r)return ;
    int mid=(l+r)>>1;
    build_tree(l,mid,LL(u));
    build_tree(mid+1,r,RR(u));
}

void update(int l,int r,int u,int i)
{
    if(l>tree[u].r||r<tree[u].l)return ;
    if(l==tree[u].l&&r==tree[u].r)
    {
        tree[u].flag=i;
        return ;
    }
    if(tree[u].flag > 0 && tree[u].flag != i)
    {
        tree[LL(u)].flag = tree[u].flag;
        tree[RR(u)].flag= tree[u].flag;
        tree[u].flag = 0;
    }
    int mid=(tree[u].l+tree[u].r)>>1;
    if(r<=mid)
        update(l,r,LL(u),i);
    else if(l>mid)
        update(l,r,RR(u),i);
    else
    {
        update(l,mid,LL(u),i);
        update(mid+1,r,RR(u),i);
    }
    if(tree[LL(u)].flag==tree[RR(u)].flag)
        tree[u].flag=tree[LL(u)].flag;
    else
        tree[u].flag=0;
}

int ans;
int yinshe[20005];
int visit[10000005];
bool visit1[20005];

void query(int l,int r,int u)
{
    if(tree[u].flag&&!visit1[tree[u].flag])
        return ;
    if(tree[u].flag)
    {
        ans+=visit1[tree[u].flag];
        visit1[tree[u].flag]=0;
        return ;
    }
    int mid=(l+r)>>1;
    query(l,mid,LL(u));
    query(mid+1,r,RR(u));
}

int main()
{
    int i,j,k,l,n,m,T;
    scanf("%d",&T);
    int a,b;
    while(T--)
    {
        memset(visit1,1,sizeof(visit1));
        memset(visit,0,sizeof(visit));
        memset(yinshe,0,sizeof(yinshe));
        scanf("%d",&n);
        int num=1;
         for(i=1; i<=n; i++)
         {
             scanf("%d%d",&aa[i],&bb[i]);
             if(!visit[aa[i]])
             {
                 yinshe[num]=aa[i];
                 visit[aa[i]]=1;
                 num++;
             }
             if(!visit[bb[i]])
             {
                 yinshe[num]=bb[i];
                 visit[bb[i]]=1;
                 num++;
             }
         }
         sort(yinshe+1,yinshe+num);
         for(i=1; i<num; i++)
             visit[yinshe[i]]=i;
         build_tree(1,num-1,1);
        for(i=1;i<=n;i++)
        update(visit[aa[i]],visit[bb[i]],1,i);
        ans=0;
        query(1,num-1,1);
        printf("%d\n",ans);
    }
    return 0;
}

顯然優化後的神多了。。。繼續。

相關推薦

POJ 2528 Mayor's posters 線段更新+離散

題意: 給出N個海報,每個海報有一個長度區間(a,b).按順序貼在牆上。 問最後可以看到幾張海報。 思路: 一想到的就是線段樹,對每個區間進行染色,最後查詢一共有多少種顏色。 第一次寫玩沒看資料大小。MLE了。。仔細一看,海報長度1QW。 然後寫了個離散化的,300MS+。

POJ 2528 Mayor's posters (線段+離散 替換)

題目大意: 就是在一段區間上貼海報, 後來的區間會把前面來的區間覆蓋掉, 為貼完海報後能看到幾張海報(只看到一部分也算看到) 大致思路: 就是區間替換更新, 標記一下當前區間的所有的海報是否一致, 用懶惰標記標記一下當前區間的類別 聽說資料比較水...布吉島嚴格的資料下這

POJ 2528 Mayor's posters 線段更新+離散

題意:很多張海報貼在牆上 求可以看到幾張海報 看那兩張圖就行了 第一張俯檢視 思路:最多2W個不同的數 離散化一下 然後成段更新 a[rt] = i代表這個區間是第i張報紙 更新玩之後一次query cover[i]=1代表可以看到第i張報紙 #include <c

hdu1828線段兩次掃描+離散

for 掃描 方向 body 理解 upd sort 有序 ide 題目鏈接 求周長並,思路和註意事項與求面積並類似,我用了最簡單的思路,即x軸做一次線段樹,y軸做一次線段樹。 還有一種方法,只做一次線段樹,在做線段樹的同時求另一個方向的長度,大概的想法我知道,不過在左右區

POJ 2528 Mayor's posters線段+離散

Mayor's posters Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 60691 Accepted: 17565 Description The citizens of Byteto

POJ 2528 Mayor's posters [ 離散 + 線段 ]

題目連結: 題意概括: 依次貼上 n 張海報,每張海報會覆蓋一個區間。後貼上的海報會覆蓋前貼的海報,問最後可以看見幾張海報 這裡的區間不是拿兩端點來維護的,是直接按單位最小區間來編號。如 [3, 5] 區間是由編號為 3、4、5的區間組成的 資料範圍:

線段 + 離散 + 詳細註釋】北大 poj 2528 Mayor's posters

/* THE PROGRAM IS MADE BY PYY */ /*----------------------------------------------------------------------------// Copyright (c) 2011 panyanyany All rig

poj 2528 Mayor's posters(離散+線段)

題意: 市長競選,每個市長都往牆上貼海報,海報之間彼此可以覆蓋,給出貼上順序和每個海報的起點和終點,問最後又多少海報時可見的。 剛接觸離散化,先寫一寫本人的理解: 如果原資料太大,建樹會超記憶體。因此可以先對這些數排序,然後將它們對映成排序後的序號。比如在該題中,[1,

poj 2528 Mayor's posters狀陣列+離散

Mayor's posters Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 63793 Accepted: 18415 Description The citizens of Byteto

POJ訓練計劃2528_Mayor&#39;s posters(線段/更新+離散)

rule you mon bsp 左右 for rst scribe i+1 解題報告 id=2528">地址傳送門 題意: 一些海報,覆蓋上去後還能看到幾張。 思路: 第一道離散化的題。 離散化的意思就是區間壓縮然後映射。 給你這麽幾個區間[1,30

POJ 2528 Mayor's posters 區間染色問題

題目意思 每個案例一個數字n,接下來n行表示張貼海報的起點與終點,求有多少海報沒有被完全遮住,經典區間染色問題。本題資料量比較大,所以需要將起點和終點離散化處理。 Sample Input 1 5 1 4 2 6 8 10

Mayor's posters 線段+離散

%%% //感覺博主對於離散的解釋很好,很感謝。 %%% //博主解釋了一下本題離散的特殊性,感謝 如三張海報為:1~10 1~4 6~10 離散化時 X[ 1 ] = 1, X[ 2 ] = 4, X[ 3 ] = 6, X[ 4 ] = 10 第一張海報時

poj2528--Mayor's posters(線段+離散)

The citizens of Bytetown, AB, could not stand that the candidates in the mayoral election campaign have been placing their electoral poste

poj 2528 Mayor's posters【區間離散

文章目錄題目連結: 題目連結: http://poj.org/problem?id=2528 題意:就是有 N 種海報,每種海報有個長度[L,R],後來的海報會覆蓋前面來的海報,問最後從最上面看,能看得到幾種海報 input: 3 3 5 6 4 5 6 8

Mayor's posters (線段+離散+加思維)

The citizens of Bytetown, AB, could not stand that the candidates in the mayoral election campaign have been placing their electoral poste

【POJ2528】Mayor's Posters-線段+離散

(本人本題完成於2016-7-23) 題目大意:有一面很長(10000000節)的牆,要在上面貼N張海報,對於第i張海報,它將會覆蓋牆面的第Li~Ri節。剛開始牆面是空的,求最後還有多少張海報是可見的

數算實習 Mayor's posters 線段 離散

Mayor’s posters每個候選人都可以在牆上放置一張海報。 所有海報的高度都與牆的高度相同; 海報的寬度可以是任何整數個單位。 牆被分成10000000段,每段的寬度是一個單位。每張海報必須完全覆蓋連續數量個單位的牆壁。 你的任務是找到所有海報放置完畢後

hdu1698Just a Hook解題報告---線段替換 & 1結點為區間總和

                                          &

zoj 1610 Count the Colors 線段更新染色

Count the ColorsTime Limit:2000MS    Memory Limit:65536KB    64bit IO Format:%lld & %llu Description Painting some colored segmen

zoj 1610 Count the Colors(線段更新染色)

連結: 題目大意: 在一條長度為8000的線段上染色,每次把區間[a,b]染成c顏色。顯然,後面染上去的顏色會覆蓋掉之前的顏色。 求染完之後,每個顏色線上段上有多少個間斷的區間。 分析與總結: 這題是成段更新lazy標記的題,並不算難,但是因為一些原因一直讓我過不去