http://acm.hdu.edu.cn/showproblem.php?pid=1156

在一张二位坐标系中,给定n个点的坐标,玩一个划线游戏(线必须穿过点),Stan先手画一条垂直的线,然后Ollie画一条水平的线(要求要穿过Stan那条线所穿过的某个点)。划分后,左上和右下点数是Ollie 的得分,左下和右上是Stan的得分。求Stan在保证最低得分(即不论Ollie后手怎么划,Stan最少能的的分数)最高,并给出基于符合的先手划法,Ollie后手的各种划线的得分(需要去重),升序输出。

这里可以发现,每次划分等同于枚举以某个点为原点,求一次局面,不同的局面分类到x轴坐标中,每个坐标中的局面集合求最小值,所有x轴坐标分类求最大值即求出了最低最大划法。

#include <iostream>
#include <cstring>
#include <string>
#include <queue>
#include <vector>
#include <set>
#include <stack>
#include <cmath>
#include <cstdio>
#include <map>
#include <algorithm>
using namespace std;
struct node
{
int x, y;
}p[];
bool cmpx(node a, node b)
{
if (a.x == b.x) return a.y < b.y;
return a.x < b.x;
}
const int N = ;
map<int, int> mpx, mpy;
int st[N];
int tre[N];
int ans[N];
int lowbit(int k)
{
return k&-k;
}
int query(int k)
{
int rec = ;
while (k)
{
rec += tre[k];
k -= lowbit(k);
}
return rec;
}
void add(int k)
{
while (k <= N)
{
tre[k]++;
k += lowbit(k);
}
}
bool cmp(node a, int b)
{
return a.x < b;
}
int main() {
cin.sync_with_stdio(false);
int n;
while (cin>>n)
{
if (n == ) break;
mpx.clear(), mpy.clear();
queue<node> wq;
fill(tre, tre + N, );
fill(ans, ans + N, );
for (int i = ; i < n; i++)
cin >> p[i].x>>p[i].y,mpx[p[i].x]++,mpy[p[i].y]++,st[i]=p[i].y;
sort(st, st + n);
int len = unique(st, st + n) - st;
sort(p, p + n, cmpx); for (int i = ; i < n; i++)
{
while (!wq.empty())
{
node ad = wq.front();
if (ad.x < p[i].x) add(upper_bound(st, st + len, ad.y) - st + ),wq.pop();
else break;
}
int py = upper_bound(st, st + len, p[i].y)-st+;
ans[i] += query(py-);
wq.push(p[i]);
}
while (!wq.empty()) wq.pop();
fill(tre, tre + , );
for (int i = n-; i >= ; i--)
{
while (!wq.empty())
{
node ad = wq.front();
if (ad.x > p[i].x) add(len+st-upper_bound(st, st + len, ad.y) + ),wq.pop();
else break;
}
int py = len + st - upper_bound(st, st + len, p[i].y) + ;
ans[i] += query(py-);
wq.push(p[i]);
}
map<int, int> getMi;
for (int i = ; i < n; i++)
{
if (getMi.find(p[i].x) == getMi.end()) getMi[p[i].x] = ans[i];
else getMi[p[i].x] = min(getMi[p[i].x], ans[i]);
}
map<int, int>::iterator mxit = getMi.end();
for (map<int, int>::iterator it = getMi.begin(); it != getMi.end(); it++)
{
if (mxit == getMi.end())
mxit = it;
else if (mxit->second < it->second)
mxit = it;
}
cout << "Stan: " << mxit->second << "; Ollie:";
set<int> rpy;
for (int i = ; i < n; i++)
if (getMi[p[i].x]==ans[i]&&ans[i] == mxit->second)
rpy.insert(n - ans[i] - mpx[p[i].x] - mpy[p[i].y] + ); for (set<int>::iterator it = rpy.begin(); it != rpy.end(); it++)
{
cout << ' ' << *it;
}
cout << ";" << endl;
}
return ;
}

HDOJ-1156 Brownie Points II 线段树/树状数组(模板)的更多相关文章

  1. UVA10869 - Brownie Points II(线段树)

    UVA10869 - Brownie Points II(线段树) 题目链接 题目大意:平面上有n个点,Stan和Ollie在玩游戏,游戏规则是:Stan先画一条竖直的线作为y轴,条件是必需要经过这个 ...

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

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

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

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

  4. HDU 1166 线段树模板&树状数组模板

    HDU1166 上好的线段树模板&&树状数组模板 自己写的第一棵线段树&第一棵树状数组 莫名的兴奋 线段树: #include <cstdio> using nam ...

  5. HDU 1166 敌兵布阵(线段树/树状数组模板题)

    敌兵布阵 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submi ...

  6. luogu3368树状数组模板2

    题目链接:https://www.luogu.org/problemnew/show/P3368 题意:与模板1不同的是这题的操作是树状数组并不在行的区间更新和单点查找,如果按照模板1那样写肯定会T. ...

  7. 树状数组模板(pascal) 洛谷P3374 【模板】树状数组1

    题目描述 如题,已知一个数列,你需要进行下面两种操作: 1.将某一个数加上x 2.求出某区间每一个数的和 输入输出格式 输入格式: 第一行包含两个整数N.M,分别表示该数列数字的个数和操作的总个数. ...

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

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

  9. POJ 2464 Brownie Points II --树状数组

    题意: 有点迷.有一些点,Stan先选择某个点,经过这个点画一条竖线,Ollie选择一个经过这条直接的点画一条横线.Stan选这两条直线分成的左下和右上部分的点,Ollie选左上和右下部分的点.Sta ...

随机推荐

  1. selinux权限问题【转】

    本文转载自:https://blog.csdn.net/u011386173/article/details/83339770 版权声明:本文为博主原创文章,未经博主允许不得转载. https://b ...

  2. Java 字符串拆分(拆分字符串)

    String sourceStr = "1,2,3,4,5"; String[] sourceStrArray = sourceStr.split(","); ...

  3. Transaction

    SqlTransaction——事务详解 事务是将一系列操作作为一个单元执行,要么成功,要么失败,回滚到最初状态.在事务处理术语中,事务要么提交,要么中止.若要提交事务,所有参与者都必须保证对数据的任 ...

  4. Linux安装svn客户端

    Red Hat Linux 1.安装$ yum install subversion 2.常见问题1.执行svn报错:cannot set LC_CTYPE localevi /etc/profile ...

  5. ASP.NET MVC AntiForgeryToken

    你开发一个网站,其中有个功能:新闻发布. 你是这样实现的: 1.保存新闻的方法是:/News/Save  POST提交 2.接受两个参数:title和content 有一天,你登录网站(浏览器会保存相 ...

  6. go helloworld

    // Sample program to show how a bytes.Buffer can also be used // with the io.Copy function. package ...

  7. SHA-256 加密原理

    网络中传输敏感信息的时候通常会对字符串做加密解密处理 SHA-256 加密原理

  8. pl/sql编译存储过程卡住的解决方法

    oracle编译存过卡住处理: 问题描述: 在编译某个存过时,由于没提交或断网或者test没停止又重新编译,导致编译存过一直卡死 问题分析: 存过或某张表被锁 问题处理: 1.查看存过是否锁住,loc ...

  9. 内连接查询输出到datagridView

    实现步骤: 1. 新建两张对应表的类 例如: 第一张表对应的类 { class ManagerInfo { public Table1 group { get; set; } //重点 需要内连接的字 ...

  10. [原][unreal][UE][spark]分析unreal engine 虚幻引擎的粒子编辑器:Cascade

    参考:https://www.raywenderlich.com/270-unreal-engine-4-particle-systems-tutorial (使用了一个飞机射击游戏的粒子来展示,全英 ...