题意: 有点迷。有一些点,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 = ;
while(x > ) { res += c[x]; x -= lowbit(x); }
return res;
} int main()
{
int i,j;
while(scanf("%d",&n)!=EOF && n)
{
maxi = ;
for(i=;i<=n;i++)
scanf("%d%d",&p[i].x,&p[i].y);
sort(p+,p+n+,cmp1); //....离散
for(i=;i<=n;i++)
{
if(p[i].x == p[i-].x) a[i] = a[i-];
else a[i] = a[i-]+;
}
for(i=;i<=n;i++) p[i].x = a[i];
sort(p+,p+n+,cmp2);
for(i=;i<=n;i++)
{
if(p[i].y == p[i-].y) b[i] = b[i-];
else b[i] = b[i-]+;
maxi = max(maxi,b[i]);
}
for(i=;i<=n;i++) p[i].y = b[i]; //离散....
memset(L,,sizeof(L));
memset(R,,sizeof(R));
for(i=;i<=n;i++)
modify(R,p[i].y,);
int Stan = -,ollie,stan,start = ;
sort(p+,p+n+,cmp1);
p[n+].x = -,p[n+].y = -;
set<int> Ollie;
for(i=;i<=n+;i++)
{
if(p[i].x == p[i-].x) continue;
stan = ollie = -;
for(j=start;j<i;j++)
modify(R,p[j].y,-);
for(j=start;j<i;j++)
{
int pos = p[j].y;
int STAN = getsum(R,maxi)-getsum(R,pos)+getsum(L,pos-); //右上+左下
int OLLIE = getsum(L,maxi)-getsum(L,pos)+getsum(R,pos-); //左上+右下
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,);
start = i;
}
printf("Stan: %d; Ollie:",Stan);
for(set<int>::iterator it=Ollie.begin();it!=Ollie.end();it++)
printf(" %d",*it);
printf(";\n");
}
return ;
}

POJ 2464 Brownie Points II --树状数组的更多相关文章

  1. hdu 1156 && poj 2464 Brownie Points II (BIT)

    2464 -- Brownie Points II Problem - 1156 hdu分类线段树的题.题意是,给出一堆点的位置,stan和ollie玩游戏,stan通过其中一个点画垂线,ollie通 ...

  2. POJ 2464 Brownie Points II (树状数组,难题)

    题意:在平面直角坐标系中给你N个点,stan和ollie玩一个游戏,首先stan在竖直方向上画一条直线,该直线必须要过其中的某个点,然后ollie在水平方向上画一条直线,该直线的要求是要经过一个sta ...

  3. POJ 2464 Brownie Points II(树状数组)

    一开始还以为对于每根竖线,只要与过了任意一点的横线相交都可以呢,这样枚举两条线就要O(n^2),结果发现自己想多了... 其实是每个点画根竖线和横线就好,对于相同竖线统计(一直不包含线上点)右上左下总 ...

  4. POJ - 2464 Brownie Points II 【树状数组 + 离散化】【好题】

    题目链接 http://poj.org/problem?id=2464 题意 在一个二维坐标系上 给出一些点 Stan 先画一条过一点的水平线 Odd 再画一条 过Stan那条水平线上的任一点的垂直线 ...

  5. UVA 10869 - Brownie Points II(树阵)

    UVA 10869 - Brownie Points II 题目链接 题意:平面上n个点,两个人,第一个人先选一条经过点的垂直x轴的线.然后还有一个人在这条线上穿过的点选一点作垂直该直线的线,然后划分 ...

  6. POJ 2155 Matrix(二维树状数组,绝对具体)

    Matrix Time Limit: 3000MS   Memory Limit: 65536K Total Submissions: 20599   Accepted: 7673 Descripti ...

  7. poj 3321:Apple Tree(树状数组,提高题)

    Apple Tree Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 18623   Accepted: 5629 Descr ...

  8. POJ 2299 Ultra-QuickSort 逆序数 树状数组 归并排序 线段树

    题目链接:http://poj.org/problem?id=2299 求逆序数的经典题,求逆序数可用树状数组,归并排序,线段树求解,本文给出树状数组,归并排序,线段树的解法. 归并排序: #incl ...

  9. poj 3321 Apple Tree(一维树状数组)

    题目:http://poj.org/problem?id=3321 题意: 苹果树上n个分叉,Q是询问,C是改变状态.... 开始的处理比较难,参考了一下大神的思路,构图成邻接表 并 用DFS编号 白 ...

随机推荐

  1. MAC 隐藏文件的显示

    显示 defaults write com.apple.finder AppleShowAllFiles -bool true 隐藏 defaults write com.apple.finder A ...

  2. eclipse easyexplorer openexplorer

    电脑重装了.. eclipse换成了4.4.2,发现原来的easyexplorer不生效了(原来是4.2),搜了下,换成了open explorer即可. 方式一样,都是丢到eclipse/plugi ...

  3. zend framework2学习(一)初步入门

    声明:本人菜鸟一枚,由于项目中需要用到zf2框架进行开发,在此记载学习使用过程中的点点滴滴.才疏学浅,请多指教............. ------------------------------- ...

  4. [译] 第三十天:Play Framework - Java开发者梦寐以求的框架 - 百花宫

    前言 30天挑战的最后一天,我决定学习 Play Framework .我本来想写Sacla,但是研究几个小时后,我发现没法在一天内公正评价Scala,下个月花些时间来了解并分享经验.本文我们先来看看 ...

  5. 个人总结 HTML+CSS

    从大一下学期接触,一直到今年,接触的时间也挺长的了,最近一些认识的盆友和同学说是想学习前端,自己也开始慢慢停下脚步,不再拼命地去学很多框架的东西,回归到基础,慢慢把基础打牢 很多知识碎片一直来不及整理 ...

  6. 百度网盘采集源码 ,直接采集网盘添加cookies功能

    名称:百度网盘采集源码 程序语言:php 数据库:mysql 程序介绍: 1.直接采集百度网盘url 2.前端基于bootstrap 3.搜索考虑到后期上亿数据,是基于coreseek,搜索时间毫秒级 ...

  7. ECMAScript 6学习笔记(一):展开运算符

    同步发布于:https://mingjiezhang.github.io/(转载请说明此出处). JavaScript是ECMAScript的实现和扩展,ES6标准的制定也为JavaScript加入了 ...

  8. SharePoint 2010 文档管理之文档推送

    前言:文档推送功能,不是一个复杂的功能,我们这里简单的应用了Ribbon定制.Js使用.对象模型推送(Server端),下面,我们来简单介绍下文档推送功能吧. 一. 功能设计: 文档推送功能,主要就是 ...

  9. File类的常用方法

    public static void GetFileInfo()    {                File file=new File("e:","two.txt ...

  10. 【转】android应用程序的安装方式与原理

    四种安装方式: 1.系统应用安装――开机时完成,没有安装界面 2.网络下载应用安装――通过market应用完成,没有安装界面 3.ADB工具安装――没有安装界面. 4.第三方应用安装――通过SD卡里的 ...