[SDOI 2013]方程
Description
求不定方程 \(x_1+x_2+\cdots +x_n=m\) 的正整数解的个数,并且要求满足限定: \(\forall i\in[1,n_1] x_i\leq a_i,\forall i\in[1,n_2] x_{n_1+i}\geq a_{n_1+i}\) 。对 \(p\) 取模, \(t\) 组询问。
\(n\leq 10^9,n_1\leq 8,n_2\leq 8,m\leq 10^9, p\leq 437367875,t\leq 5\)
Solution
如果没有约束,显然答案就是 \(C_{m-1}^{n-1}\) 。
对于第二种约束,显然直接在总数中减去就好。
考虑如何处理第一种约束,其实直接容斥就好了,处理方法类似于第二种约束。
注意到模数不一定为质数,还需要扩展 \(lucas\) 。
Code
#include <bits/stdc++.h>
using namespace std;
int n, n1, n2, p, t, a[10], ans, m;
int fac[200005], pi[100005], pk[100005], tot;
int quick_pow(int a, int b, int p) {
int ans = 1;
while (b) {
if (b&1) ans = 1ll*ans*a%p;
b >>= 1, a = 1ll*a*a%p;
}
return ans;
}
void ex_gcd(int a, int b, int &x, int &y) {
if (b == 0) {x = 1; y = 0; return; }
ex_gcd(b, a%b, x, y);
int t = x; x = y; y = t-a/b*y;
}
int inv(int a, int p) {
int x, y; ex_gcd(a, p, x, y);
return (x%p+p)%p;
}
int mul(int a, int pi, int pk) {
if (a <= pi) return fac[a];
int ans = fac[pk]; ans = quick_pow(ans, a/pk, pk);
ans = 1ll*ans*fac[a%pk]%pk;
return 1ll*ans*mul(a/pi, pi, pk)%pk;
}
int C(int n, int m, int pi, int pk) {
int t = 0;
for (int i = n; i; i /= pi) t += i/pi;
for (int i = m; i; i /= pi) t -= i/pi;
for (int i = n-m; i; i /= pi) t -= i/pi;
if (quick_pow(pi, t, pk) == 0) return 0;
fac[0] = 1; for (int i = 1; i <= pk; i++) if (i%pi) fac[i] = 1ll*i*fac[i-1]%pk; else fac[i] = fac[i-1];
int a = mul(n, pi, pk), b = mul(m, pi, pk), c = mul(n-m, pi, pk);
return 1ll*a*quick_pow(pi, t, pk)%pk*inv(b, pk)%pk*inv(c, pk)%pk;
}
int ex_lucas(int n, int m, int p) {
int ans = 0;
for (int i = 1; i <= tot; i++)
(ans += 1ll*C(n, m, pi[i], pk[i])*(p/pk[i])%p*inv(p/pk[i], pk[i])%p) %= p;
return ans;
}
void dfs(int c, int r, int f) {
if (c == n1+1) {
if (r < n) return;
(ans += ex_lucas(r-1, n-1, p)*f) %= p; return;
}
dfs(c+1, r, f); dfs(c+1, r-a[c], -f);
}
void work() {
scanf("%d%d", &t, &p);
int T = p;
for (int i = 2, x = sqrt(T); i <= x; i++)
if (T%i == 0) {
int tol = 1; while (T%i == 0) tol *= i, T /= i;
pi[++tot] = i, pk[tot] = tol;
}
if (T != 1) pi[++tot] = pk[tot] = T;
while (t--) {
scanf("%d%d%d%d", &n, &n1, &n2, &m);
for (int i = 1; i <= n1; i++) scanf("%d", &a[i]);
for (int i = 1; i <= n2; i++) {scanf("%d", &T); if (T) m -= T-1; }
ans = 0; dfs(1, m, 1); printf("%d\n", (ans+p)%p);
}
}
int main() {work(); return 0; }
[SDOI 2013]方程的更多相关文章
- [BZOJ 3123] [SDOI 2013]森林(可持久化线段树+并查集+启发式合并)
[BZOJ 3123] [SDOI 2013]森林(可持久化线段树+启发式合并) 题面 给出一个n个节点m条边的森林,每个节点都有一个权值.有两种操作: Q x y k查询点x到点y路径上所有的权值中 ...
- BZOJ 3203 sdoi 2013 保护出题人
由于样例解释很清晰,所以很容易得到以下结论: 1.每一关都是独立的,且僵尸的相对位置不会变 2.每一关的攻击力=Max(sum(i)/dis(i)) 其实sum(i)是僵尸攻击力的前缀和,dis(i) ...
- [SDOI 2013]森林
Description 题库链接 给你 \(n\) 个节点,初始 \(m\) 条边, \(t\) 组操作.让你支持: 询问树上路径点权 \(K\) 小: 支持加边操作. 强制在线,所有状态保证是一个树 ...
- 解题:SDOI 2013 保护出题人
题面 首先是愉快的推式子 $dp[i]=max(dp[i],\frac{sum[i]-sum[j-1]}{x[i]+(i-j)*d})(1<=j<=i<=n)$(考虑有一只僵尸正好走 ...
- 「BZOJ 3123」「SDOI 2013」森林「启发式合并」
题意 你有一个森林,你需要支持两个操作 查询两个结点路径上权值第\(k\)小 两个点之间连一条边 强制在线,结点数\(\leq 8\times 10^4\) 题解 如果可以离线,这就是一个主席树板子题 ...
- bzoj 3202 [Sdoi 2013] 项链 —— 置换+计数
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=3202 参考了博客: https://www.cnblogs.com/zhoushuyu/p/ ...
- [SDOI 2013] 直径
[题目链接] https://www.lydsy.com/JudgeOnline/problem.php?id=3124 [算法] 树的直径 [代码] #include<bits/stdc++. ...
- [日常摸鱼]bzoj3122 [Sdoi]2013 随机数生成器
又是写了一晚上才过的题- 题意:有一个数列$x_n=(ax_{n-1}+b) mod p$,给你$x_1,a,b,p,t$,求最小的$x_i=t$的$i$,可能不存在 一开始很自然的推出了式子$x_n ...
- HNOI 2012 永无乡
codevs 1477 永无乡 http://codevs.cn/problem/1477/ 2012年湖南湖北省队选拔赛 时间限制: 1 s 空间限制: 128000 KB 题目描述 Des ...
随机推荐
- 2018上C语言程序设计(高级)- 第0次作业成绩
作业链接: https://edu.cnblogs.com/campus/hljkj/CS201702/homework/1617 评分规则 本次作业作为本学期的第一次作业,大家态度较诚恳,篇幅都比较 ...
- C语言程序设计(基础)- 第7周作业(新)
要求一(25经验值) 完成PTA中题目集名为<usth-C语言基础-第七周作业>和<usth-C语言基础-12周PTA作业>中的所有题目. 注意1:<usth-C语言基础 ...
- python 进程复习
import os import time ret = os.fork() # 创建子线程 if ret ==0: # 子进程中返回值为0,父进程>0 while True: print('.. ...
- C++中文件的读写
C++中文件的读写 在C++中如何实现文件的读写? 一.ASCII 输出 为了使用下面的方法, 你必须包含头文件<fstream.h>(译者注:在标准C++中,已经使用<fstrea ...
- windows系统下安装 node.js (node.js安装及环境配置)
node.js简介 Node.js 是一个基于 Chrome V8 引擎的 JavaScript 运行环境. Node.js 使用了一个事件驱动.非阻塞式 I/O 的模型,使其轻量又高效. Node. ...
- python 面向对象之继承与派生
一:初识继承 1,什么是继承? 继承指的是类与类之间的关系,是一种什么"是"什么的关系,继承的功能之一就是用来解决代码重用问题 继承是一种创建新类的方式,在python中,新建的类 ...
- react中的DOM操作
前面的话 某些情况下需要在典型数据流外强制修改子代.要修改的子代可以是 React 组件实例,也可以是 DOM 元素.这时就要用到refs来操作DOM 使用场景 下面是几个适合使用 refs 的情况 ...
- 易错点---所有的字符都自带bool值
所有的字符都自带布尔值,只有0,None,空为False,其他全部为真!!!!!!!!!!! count = 0 while count < 3 : inp_age =input('Enter ...
- PC或者手机弹出窗效果
http://layer.layui.com/ 这个网站提供弹窗,是在jq封装的,弹窗之后,背景页面还可以滑动. 这个里面的js可能也会包含css,这个css不能移动位置,否则会报错,还有谷歌浏览器在 ...
- Python内置函数(45)——ascii
英文文档: ascii(object) As repr(), return a string containing a printable representation of an object, b ...