一个经典的二维数点模型,如果某个人 $ x $ 两个速度都比另一个人 $ y $ 大,显然 $y$ 是不可能成为winner的。

但这里只考虑两个人$x$,$y$在两个属性各有千秋的时候,一定存在正整数$S$,$R$使得$x$,$y$都有可能成为winner。

这时考虑单调栈中顶端的两个人$a$和$b$,以及此时准备加入的人$c$,很明显当两两无法比较时,可能存在如下情况:

  1. 当$S$与$R$的比值小于一定值时,$a$总能胜$b$。
  2. 当$S$与$R$的比值大于一定值时,$c$总能胜$b$。

可以看到,当前者的阀值大于后者的阀值时,就不存在任何一组$S$和$R$,能使$b$成为winner了。(注:这里已经将人按照$s$值排了序)

前者的不等式大概长这样(当$a$能胜$b$):

$ \frac{ s_{a} \, s_{b} \, ( r_{a} - r_{b} ) }{ r_{a} \, r_{b} \, ( s_{b} - s_{a} ) } > \frac{ S }{ R } $

后者的不等式大概长这样(当$c$能胜$b$):

$ \frac{ s_{c} \, s_{b} \, ( r_{b} - r_{c} ) }{ r_{c} \, r_{b} \, ( s_{c} - s_{b} ) } < \frac{ S }{ R } $

将式子化简后就能判断$b$有没有win的可能性了。

这里简单证明为什么单调栈做可以不遗漏地剔除所有不合法的人:

  1. 在单调栈维护的过程中,保证了占中连续三个人之间是合法的。
  2. 也就是对于任意的 $x_{i }$,$x_{i+1}$,$x_{i+2}$,假设$x_{i}$能胜$x_{i+1}$的阀值是$v_{i}$,都有$v_{i} < v_{i+1}$,即合法的。
  3. 现在要证明对于在栈中的任意的$x_{i}$,$x_{j}$,$x_{k}$,其中$i < j < k$,都不能使$x_{j}$被剔除。

考虑另一种思路,可以将以上棘手的证明转化:

将每一个人的属性以$(\frac{1}{s_{i}}, \frac{1}{r_{i}})$的形式投射到坐标系上。答案$t = \frac{S}{s_{i}} + \frac{R}{r_{i}} = (S, R) \cdot (\frac{1}{s_{i}} + \frac{1}{r_{i}})$。

即在$(S,R)$方向上的投影。

对于任意一个$(S,R)$,在它上投影相等的点在一条垂直于它的直线上,该直线越靠近原点,投影长度越小。

枚举的任意一个$(S,R)$就相当于枚举了一条这直线,最先出现在直线平行线上的点就是winner。

显然,这个点在下凸壳上。并且可以得到,对于下凸壳上的每一个点都存在一组$(S,R)$使得其为winner。

而我们单调栈所做的就相当于自右下而左上地求出这样一个下凸壳。事实证明,两者的表达式是完全相同的。

$ \bigodot $ 技巧&套路:

  • 单调栈的应用,单调栈剔除的变式。
 #include <cstdio>
#include <set>
#include <algorithm> typedef long long LL;
const int N = ; struct No {
int x, y, id;
inline friend bool operator < (No a, No b) {
return (a.x == b.x)? (a.y < b.y) : (a.x < b.x);
}
} p[N]; int n, sta[N], top;
std::set<No> S; inline int Check(No a, No b, No c) {
return (LL) a.x * (a.y - b.y) * c.y * (c.x - b.x) > (LL) c.x * (b.y - c.y) * a.y * (b.x - a.x);
} int main() {
scanf("%d", &n);
for (int i = ; i <= n; ++i) {
scanf("%d%d", &p[i].x, &p[i].y);
p[i].id = i;
}
std::sort(p + , p + + n); for (int i = ; i <= n; ++i) {
while (top >= && p[sta[top]].y <= p[i].y) --top;
while (top >= && Check(p[sta[top - ]], p[sta[top]], p[i])) --top;
sta[++top] = i;
}
for (int i = ; i <= top; ++i) {
S.insert(p[sta[i]]);
}
top = ;
for (int i = ; i <= n; ++i) {
if (S.count(p[i])) sta[++top] = p[i].id;
}
std::sort(sta + , sta + + top);
for (int i = ; i <= top; ++i) {
printf("%d ", sta[i]);
} return ;
}

【Cf #299 C】Tavas and Pashmaks(单调栈,凸性)的更多相关文章

  1. CF535E Tavas and Pashmaks 单调栈、凸包

    传送门 题意:有一场比赛,$N$个人参加.每个人有两种参数$a,b$,如果存在正实数$A,B$使得$\frac{A}{a_i} + \frac{B}{b_i}$在$i=x$处取得最大值(可以有多个最大 ...

  2. CF 602 D. Lipshitz Sequence 数学 + 单调栈 + 优化

    http://codeforces.com/contest/602/problem/D 这题需要注意到的是,对于三个点(x1, y1)和(x2, y2)和(x3, y3).如果要算出区间[1, 3]的 ...

  3. Codeforces Round #299 (Div. 1)C. Tavas and Pashmaks (凸壳)

    C. Tavas and Pashmaks   Tavas is a cheerleader in the new sports competition named "Pashmaks&qu ...

  4. CF Mike and Feet (求连续区间内长度为i的最小值)单调栈

    Mike and Feet time limit per test 1 second memory limit per test 256 megabytes input standard input ...

  5. [CSP-S模拟测试]:导弹袭击(数学+凸包+单调栈)

    题目背景 $Guess$准备向敌军阵地发起进攻了!$Guess$的武器是自动制导导弹.然而在机房是不允许游戏的,所以班长$XZY$对游戏界面进行了降维打击,结果... 题目描述 众所周知,环境因素对导 ...

  6. 【单调栈】最长不上升子序列变式,洛谷 P2757 导弹的召唤

    题目背景 易琢然今天玩使命召唤,被敌军用空对地导弹轰炸,很不爽:众所周知,易琢然很不老实,他开了外挂: 外挂第一次可以打掉任意高度的导弹,之后每一次都不能打掉大于上一次高度的导弹: 但易琢然水平太差, ...

  7. BZOJ1012: [JSOI2008]最大数maxnumber [线段树 | 单调栈+二分]

    1012: [JSOI2008]最大数maxnumber Time Limit: 3 Sec  Memory Limit: 162 MBSubmit: 8748  Solved: 3835[Submi ...

  8. BZOJ 4453: cys就是要拿英魂![后缀数组 ST表 单调栈类似物]

    4453: cys就是要拿英魂! Time Limit: 3 Sec  Memory Limit: 128 MBSubmit: 90  Solved: 46[Submit][Status][Discu ...

  9. BZOJ 3238: [Ahoi2013]差异 [后缀数组 单调栈]

    3238: [Ahoi2013]差异 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 2326  Solved: 1054[Submit][Status ...

随机推荐

  1. lua中table的常用方法

    转载:https://blog.csdn.net/Fenglele_Fans/article/details/83627021 1:table.sort() language = {"lua ...

  2. 2018NOIP爆0记第一弹

    初赛篇 选择即王道 迪杰斯特拉那道题的A选项自己yy一下觉得甚是不妥,就没选 就和30分完美选择题擦肩而过. 填空最后一题不太会搞,就跳过了,最后蒙了个512上去...其实还有点接近的... 5分 然 ...

  3. RHEL7 利用双网卡绑定实现VLAN

    使用nmcli创建bond配置 #nmcli connection add type bond ifname bond0 con-name bond0 mode active-backup #nmcl ...

  4. VGG——Very deep convolutional networks for large-scale image recognition

    1. 摘要 在使用非常小(3×3)的卷积核情况下,作者对逐渐增加网络的深度进行了全面的评估,通过设置网络层数达 16-19 层,最终效果取得了显著提升. 2. 介绍 近来,卷积神经网络在大规模图像识别 ...

  5. whoami,who,w命令详解

    http://www.voidcn.com/blog/wszzdanm/article/p-6145895.html 命令功能:显示登录用户的信息 命令格式: 常用选项: 举例: w 显示已经登录的用 ...

  6. 投稿007期|令人震惊到发指的PyObject对象代码设计之美

    前言 最近在重温经典漫画<SlamDunk>的全国大赛篇,其中的一个情形可以很好的诠释虎躯一震这个状态——当樱木看到流川枫一次高难度投篮时内心的感受:“经过两万次射球练习后,樱木首次明白到 ...

  7. 在Gulp中使用BrowserSync

    博客已迁移至http://zlwis.me. 很早就听说过BrowserSync,也看过一些相关文章,可就是没用过.之前一直在用Gulp开发项目,每次编写完Sass后还要用按F5刷新页面看效果,想想也 ...

  8. ffmpeg——压缩mav格式音频

    今天偶然帮朋友压缩一个mav格式的音频.开始用压缩码率的方式,mav格式的音频体积一点都没变,查资料需要压缩音频文件的采样率和声道才能压缩mav格式的音频. 压缩要求是:将一个mav格式的音频文件,由 ...

  9. 作业1-MathExam

    MathExam 一.预估与实际 PSP2.1 Personal Software Process Stages 预估耗时(分钟) 实际耗时(分钟) Planning 计划 10 30 • Estim ...

  10. HDU 3092 Least common multiple 01背包

    题目链接: http://acm.hdu.edu.cn/showproblem.php?pid=3092 Least common multiple Time Limit: 2000/1000 MS ...