Fork me on GitHub

POJ 2464 Brownie Points II --树状数组

题意: 有点迷。有一些点,Stan先选择某个点,经过这个点画一条竖线,Ollie选择一个经过这条直接的点画一条横线。Stan选这两条直线分成的左下和右上部分的点,Ollie选左上和右下部分的点。Stan画一条竖线之后,Ollie有很多种选择,在所有选择中,Stan能获得 “分数最小值的最大值” ,而Ollie的选择便是让自己越多越好。问最后Stan最多能得到的分数是多少,以及在这种情况下Ollie能得到的分数有多少种可能。

解法: 因为Stan先选,然后主动权在Ollie手中,Ollie会优先让自己得到更多的分数,然后再考虑让Stan得到的分数最小。然后才能求得Stan得到的“分数最小值的最大值”。

既然是Stan先选,那么我们最好按x从小到大排序,y坐标离散,依次处理,又因为在同一条竖线可能有很多店,所以直到坐标变化时才来同一处理那些横坐标相同的点,Ollie在这些点对应的纵坐标中做选择,使达到上述说的效果。

由于竖线往右移,那么维护两个树状数组,一个是当前竖线右边的点的情况,一个是左边的。然后扫过去,遇到p[i].x!=p[i-1].x时,就可以处理前面的一个或多个横坐标相同的点了,然后按上述说的做就行了。

代码:

#include <iostream>
#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <cmath>
#include <algorithm>
#include <set>
using namespace std;
#define N 200007

struct node{
    int x,y;
}p[N];
int n,maxi;
int L[N],R[N],a[N],b[N];
int lowbit(int x) { return x&-x; }
int cmp1(node ka,node kb) { return ka.x < kb.x; }
int cmp2(node ka,node kb) { return ka.y < kb.y; }

void modify(int *c,int x,int val)
{
    while(x <= maxi)
        c[x] += val, x += lowbit(x);
}

int getsum(int *c,int x)
{
    int res = 0;
    while(x > 0) { res += c[x]; x -= lowbit(x); }
    return res;
}

int main()
{
    int i,j;
    while(scanf("%d",&n)!=EOF && n)
    {
        maxi = 0;
        for(i=1;i<=n;i++)
            scanf("%d%d",&p[i].x,&p[i].y);
        sort(p+1,p+n+1,cmp1);                         //....离散
        for(i=1;i<=n;i++)                           
        {
            if(p[i].x == p[i-1].x) a[i] = a[i-1];
            else a[i] = a[i-1]+1;
        }
        for(i=1;i<=n;i++) p[i].x = a[i];
        sort(p+1,p+n+1,cmp2);
        for(i=1;i<=n;i++)    
        {
            if(p[i].y == p[i-1].y) b[i] = b[i-1];
            else b[i] = b[i-1]+1;
            maxi = max(maxi,b[i]);
        }
        for(i=1;i<=n;i++) p[i].y = b[i];               //离散....
        memset(L,0,sizeof(L));
        memset(R,0,sizeof(R));
        for(i=1;i<=n;i++)
            modify(R,p[i].y,1);
        int Stan = -1,ollie,stan,start = 1;
        sort(p+1,p+n+1,cmp1);
        p[n+1].x = -1,p[n+1].y = -1;
        set<int> Ollie;
        for(i=2;i<=n+1;i++)
        {
            if(p[i].x == p[i-1].x) continue;
            stan = ollie = -1;
            for(j=start;j<i;j++)
                modify(R,p[j].y,-1);
            for(j=start;j<i;j++)
            {
                int pos = p[j].y;
                int STAN = getsum(R,maxi)-getsum(R,pos)+getsum(L,pos-1);   //右上+左下
                int OLLIE = getsum(L,maxi)-getsum(L,pos)+getsum(R,pos-1);  //左上+右下
                if(OLLIE == ollie) stan = min(stan,STAN);     //在保证Ollie取最多的情况下让Stan得分最少
                else if(OLLIE > ollie) stan = STAN, ollie = OLLIE;
            }
            if(stan > Stan) Stan = stan, Ollie.clear(), Ollie.insert(ollie);
            else if(stan == Stan) Ollie.insert(ollie);
            for(j=start;j<i;j++)
                modify(L,p[j].y,1);
            start = i;
        }
        printf("Stan: %d; Ollie:",Stan);
        for(set<int>::iterator it=Ollie.begin();it!=Ollie.end();it++)
            printf(" %d",*it);
        printf(";\n");
    }
    return 0;
}
View Code

 

posted @ 2014-10-29 21:46  whatbeg  阅读(412)  评论(0编辑  收藏  举报