@loj - 6353@「CodePlus 2018 4 月赛」组合数问题 2
@description@
请你找到 k 个不同的组合数,使得对于其中任何一个组合数 \(C_a^b\) 有 \(0\leq b\leq a\leq n\)。所谓不同的组合数,即对于组合数 \(C_{a_1}^{b_1}\) 和 \(C_{a_2}^{b_2}\) ,若 \(a_1\neq a_2\) 或者 \(b_1\neq b_2\) ,则我们认为这两个组合数是不同的。问这 \(k\) 个组合数的和最大是多少?
input
第一行两个整数 n, k。
output
一行一个整数,代表 k 个组合数的和对 10^9+7 取模之后的结果;数据保证一定有至少 k 个数可以选。
sample input
2 3
sample output
4
对于 \(20\%\) 的数据,\(n\leq 10\)。
对于 \(40\%\) 的数据,\(n\leq 500\)。
对于另外 \(20\%\) 的数据,\(k=1\)。
对于 \(100\%\) 的数据, \(1\leq n\leq 10^6,1\leq k\leq 10^5\) 。
@solution@
问题相当于求前 k 大的组合数。
当 \(a < b < m/2\) 或 \(a > b > m/2\) 时,有 \(C_{m}^a < C_m^{b}\);
当 \(a < b\) 时,有 \(C_{a}^p < C_{b}^p\)。
这是可以从杨辉三角中看出来的。
我们可以把最大的那个组合数 \(C_n^{n/2}\) 加入优先队列 ,然后向四周扩展。每一次从优先队列中取出最大值 \(C_{a}^{b}\),扩展出 \(C_{a-1}^{b}\), \(C_{a}^{b-1}\),\(C_{a}^{b+1}\),然后将它们加入优先队列。扩展 k 次即可。
同时要注意不要重复经过某一个点。开一个 set 判一下重。
那么问题来了:我们的组合数是取了模的,塞在优先队列里面怎么比较大小呢?逼我写高精度?
这个时候,一个闻所未闻的操作就来了:两边同时取对数。
我们组合数公式长这样:
\]
取完对数长这样:
\]
(当然底数不一定为 2)
可以发现这个式子是绝对不会溢出的,而且还可以前缀和预处理。
又因为底数大于 1,所以对数的大小关系等同于原数的大小关系。
所以我们就可以比较组合数之间的大小了。
那么问题来了:double 的精度真的不会翻车吗_(:з」∠)_?
@accepted code@
#include<set>
#include<cmath>
#include<queue>
#include<cstdio>
#include<iostream>
using namespace std;
const int MOD = int(1E9) + 7;
const int MAXN = 1000000;
int fct[MAXN + 5], inv[MAXN + 5];
double lgsum[MAXN + 5];
int pow_mod(int b, int p) {
int ret = 1;
while( p ) {
if( p & 1 ) ret = 1LL*ret*b%MOD;
b = 1LL*b*b%MOD;
p >>= 1;
}
return ret;
}
void init() {
fct[0] = 1;
for(int i=1;i<=MAXN;i++)
fct[i] = 1LL*fct[i-1]*i%MOD;
inv[MAXN] = pow_mod(fct[MAXN], MOD-2);
for(int i=MAXN-1;i>=0;i--)
inv[i] = 1LL*inv[i+1]*(i+1)%MOD;
lgsum[1] = log2(1);
for(int i=2;i<=MAXN;i++)
lgsum[i] = lgsum[i-1] + log2(i);
}
int C(int n, int m) {
return 1LL*fct[n]*inv[m]%MOD*inv[n-m]%MOD;
}
struct node{
int n, m;
node(int _n=0, int _m=0):n(_n), m(_m){}
};
bool operator < (node a, node b) {
if( lgsum[a.n] - lgsum[a.m] - lgsum[a.n - a.m] != lgsum[b.n] - lgsum[b.m] - lgsum[b.n - b.m] )
return lgsum[a.n] - lgsum[a.m] - lgsum[a.n - a.m] < lgsum[b.n] - lgsum[b.m] - lgsum[b.n - b.m];
else return (a.n == b.n) ? a.m < b.m : a.n < b.n;
}
set<node>Set;
priority_queue<node>que;
int main() {
int n, k, ans = 0; init();
scanf("%d%d", &n, &k);
node s = node(n, n/2);
Set.insert(s), que.push(s);
for(int i=1;i<=k;i++) {
s = que.top(); que.pop();
ans = (ans + C(s.n, s.m))%MOD;
if( s.m != 0 ) {
if( !Set.count(node(s.n, s.m - 1)) )
Set.insert(node(s.n, s.m - 1)), que.push(node(s.n, s.m - 1));
}
if( s.m != s.n ) {
if( !Set.count(node(s.n, s.m + 1)) )
Set.insert(node(s.n, s.m + 1)), que.push(node(s.n, s.m + 1));
if( !Set.count(node(s.n - 1, s.m)) )
Set.insert(node(s.n - 1, s.m)), que.push(node(s.n - 1, s.m));
}
}
printf("%d", ans);
}
@details@
重新刷新了 double 的精度问题。
MD 每次我写二分你都卡我精度你这次偏偏不会卡这玩意儿的精度?
取对数这种操作也不是没有见过,《麦森数》那道题就有用到。
看来印象不深刻 QAQ……我果然还是太弱了 QAQ。
@loj - 6353@「CodePlus 2018 4 月赛」组合数问题 2的更多相关文章
- LOJ#6354. 「CodePlus 2018 4 月赛」最短路[最短路优化建图]
题意 一个 \(n\) 个点的完全图,两点之间的边权为 \((i\ xor\ j)*C\) ,同时有 \(m\) 条额外单向路径,问从 \(S\) 到 \(T\) 的最短路. \(n\leq 10^5 ...
- loj #6302. 「CodePlus 2018 3 月赛」寻找车位【线段树+单调队列】
考虑静态怎么做:枚举右边界,然后枚举上边界,对应的下边界一定单调不降,单调栈维护每一列从当前枚举的右边界向左最长空位的长度,这样是O(nm)的 注意到n>=m,所以m<=2000,可以枚举 ...
- @loj - 6354@「CodePlus 2018 4 月赛」最短路
目录 @description@ @solution@ @accepted code@ @details@ @description@ 企鹅国中有 N 座城市,编号从 1 到 N . 对于任意的两座城 ...
- 【LibreOJ】#6354. 「CodePlus 2018 4 月赛」最短路 异或优化建图+Dijkstra
[题目]#6354. 「CodePlus 2018 4 月赛」最短路 [题意]给定n个点,m条带权有向边,任意两个点i和j还可以花费(i xor j)*C到达(C是给定的常数),求A到B的最短距离.\ ...
- [LOJ#6259]「CodePlus 2017 12 月赛」白金元首与独舞
[LOJ#6259]「CodePlus 2017 12 月赛」白金元首与独舞 试题描述 到河北省 见斯大林 / 在月光下 你的背影 / 让我们一起跳舞吧 うそだよ~ 河北省怎么可能有 Stalin. ...
- loj #6250. 「CodePlus 2017 11 月赛」找爸爸
#6250. 「CodePlus 2017 11 月赛」找爸爸 题目描述 小 A 最近一直在找自己的爸爸,用什么办法呢,就是 DNA 比对. 小 A 有一套自己的 DNA 序列比较方法,其最终目标是最 ...
- [LOJ 6249]「CodePlus 2017 11 月赛」汀博尔
Description 有 n 棵树,初始时每棵树的高度为 H_i,第 i 棵树每月都会长高 A_i.现在有个木料长度总量为 S 的订单,客户要求每块木料的长度不能小于 L,而且木料必须是整棵树(即不 ...
- [LOJ 6248]「CodePlus 2017 11 月赛」晨跑
Description “无体育,不清华”.“每天锻炼一小时,健康工作五十年,幸福生活一辈子” 在清华,体育运动绝对是同学们生活中不可或缺的一部分.为了响应学校的号召,模范好学生王队长决定坚持晨跑.不 ...
- loj6300 「CodePlus 2018 3 月赛」博弈论与概率统计
link 题意: A和B玩游戏,每轮A赢的概率为p.现在有T组询问,已知A赢了n轮输了m轮,没有平局,赢一局A得分+1,输一局得分-1,问A得分期望值? $n+m,T\leq 2.5\times 10 ...
随机推荐
- 【maven】maven pom文件详解
maven pom文件详解 最近配置maven中的pom文件,maven中有些属性不太清楚,在这里记录一下 <project xmlns="http://maven.apache.or ...
- beego应用做纯API后端如何使用jwt实现无状态权限验证
jwt是什么,可以百度下其它文章,我原来看到一个讲的详细的,现在找不到了.先简单介绍下我个人的理解,就是一个token,只不过通过加密解密的手段,能让这一串字符带有一些简单的信息.这样解密jwt后不用 ...
- 开户项目的sql查询语句备查
查询全国所有的券商名录 SELECT id,security,province,city,borough FROM t_security GROUP BY security ; 查询某个省份的所有券商 ...
- 使用session实现一次性验证码
在登录页面和各种页面,会看到有验证码输入,这样做的目的是为了防止密码猜测工具破解密码,保护了用户密码安全,验证码只能使用一次,这样就给密码猜测工具带来了很大的困难,基本上阻断了密码猜测工具的使用. 可 ...
- 未压缩的jQuery
/*! * jQuery JavaScript Library v3.4.1 * https://jquery.com/ * * Includes Sizzle.js * https://sizzle ...
- 凸优化 & 1概念
---恢复内容开始--- 放射集合 系数之和为1 相加仍然能在集合内,就是 纺射集合 子空间加一个常熟 就是纺射集合 , 例题2.1 一类特殊的线性方程组的解可以看作纺射 集合 纺射包 aff C 是 ...
- 洛谷P2468 [SDOI2010]粟粟的书架
来了来了,随便拽一道题写题解[大雾] 最近发现自己基础奇差于是开始复习之前学过的东西,正好主席树我几乎完全没学会,然后打开洛谷试炼场… 发现了这么一道二合一的题. 这道题其实分成两个部分,前50%是一 ...
- 【洛谷P2907】 【USACO08OPEN】农场周围的道路 水模拟分治
P2907 [USACO08OPEN]农场周围的道路Roads Around The Farm 题目描述 Farmer John's cows have taken an interest in ex ...
- 下载并安装Cent OS 6.5
到官网下载centos 6.5指引 官网:https://www.centos.org/ [当然也可以通过百度搜索,然后打开] 进入官网,选择"Get CentOS Now" ...
- CLTPHP5.0发布
CLTPHP内容管理系统,包含系统设置,权限管理,模型管理,数据库管理,栏目管理,会员管理,网站功能,模版管理,微信管理等相关模块. CLTPHP内容管理系统=ThinkPHP5+layuiAdm ...