Loj#6432「PKUSC2018」真实排名(二分查找+组合数)

题面

Loj

题解

普通的暴力是直接枚举改或者不改,最后在判断最后对哪些点有贡献。

而这种方法是很难优化的。所以考虑在排序之后线性处理。首先先假设没有重复的元素

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;
} 

原文地址:https://www.cnblogs.com/water-mi/p/10287346.html

时间: 2024-07-30 18:40:59

Loj#6432「PKUSC2018」真实排名(二分查找+组合数)的相关文章

Loj 6433. 「PKUSC2018」最大前缀和 (状压dp)

题面 Loj 题解 感觉挺难的啊- 状压\(dp\) 首先,有一个性质 对于一个序列的最大前缀和\(\sum_{i=1}^{p} A[i]\) 显然对于每个\(\sum_{i=p+1}^{x}A[i](p+1 \leq x \leq n)<0\) 我们可以以\(p\)分成两个集合 \(n\leq 20\) 所以状压一下 \(sum[i]\)表示当前状态表示的和 \(f[i]\)表示用当前状态的数,组成最大前缀和为\(sum[i]\)的方案数 \(g[i]\)表示当前状态的数,组成的序列,每个前缀

Loj#6434「PKUSC2018」主斗地(搜索)

题面 Loj 题解 细节比较多的搜索题. 首先现将牌型暴力枚举出来,大概是\(3^{16}\)吧. 然后再看能打什么,简化后无非就三种决策:单牌,\(3+x\)和\(4+x\). 枚举网友打了几张\(3\)和\(4\),然后再枚举吉老师(\(\mathbf {orz}\))打了几张\(3\)和\(4\). 接着枚举\(3\)搭配了几个\(2\),然后贪心地从大到小去除吉老师手中大小为\(2\)的对子,从小到大去除网友手中大小为\(2\)的对子.之后就是检查单牌是否合法了. #include <c

loj 6434「PKUSC2018」主斗地

loj 最可做的斗地主系列题(?) 显然的想法是爆搜可怜的牌,然后接着找是否有合法出牌方案.因为总的方案数只有几百万种,所以可以直接枚举每种方案 然后是优化check过程.首先可以发现对子三张牌顺子连对三顺可以拆成若干单牌,飞机可以拆成若干三带一或三带二,所以只有我们只用考虑单牌,三带一,三带二,四带二.如果只考虑单牌,那么一定是两者的牌分别排好序后,可怜某张牌要严格小于网友的对应位置的牌,所以这个可以从大到小枚举牌大小,然后看可怜的每种牌是否都有网友的更大的牌可以配上对,复杂度为\(O(14)

loj#2552. 「CTSC2018」假面

题目链接 loj#2552. 「CTSC2018」假面 题解 本题严谨的证明了我菜的本质 对于砍人的操作好做找龙哥就好了,blood很少,每次暴力维护一下 对于操作1 设\(a_i\)为第i个人存活的概率,\(d_i\)为死掉的概率,\(g_{i,j}\)是除i以外活了j个人的概率 那个选中i人的答案就是 \[a_i\times\sum_{j = 0} ^{k - 1}\frac{g_{i,j}}{j + 1}\] 对于\(g_{i,j}\) ,设\(f_{i,j}\)表示前\(i\)个人有\(

loj#2076. 「JSOI2016」炸弹攻击 模拟退火

目录 题目链接 题解 代码 题目链接 loj#2076. 「JSOI2016」炸弹攻击 题解 模拟退火 退火时,由于答案比较小,但是温度比较高 所以在算exp时最好把相差的点数乘以一个常数让选取更差的的概率降低 代码 #include<ctime> #include<cmath> #include<cstdio> #include<cstring> #include<algorithm> #define gc getchar() #define

Loj #2541「PKUWC2018」猎人杀

Loj #2541. 「PKUWC2018」猎人杀 题目链接 好巧妙的题! 游戏过程中,概率的分母一直在变化,所以就非常的不可做. 所以我们将问题转化一下:我们可以重复选择相同的猎人,只不过在一个猎人被选择了过后我们就给他打上标记,再次选择他的时候就无效.这样与原问题是等价的. 证明: 设\(sum=\sum_iw_i,kill=\sum_{i被杀死了}w_i\). 攻击到未被杀死的猎人\(i\)的概率为\(P\). 则根据题意\(P=\frac{w_i}{sum-kill}\). 问题转化后:

Loj #2542. 「PKUWC2018」随机游走

Loj #2542. 「PKUWC2018」随机游走 题目描述 给定一棵 \(n\) 个结点的树,你从点 \(x\) 出发,每次等概率随机选择一条与所在点相邻的边走过去. 有 \(Q\) 次询问,每次询问给定一个集合 \(S\),求如果从 \(x\) 出发一直随机游走,直到点集 \(S\) 中所有点都至少经过一次的话,期望游走几步. 特别地,点 \(x\)(即起点)视为一开始就被经过了一次. 答案对 $998244353 $ 取模. 输入格式 第一行三个正整数 \(n,Q,x\). 接下来 \(

Loj #2192. 「SHOI2014」概率充电器

Loj #2192. 「SHOI2014」概率充电器 题目描述 著名的电子产品品牌 SHOI 刚刚发布了引领世界潮流的下一代电子产品--概率充电器: 「采用全新纳米级加工技术,实现元件与导线能否通电完全由真随机数决定!SHOI 概率充电器,您生 活不可或缺的必需品!能充上电吗?现在就试试看吧!」 SHOI 概率充电器由 \(n-1\) 条导线连通了 \(n\) 个充电元件.进行充电时,每条导线是否可以导电以 概率决定,每一个充电元件自身是否直接进行充电也由概率决定.随后电能可以从直接充电的元件经

Loj #3111. 「SDOI2019」染色

Loj #3111. 「SDOI2019」染色 题目描述 给定 \(2 \times n\) 的格点图.其中一些结点有着已知的颜色,其余的结点还没有被染色.一个合法的染色方案不允许相邻结点有相同的染色. 现在一共有 \(c\) 种不同的颜色,依次记为 \(1\) 到 \(c\).请问有多少对未染色结点的合法染色方案? 输入格式 第一行有两个整数 \(n\) 和 \(c\),分别描述了格点图的大小和总的颜色个数. 之后两行,每行有 \(n\) 个整数:如果是 \(0\) 则表示对应结点未被染色,否