【南开OJ2264】节操大师(贪心+二分+并查集/平衡树)

好久没更新了,今天就随便写一个吧

题目内容

MK和他的小伙伴们(共n人,且保证n为2的正整数幂)想要比试一下谁更有节操,于是他们组织了一场节操淘汰赛。他们的比赛规则简单而暴力:两人的节操正面相撞,碎的一方出局,而没碎的一方晋级(脑补一下端午节的碰鸡蛋游戏>_<)。最后经过数轮淘汰决出冠军“节操大师”。

通过理性的研究,你测算出他们的节操值分别为1,2,...,n,我们不妨称这个值为“硬度”吧。同时你又测出了一个节操常数k:当两个硬度相差超过k的节操相撞时,硬度小的节操必碎;而当两个硬度相差不超过k的节操相撞时,由于现场操作的不确定因素,两个节操都有碎的可能(当然我们假设不会出现两边都碎的情况囧)。

显然,节操值较低的人也许没有任何可能得到冠军。下面就请你预测,这次比赛的冠军“节操大师”的节操最小值为多少。

(n≤131072, 保证n为2的正整数幂)

首先要二分答案,然后从结果考虑,最终假如是X决赛获胜,那么我们贪心地想,让X与他能获胜的节操值最大的人进行半决赛,这样可以达到最优的结果

如果那个人已经参与了比赛,就选第二大的,依次类推。

于是这个题就有了两种做法,一种是用并查集,一种是用平衡树(或者是set)

每次查询后利用并查集并点(logn)或者直接用set删点(logn)

最后的复杂度就是nlognlogn

#include <iostream>
#include <vector>
#include <cstdio>
#include <cstring>
#include <set>
using namespace std;
const int maxn = 131073;
vector <int> V;
set<int> S;
int n, k;

bool ok(int u)
{
    S.clear();
    for(int i = 0; i <= n; i++) S.insert((-i));
    V.clear();
    V.push_back(-u); S.erase(-u);
    while(V.size() != n)
    {
        int l = V.size();
        for(int i = 0; i < l; i++)
        {
            int x = V[i];
            int y = *S.lower_bound(x-k);
            if(y == 0) return false;
            V.push_back(y);
            S.erase(y);
        }
        //for(int i = 0; i < l; i++) printf("%d ", -V[i]);
        //cout<<endl;
    }
    return true;
}

int main()
{
    while(cin>>n>>k)
    {
        int l = 1, r = n, mid;
        while(l < r)
        {
            mid = (l+r)/2;
            if(ok(mid)) r = mid;
            else l = mid+1;
        }
        cout<<l<<endl;
    }
}
时间: 2024-10-06 04:06:09

【南开OJ2264】节操大师(贪心+二分+并查集/平衡树)的相关文章

POJ 2263 Heavy Cargo(二分+并查集)

题目地址:POJ 2263 这题是在网上的一篇关于优先队列的博文中看到的..但是实在没看出跟优先队列有什么关系..我用的二分+并查集做出来了... 二分路的载重量.然后用并查集检查是否连通. 代码如下: #include <iostream> #include <stdio.h> #include <string.h> #include <stdlib.h> #include <math.h> #include <ctype.h> #

HDU 3081Marriage Match II(二分+并查集+网络流之最大流)

题目地址:http://acm.hdu.edu.cn/showproblem.php?pid=3081 有一段时间没写最大流的题了,这题建图居然想了好长时间...刚开始是按着最终的最大流即是做多轮数去想建图,结果根本没思路,后来想了想,可以用二分答案的思想来找最终答案.然后很明显的并查集,但是并查集学的略渣,居然卡在并查集上了..= =. 但是也不是并查集的事..是我建图的思想太正了,稍微用点逆向思维并查集就可以很好利用了. 建图思路是:建立一个源点与汇点,将女孩与源点相连,男孩与汇点相连,权值

poj 1827 A Bunch Of Monsters 贪心(并查集优化)

Description Background Jim is a brave explorer. One day, he set out for his next destination, a mysterious hill. When he arrived at the foot of the hill, he was told that there were a bunch of monsters living in that hill, and was dissuaded from cont

hdu 3081 Marriage Match II(最大流 + 二分 + 并查集)

Marriage Match II                                                                           Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Problem Description Presumably, you all have known the question of stable

POJ 1797 Heavy Transportation(二分+并查集/kruskal)

Heavy Transportation Time Limit: 3000MS   Memory Limit: 30000K Total Submissions: 24398   Accepted: 6472 Description Background Hugo Heavy is happy. After the breakdown of the Cargolifter project he can now expand business. But he needs a clever man

P2898 [USACO08JAN]haybale猜测Haybale Guessing 二分 + 并查集

二分 + 并查集 题目链接:https://www.luogu.org/problemnew/show/2898 题目大意: 给一段长度为n,每个位置上的数都不同的序列a[1..n]和q和问答, 每个问答是(x, y, r)代表RMQ(a, x, y) = r, 要你给出最早的有矛盾的那个问答的编号. 首先,要你求的是出现矛盾的,那什么时候才会出现矛盾呢? 可以总结为两种情况: ①之前已更新这个区间最小值为x,又要更新此区间的子区间(或这个区间)的最小值为更小的数. 这样~ (黑色是已更新区间,

POJ 3228 网络流+二分&amp;并查集

点击打开链接 题意:有n个城镇,第一行是金矿和金子数量,然后第二行是装金子的地方和能装的数量,在下面是m条道路,问你选择的道路中最大值最小,使得所有金子运到装金子的地方 思路:最大值最小,根本不用考虑一看就是二分,然后想了想就是个网络流的模型嘛,很简单,被坑了几次道路是双向的,改过之后A掉,然后看了看讨论还可以用并查集写,这里两种方法都写了,先是网络流的直接二分最大值,然后满足条件的边建模型,跑一边就行了 #include <queue> #include <vector> #in

bzoj 1196: [HNOI2006]公路修建问题 二分+并查集

题目链接 1196: [HNOI2006]公路修建问题 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1576  Solved: 909[Submit][Status][Discuss] Description OI island是一个非常漂亮的岛屿,自开发以来,到这儿来旅游的人很多.然而,由于该岛屿刚刚开发不久,所以那里的交通情况还是很糟糕.所以,OIER Association组织成立了,旨在建立OI island的交通系统. OI isla

[BZOJ1594] [Usaco2008 Jan]猜数游戏(二分 + 并查集)

传送门 题中重要信息,每堆草的数量都不一样. 可以思考一下,什么情况下才会出现矛盾. 1.如果两个区间的最小值一样,但是这两个区间没有交集,那么就出现矛盾. 2.如果两个区间的最小值一样,并且这两个区间有交集,那么这个最小值一定在交集中,但是如果这个交集被某个最小值较大的区间,或是一些最小值较大的区间的并集包含,那么也是矛盾的. 可以二分答案,将这些区间按照最小值从大到小排序,然后可以用线段树维护,也可以用并查集来搞. 下面是用并查集来搞的. 每到一个区间,可以将[l,r]中的f变成r+1,如果