`
leili
  • 浏览: 180334 次
社区版块
存档分类
最新评论

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;
}

显然优化后的神多了。。。继续。

分享到:
评论

相关推荐

    POJ2528-Mayor's posters【区间映射压缩(离散化)+线段树】

    POJ2528-Mayor's posters 【区间映射压缩(离散化)+线段树】 解题报告+AC代码 http://hi.csdn.net/!s/83XZJU ========&gt; 我的所有POJ解题报告链接: http://blog.csdn.net/lyy289065406/article/details/6642573

    poj-2528 Mayor's posters 测试数据及答案

    【标题】"poj-2528 Mayor's posters 测试数据及答案"涉及的是一个编程竞赛中的问题,这个问题来源于POJ(Programming Online Judge)平台,编号为2528。POJ是一个在线的编程练习系统,它提供了一系列的问题供程序员...

    POJ2528-Mayor's posters 测试数据

    【标题】"POJ2528-Mayor's posters"是一个经典的编程竞赛题目,源自2003年Alberta Collegiate Programming Contest。该比赛每年都会吸引众多编程爱好者参加,旨在锻炼参赛者的算法设计和实现能力。题目“市长的海报...

    线段树入门学习(三)懒操作(兼解POJ1823) JAVA

    在本篇博文中,我们将深入探讨线段树的概念,以及如何实现懒惰更新(Lazy Propagation)策略,同时结合POJ1823问题进行实战应用。懒惰更新是优化线段树性能的关键技巧,它避免了频繁地对每个节点进行更新,从而减少...

    线段树练习POJ 3264

    线段树是一种数据结构,常用于处理区间查询与更新的问题,它能高效地支持动态维护区间最值、求和等问题。在本题“POJ 3264”中,我们可能面临的是一个区间最值或者区间求和的计算任务。线段树的基本思想是将数组或...

    POJ3277.rar_3277 poj_poj3277_多个面积_线段树

    标题中的“POJ3277.rar_3277 poj_poj3277_多个面积_线段树”是指一个与编程竞赛相关的压缩文件,特别提到了POJ(Problemset Online Judge)的第3277号问题。POJ是一个著名的在线编程竞赛平台,参与者需要解决各种...

    线段树入门学习(二)(兼解POJ3468) JAVA

    在本教程中,我们将深入探讨线段树的基本概念,以及如何使用Java实现线段树来解决实际问题,如POJ3468题目。 首先,理解线段树的核心在于它的分治思想。线段树将一个大区间(通常是一维数组)分成多个小区间,每个...

    poj 2352 stars(树状数组,线段树)

    ### POJ 2352 Stars - 树状数组与线段树实现 #### 题目背景 POJ 2352 Stars 是一道经典的算法题目,它提供了使用树状数组或线段树来解决的可能性。这两种数据结构都是为了高效处理区间查询问题而设计的。在这篇...

    poj2823.cpp.tar.gz_线段树

    线段树是一种数据结构,主要用于高效地处理区间(或段)上的查询与更新操作。在题目"poj2823"中,我们看到它被应用于寻找区域间的最大值和最小值。线段树通常用于解决动态规划问题,特别是在数组或序列上执行范围...

    线段树题目

    - **POJ 2528**:这是一道稍微复杂一点的线段覆盖问题,需要使用离散化技术。题目中的输入数据范围可能很大,直接存储会消耗大量的内存空间。因此,首先需要对所有涉及的点进行离散化处理,压缩点的范围,减少所需的...

    线段树例题(唐文斌).pdf

    ### 线段树例题解析 #### Brackets (SPOJ61, BRCKTS) **题目描述:** 此题要求我们维护一个长度为 \(N\) 的小括号序列 \(A\),并支持两种操作: 1. **Replace(i)**:将第 \(i\) 个位置的括号方向反转。 2. **Check...

    POJ 部分题解 解题报告

    10. **POJ 2528 线段树.txt**:这是第三个线段树相关的题目,解题报告可能会深入讲解线段树在不同场景下的应用。 通过阅读这些解题报告,你可以学习到各种高级算法的应用,如动态规划、搜索算法、数据结构(如线段...

    ACMer需要掌握的算法讲解 (2).docx

    * 坐标离散化:POJ3429 * 代码快速写成,精简但不失风格:POJ2525、POJ1684、POJ1421、POJ1048、POJ2050、POJ3306 * 保证正确性和高效性:POJ3434 六、动态规划 * 需要用数据结构优化的动态规划:POJ2754、POJ3378...

    poj题目分类

    * 线段树:例如 poj2528、poj2828、poj2777、poj2886、poj2750。 * 静态二叉检索树:例如 poj2482、poj2352。 * 树状树组:例如 poj1195、poj3321。 * RMQ:例如 poj3264、poj3368。 * 并查集的高级应用:例如 ...

    ACM数据结构之树状数组和线段树

    4. **更新操作**:如果需要对区间内的元素进行更新,只需要更新覆盖该区间的线段树节点即可。 通过以上步骤,可以高效地解决 POJ3264 BalancedLineup 这类问题。 总结来说,线段树和树状数组都是强大的数据结构,...

    二维树状数组学习之二:练习POJ 1195

    二维树状数组(也称为2D BIT,即二维前缀和)是一种在计算机科学和算法设计中用于高效处理数组更新和查询的数据结构。它在处理矩阵或网格状数据时非常有用,尤其对于需要频繁进行区间求和或者更新的场景。在本篇中,...

    poj训练计划.doc

    - 线段树:如`poj2528, poj2828`。 - RMQ(区间查询):如`poj3264, poj3368`。 - 并查集的高级应用:如`poj1703, poj2492`。 - **搜索** - 最优化剪枝和可行性剪枝:如`poj3411, poj1724`。 - **动态规划** ...

    POJ2777 个人代码

    POJ题解 个人写法 线段树每个人都不一样

    poj 2763 housewife Wind

    poj 2763 程序 线段树 LCA 2000MS AC

    acm poj300题分层训练

    poj2528、poj2777等是线段树的题目,poj2482、poj2352等涉及平衡树,poj1195、poj3321则训练了树状数组。 5. **搜索算法的优化**:poj1699、poj3411等训练搜索技巧和剪枝方法,poj3373、poj1691则介绍了记忆化搜索。...

Global site tag (gtag.js) - Google Analytics