Loj#6432「PKUSC2018」真实排名(二分查找+组合数)
题面
题解
普通的暴力是直接枚举改或者不改,最后在判断最后对哪些点有贡献。
而这种方法是很难优化的。所以考虑在排序之后线性处理。首先先假设没有重复的元素
struct Node { int poi, id; } a[N];
bool operator < (const Node &a, const Node &b) { return a.poi < b.poi; }
bool operator < (const Node &a, const int &b) { return a.poi < b; }
bool operator < (const int &a, const Node &b) { return a < b.poi; }
int main() {
read(n);
for(int i = 1; i <= n; ++i) read(a[i].poi), a[i].id = i;
sort(a + 1, a + n + 1);
}
对于一个点,我们同样是枚举它改或者不改,但是,接着我们来判断哪些点的变化可以对这个点产生贡献,
决策1:不改
不改的话,那么这个元素后面的元素不管变还是不变都可以产生贡献,假设当前处理到$i$,则其后面有$n-i$个元素。
接着考虑前面的元素,前面的元素改变可以对它产生贡献当且仅当它小于$a_i/2$。这里可以使用二分查找。假设一共有$site$个元素满足上面这个条件。
则这个决策所产生的贡献为:$C_{n-i+site}^k$
决策2:改
当这个元素改的时候,怎么保证它的$rank$不变呢?那么就要保证区间$[a_i,2a_i]$这个区间内的所有数字都要变。同样可以二分来确定这个区间内有多少个元素。假设右界为$tmp$,则有$tot=tmp-i+1$个元素是必须要变的
则这个决策所产生的贡献为:$C_{n-tot}^{k-tot}$
重复的元素
之前的所有决策都是在元素不重复的情况下计算的贡献。那么当元素重复时,怎么计算呢?假设现在同一个元素已经出现了$cf$次。
考虑不改的决策,由于$rank$的含义是大于等于它的数不变,所以这个决策的贡献变为:$C_{n-i+site+cf-1}^k$
接着考虑改变的决策,同样,根据$rank$的定义,这些重复的数字也需要改变。所以$tot$变为:
$$
tot=tmp-i+1+cf-1=tmp-i+cf
$$
接着还有一些细节,比如对于$0$的特判(直接就是$C(n,k)$)之类的
#include <cmath>
#include <cstdio>
#include <cstring>
#include <algorithm>
using std::lower_bound;
using std::upper_bound;
using std::min; using std::max;
using std::swap; using std::sort;
typedef long long ll;
template<typename T>
void read(T &x) {
int flag = 1; x = 0; char ch = getchar();
while(ch < '0' || ch > '9') { if(ch == '-') flag = -flag; ch = getchar(); }
while(ch >= '0' && ch <= '9') x = x * 10 + ch - '0', ch = getchar(); x *= flag;
}
const int N = 1e5 + 10, P = 998244353;
int n, k, ret[N], fac[N], inv[N];
struct Node { int poi, id; } a[N];
bool operator < (const Node &a, const Node &b) { return a.poi < b.poi; }
bool operator < (const Node &a, const int &b) { return a.poi < b; }
bool operator < (const int &a, const Node &b) { return a < b.poi; }
inline void add(int &x) { ++x; if(x == P) x = 0; }
int find(double val, int site) {
int l = 1, r = site - 1, ret = 0;
while(l <= r) {
int mid = (l + r) >> 1;
if(1. * a[mid].poi < val) ret = mid, l = mid + 1;
else r = mid - 1;
} return ret;
}
int C(int n, int m) {
if(n < m) return 0;
return (int)(1ll * (1ll * fac[n] * inv[m] % P) * inv[n - m] % P);
}
int qpow(int a, int b) {
int ret = 1;
for(; b; b >>= 1, a = 1ll * a * a % P) if(b & 1) ret = 1ll * ret * a % P;
return ret;
}
int main () {
read(n), read(k), fac[0] = inv[0] = 1;
for(int i = 1; i <= n; ++i) fac[i] = 1ll * fac[i - 1] * i % P;
inv[n] = qpow(fac[n], P - 2);
for(int i = n; i >= 1; --i) inv[i - 1] = 1ll * inv[i] * i % P;
for(int i = 1; i <= n; ++i) read(a[i].poi), a[i].id = i;
sort(a + 1, a + n + 1);
for(int i = 1, cf = 0; i <= n; ++i) {
if(a[i].poi == 0) { ret[a[i].id] = C(n, k); continue; }
int site = find(1. * a[i].poi / 2., i);
if(a[i].poi == a[i - 1].poi) ++cf;
else cf = 1;
(ret[a[i].id] += C(site + n - i + cf - 1, k)) %= P;//改的决策
int tmp = lower_bound(a + i + 1, a + n + 1, a[i].poi * 2) - a - 1;
if(tmp != -1) {
int tot = tmp - i + cf;
if(k >= tot) (ret[a[i].id] += C(n - tot, k - tot)) %= P;
}//不改的决策
}
for(int i = 1; i <= n; ++i) printf("%d\n", ret[i]);
return 0;
}
Loj#6432「PKUSC2018」真实排名(二分查找+组合数)的更多相关文章
- LOJ #6432. 「PKUSC2018」真实排名(组合数)
题面 LOJ #6432. 「PKUSC2018」真实排名 注意排名的定义 , 分数不小于他的选手数量 !!! 题解 有点坑的细节题 ... 思路很简单 , 把每个数分两种情况讨论一下了 . 假设它为 ...
- Loj 6432. 「PKUSC2018」真实排名 (组合数)
题面 Loj 题解 枚举每一个点 分两种情况 翻倍or不翻倍 \(1.\)如果这个点\(i\)翻倍, 要保持排名不变,哪些必须翻倍,哪些可以翻倍? 必须翻倍: \(a[i] \leq a[x] < ...
- LOJ 6432 「PKUSC2018」真实排名——水题
题目:https://loj.ac/problem/6432 如果不选自己,设自己的值是 x ,需要让 “ a<x && 2*a>=x ” 的非 x 的值不被选:如果选自己 ...
- LOJ #6432. 「PKUSC2018」真实排名
题目在这里...... 对于这道题,现场我写炸了......谁跟我说组合数O(n)的求是最快的?(~!@#¥¥%……& #include <cstdio> #include < ...
- 【LOJ】#6432. 「PKUSC2018」真实排名
题解 简单分析一下,如果这个选手成绩是0,直接输出\(\binom{n}{k}\) 如果这个选手的成绩没有被翻倍,那么找到大于等于它的数(除了它自己)有a个,翻倍后不大于它的数有b个,那么就从这\(a ...
- #6432. 「PKUSC2018」真实排名(组合数学)
题面 传送门 题解 这数据范围--这输出大小--这模数--太有迷惑性了-- 首先对于\(0\)来说,不管怎么选它们的排名都不会变,这个先特判掉 对于一个\(a_i\)来说,如果它不选,那么所有大于等于 ...
- 「PKUSC2018」真实排名(排列组合,数学)
前言 为什么随机跳题会跳到这种题目啊? Solution 我们发现可以把这个东西分情况讨论: 1.这个点没有加倍 这一段相同的可以看成一个点,然后后面的都可以. 这一段看成一个点,然后前面的不能对他造 ...
- 「PKUSC2018」真实排名(组合)
一道不错的组合数问题! 分两类讨论: 1.\(a_i\) 没有翻倍,那些 \(\geq a_i\) 和 \(a_j\times 2<a_i\) 的数就没有影响了.设 \(kth\) 为 \(a_ ...
- 「PKUSC2018」真实排名
题面 题解 因为操作为将一些数字翻倍, 所以对于一个数\(x\), 能影响它的排名的的只有满足\(2y\geq x\)或\(2x>y\)的\(y\) 将选手的成绩排序,然后考虑当前点的方案 1. ...
随机推荐
- LightOJ 1062 - Crossed Ladders 基础计算几何
http://www.lightoj.com/volume_showproblem.php?problem=1062 题意:问两条平行边间的距离,给出从同一水平面出发的两条相交线段长,及它们交点到水平 ...
- 【hdu5217-括号序列】线段树
题意:给一串括号,有2个操作,1.翻转某个括号.2.查询某段区间内化简后第k个括号是在原序列中的位置.1 ≤ N,Q ≤ 200000. 题解: 可以知道,化简后的序列一定是)))((((这种形式的. ...
- Spring理论基础-面向切面编程
AOP是Aspect-Oriented Programming的缩写,中文翻译是面向切面编程.作为Spring的特征之一,是要好好学习的. 首先面向切面编程这个名称很容易让人想起面向对象编程(OOP) ...
- vue router mode 设置"hash"与"history"的区别
router官网的说明如下: ********************************************我是官网说明分隔符--开始**************************** ...
- Ribbon的主要组件与工作流程
一:Ribbon是什么? Ribbon是Netflix发布的开源项目,主要功能是提供客户端的软件负载均衡算法,将Netflix的中间层服务连接在一起.Ribbon客户端组件提供一系列完善的配置项如连接 ...
- 配置连接的IP、端口、以及相应的数据库
解压后里面有:lib 源文件 .examples 例子.test测试 将lib目录拷贝到你的项目中,就可以开始你的predis操作了. //使用autoload加载相关库,这边重点就是为了requir ...
- 设计模式之笔记--桥接模式(Bridge)
桥接模式(Bridge) 定义 桥接模式(Bridge),将抽象部分与它的实现部分分离,使它们都可以独立地变化. 类图 描述 Abstraction:定义抽象部分的接口,通常在这个接口里面要维护一个实 ...
- 《Java编程思想》阅读笔记一
Java编程思想 这是一个通过对<Java编程思想>(Think in java)第四版进行阅读同时对java内容查漏补缺的系列.一些基础的知识不会被罗列出来,这里只会列出一些程序员经常会 ...
- AC日记——[HNOI2010]BOUNCE 弹飞绵羊 洛谷 P3203
[HNOI2010]BOUNCE 弹飞绵羊 思路: SBlct: 代码: #include <bits/stdc++.h> using namespace std; #define max ...
- AC日记——[POI2014]KUR-Couriers 洛谷 P3567
[POI2014]KUR-Couriers 思路: 卡空间,sb题: 代码: #include <bits/stdc++.h> using namespace std; #define m ...