POJ 2828 poj 2828 Buy Tickets 【树状数组,已知前n项和为K,返回n值】

题目链接:http://poj.org/problem?id=2828

在一个队列中,一个人想要插队,告诉你每个新来的人会插在i个人后面,求出最后的队列。

如果我们用模拟的话,那么时间复杂度肯定是超了;想想,如果我们逆序,那么最后来的人的位置一定是固定的,这样的话,我们将问题转化成逆序扫描给出数据,插在i个人后面这个数据就变成了在这个人前面需要留出多少个空位。如此我们只需要用树状数组记录前n项总共有多少个空位,每扫描一个数据,就找出能使得他前面正好有i个空位。

这题用树状数组或者线段树都可以,今天写了树状数组的。开始不知道有那个已知前n项和为K,返回n值的函数,就用二分查找了那个值,差点就T了,后来改用那个函数,节省了差不多一秒种。

代码(二分):

#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <cmath>
#define N 200020
using namespace std;
int n;
int a[N];
int c[N];
int ans[N];
struct fuck
{
    int x,num;
}peo[N];
int add(int x,int a)
{
    for(int i=x;i<=n;i+=(i&(-i)))
    {
        c[i]+=a;
    }
}
int getsum(int i)
{
    int cnt=0;
    while(i>=1)
    {
        cnt+=c[i];
        i-=(i&(-i));
    }
    return cnt;
}
int main()
{
    while(~scanf("%d",&n))
    {
        for(int i=n;i>=1;i--)
        {
            scanf("%d%d",&peo[i].x,&peo[i].num);
            add(i,1);
        }
        for(int i=1;i<=n;i++)
        {
            int temp=peo[i].x;
            int lb=0,ub=n+1;
            while(ub-lb>1)
            {
                int mid=(ub+lb)/2;
                if(getsum(mid)>temp)
                    ub=mid;
                else
                    lb=mid;
            }
            add(ub,-1);
            ans[ub]=peo[i].num;
        }
        for(int i=1;i<=n;i++)
            printf("%d%c",ans[i]," \n"[i==n]);
    }
    return 0;
}


使用那个函数的代码如下:

#include <iostream>
#include <cstdio>
#include <algorithm>
#include <cstring>
#include <cmath>
#define N 200020
using namespace std;
int n;
int a[N];
int c[N];
int ans[N];
struct fuck
{
    int x,num;
}peo[N];
int add(int x,int a)
{
    for(int i=x;i<=n;i+=(i&(-i)))
    {
        c[i]+=a;
    }
}
int getsum(int i)
{
    int cnt=0;
    while(i>=1)
    {
        cnt+=c[i];
        i-=(i&(-i));
    }
    return cnt;
}
int getK(int K)
{
    int ans = 0,cnt=0;
    for(int i=18;i>=0;i--)//i>=0
    {
        ans+=(1<<i);
        if(ans>=n||cnt+c[ans]>=K) ans-=(1<<i);
        else cnt+=c[ans];
    }
    return ans+1;
}
int main()
{
    while(~scanf("%d",&n))
    {
        for(int i=1;i<=n;i++)
        {
            scanf("%d%d",&peo[i].x,&peo[i].num);
            add(i,1);
        }
        for(int i=n;i>=1;i--)
        {
            int temp=peo[i].x;
//            int lb=0,ub=n+1;
//            while(ub-lb>1)
//            {
//                int mid=(ub+lb)/2;
//                if(getsum(mid)>temp)
//                    ub=mid;
//                else
//                    lb=mid;
//            }
            int ub=getK(temp+1);
            ans[ub]=peo[i].num;
            add(ub,-1);
        }
        for(int i=1;i<=n;i++)
            printf("%d%c",ans[i]," \n"[i==n]);
    }
    return 0;
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值