【BZOJ 3907】网格(Catalan数)
题目链接
这个题推导公式跟\(Catalan\)数是一样的,可得解为\(C_{n+m}^n-C_{n+m}^{n+1}\)
然后套组合数公式\(C_n^m=\frac{n!}{m!(n-m)!}\)
用阶乘分解的方法对分子和分母分解质因数然后指数相减,最后把剩下的高精度乘起来就行了,这样就避免了高精除法。可以用快速幂,但我太lan了,就直接暴力乘起来了。
说一下怎么阶乘分解,直接对每个数分解质因数的时间复杂度是\(O(n\sqrt{n})\),这显然是不可忍受的。
于是,考虑先用线筛求出\(1~n\)之间所有质数,然后枚举所有质数\(p\),\(1~n\)中所有\(p\)的倍数包含一个质因子\(p\),所有\(p^2\)的倍数包含一个质因子\(p\),...
所以,\(n!\)中\(p\)的指数为\(\sum_{i=1}^{\left\lfloor \log_p n\right\rfloor}\left\lfloor n/p^i \right\rfloor\)
枚举\(p\)求就行了,时间复杂度\(O(n\ log\ n)\)。
Code:
#include <cstdio>
#include <cstring>
const int MAXN = 10010;
int prime[MAXN], v[MAXN], c[MAXN], d[MAXN];
int n, cnt, m;
const int MOD = 10000;
struct Int{
int s[MAXN];
Int(int x){ memset(s, 0, sizeof s); s[++s[0]] = x % 10, x /= 10; }
void mul(int x){
s[1] *= x;
for(int i = 2; i <= s[0]; ++i){
s[i] = s[i] * x + s[i - 1] / MOD;
s[i - 1] %= MOD;
}
while(s[s[0]] >= MOD){
s[++s[0]] = s[s[0] - 1] / MOD;
s[s[0] - 1] %= MOD;
}
}
void cut(Int x){
for(int i = 1; i <= x.s[0]; ++i)
s[i] -= x.s[i];
for(int i = 1; i <= x.s[0]; ++i){
if(s[i] < 0){
s[i] += MOD;
--s[i + 1];
}
}
if(!s[s[0]]) --s[0];
}
void print(){
printf("%d", s[s[0]]);
for(int i = s[0] - 1; i; --i)
printf("%04d", s[i]);
}
}a(1), b(1);
int main(){
scanf("%d%d", &n, &m);
for(int i = 2; i <= n + m; ++i){
if(!v[i]){
v[i] = i;
prime[++cnt] = i;
}
for(int j = 1; j <= cnt; ++j){
if(prime[j] > v[i] || prime[j] * i > n + m) break;
v[prime[j] * i] = prime[j];
}
}
for(int i = 1; i <= cnt; ++i)
for(int j = prime[i]; j <= n + m; j *= prime[i])
c[i] += (n + m) / j;
for(int i = 1; i <= cnt && prime[i] <= n; ++i)
for(int j = prime[i]; j <= n; j *= prime[i])
c[i] -= n / j;
for(int i = 1; i <= cnt && prime[i] <= m; ++i)
for(int j = prime[i]; j <= m; j *= prime[i])
c[i] -= m / j;
for(int i = 1; i <= cnt; ++i)
for(int j = 1; j <= c[i]; ++j)
a.mul(prime[i]);
for(int i = 1; i <= cnt; ++i)
for(int j = prime[i]; j <= n + m; j *= prime[i])
d[i] += (n + m) / j;
for(int i = 1; i <= cnt && prime[i] <= n + 1; ++i)
for(int j = prime[i]; j <= n + 1; j *= prime[i])
d[i] -= (n + 1) / j;
for(int i = 1; i <= cnt && prime[i] <= m - 1; ++i)
for(int j = prime[i]; j <= m - 1; j *= prime[i])
d[i] -= (m - 1) / j;
for(int i = 1; i <= cnt; ++i)
for(int j = 1; j <= d[i]; ++j)
b.mul(prime[i]);
a.cut(b);
a.print();
return 0;
}
【BZOJ 3907】网格(Catalan数)的更多相关文章
- BZOJ 3907: 网格 [Catalan数 高精度]
3907: 网格 Time Limit: 1 Sec Memory Limit: 256 MBSubmit: 402 Solved: 180[Submit][Status][Discuss] De ...
- BZOJ 3907: 网格
Description 求不跨过直线 \(y=x\) ,到达 \((n,m)\) 的方案数. Sol 组合数学+高精度. 这个推导过程跟 \(Catalan\) 数是一样的. 答案就是 \(C^{n+ ...
- bzoj 3907: 网格 组合数学
3907: 网格 Time Limit: 1 Sec Memory Limit: 256 MBSubmit: 13 Solved: 7[Submit][Status][Discuss] Descr ...
- BZOJ 3907: 网格( 组合数 + 高精度 )
(0,0)->(n,m)方案数为C(n,n+m), 然后减去不合法的方案. 作(n,m)关于y=x+1的对称点(m-1,n+1), 则(0,0)->(m-1,n+1)的任意一条路径都对应( ...
- bzoj 3907 网格 bzoj2822 [AHOI2012]树屋阶梯——卡特兰数(阶乘高精度模板)
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3907 https://www.lydsy.com/JudgeOnline/problem.p ...
- BZOJ 3907: 网格【组合数学】
Description 某城市的街道呈网格状,左下角坐标为A(0, 0),右上角坐标为B(n, m),其中n >= m.现在从A(0, 0)点出发,只能沿着街道向正右方或者正上方行走,且不能经过 ...
- [Catalan数三连]网格&有趣的数列&树屋阶梯
如何让孩子爱上打表 Catalan数 Catalan数是组合数学中一个常出现在各种计数问题中的数列. 以比利时的数学家欧仁·查理·卡塔兰 (1814–1894)的名字来命名. 先丢个公式(设第n项为$ ...
- BZOJ 1485: [HNOI2009]有趣的数列 [Catalan数 质因子分解]
1485: [HNOI2009]有趣的数列 Description 我们称一个长度为2n的数列是有趣的,当且仅当该数列满足以下三个条件: (1)它是从1到2n共2n个整数的一个排列{ai}: (2)所 ...
- BZOJ 1856: [Scoi2010]字符串 [Catalan数]
1856: [Scoi2010]字符串 Time Limit: 5 Sec Memory Limit: 64 MBSubmit: 1418 Solved: 790[Submit][Status][ ...
随机推荐
- 【数据库】 SQL 常用语句之系统语法
[数据库] SQL 常用语句之系统语法 1. 获取取数据库服务器上所有数据库的名字 SELECT name FROM master.dbo.sysdatabases 2. 获取取数据库服务器上所有非系 ...
- android service笔记
1.service 默认在主线程运行,所以不能在service中直接做访问网络,操作文件等耗时操作,要另外开启线程 2.通过startservice开启的服务,一旦服务开启,这个服务和开启他的调用者之 ...
- Vue学习(四):条件渲染
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- ASP NET Core ---REST & HTTP GET
参照 草根专栏- ASP.NET Core + Ng6 实战:https://v.qq.com/x/page/h0764n405ll.html 一.REST (Representational Sta ...
- LAMP架构应用实战—Apache服务介绍与安装01
LAMP架构应用实战—Apache服务介绍与安装01 一:Apache是什么 Apache是Apache基金会开发的一个高性能.功能强大.安全可靠.灵活的开放源码的WEB服务软件 二:Apache ...
- Win10下Pytorch的安装和使用[斗之力三段]
简介: 看到paper的代码是用Pytorch实现的,试图理解代码,但是看不懂,只能先学一些基础教程来帮助理解.笔记本电脑配置较低,所以安装一个没有CUDA的版本就可以了.安装完之后,就可以跟着教程边 ...
- LeetCode 25 —— K 个一组翻转链表
1. 题目 2. 解答 首先,利用快慢指针确定链表的总结点数. 偶数个结点时,结点个数等于 i * 2. 奇数个结点时,结点个数等于 i * 2 + 1. 然后将链表的每 K 个结点划分为一组.循环对 ...
- Hyperledger02
docker 思想 模块化: 集装箱 标准化: 运输标准化, 存储方式标准化,API接口的标准化 安全性: 隔离 docker解决什么问题 devops 我这程序程序没问题啊 系统好卡.哪个程序死循环 ...
- windows基础知识(win7)
右击 显示: 对设备进行管理: 在计算机属性中,开远程连接 控制面板: 控制面板下的操作中心: 控制面板下的管理工具: 控制面板下的默认程序: 控制面板下的日期时间: 控制面板下的鼠标: 控制面板下的 ...
- Week1 Team Homework #1 from Z.XML-总结学长经验教训
谭传奇学长: 我们的弯路可能是,一开始没有从最基础的部分开始迭代开发,一开始就想的太远了一些,每一步开的有点太大了,所以可能有些东西最后就连不上,也没有能够按时完成.如果可以先做出一个能用的版本,然后 ...