题目

输入格式

第一行,两个正整数 S 和 q,q 表示询问数量。

接下来 q 行,每行一个正整数 n。

输出格式

输出共 q 行,分别为每个询问的答案。

输入样例

30 3

9

29

1000000000000000000

输出样例

0

9

450000036

提示

对于100%的数据,2<=S<=2*106,1<=n<=1018,1<=q<=10^5

题解

DZY系列多神题

容易知道\(S\)所有质因子的指数最大为\(1\),否则结果都为\(0\)

如果满足,由\(S\)的范围可知其质因子最多有\(7\)个

那么\(n = \sum\limits_{i = 1}^{k} p_i * t_i\)

\(t_i\)表示第\(i\)个质因子选了几个

很像一个背包,但是\(n\)很大,考虑转化

我们先将\(n\)减去所有\(p_i\),保证至少选了一个

因为\(p_i\)是\(S\)的因子,所以可以写成\(p_i * t_i = Sx + p_iy\)且\([p_iy < S]\)

也就是分成若干个\(S\)和剩余不足\(S\)的部分

那么最终的\(n\)一定是由若干个前面部分的\(S\)和后面部分的\(p_iy\)相加而得

由于任意的\(p_iy < S\),所以\(\sum p_iy < k * S\),如果做背包,状态数为\(k^2 * S \approx 10^8\)

可以吧,做一个\(O(k * kS)\)的多重背包

看起来很汗,但可以跑过

至于这个多重背包的求法,就用一个类似滑动窗口的方法就可以实现\(O(k * kS)\)了

然后对于每个\(n\),枚举多出来的部分\(n \mod S + i * S\),\(0 \le i < 7\)

除了后面多出来的,前面的若干\(S\)要分配给那些质因子,用组合数挡板法即可

就可以\(O(7p)\)询问了

#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
#define LL long long int
#define Redge(u) for (int k = h[u],to; k; k = ed[k].nxt)
#define REP(i,n) for (int i = 1; i <= (n); i++)
#define BUG(s,n) for (int i = 1; i <= (n); i++) cout<<s[i]<<' '; puts("");
using namespace std;
const int maxn = 100005,maxm = 100005,INF = 1000000000,P = 1e9 + 7;
inline LL read(){
LL out = 0,flag = 1; char c = getchar();
while (c < 48 || c > 57){if (c == '-') flag = -1; c = getchar();}
while (c >= 48 && c <= 57){out = (out << 3) + (out << 1) + c - 48; c = getchar();}
return out * flag;
}
int f[15000000],g[15000000],inv[10];
int S,p[maxn],pi,sum;
LL N;
bool Sp(){
int x = S;
for (int i = 2; i * i <= x; i++)
if (x % i == 0){
int cnt = 0;
p[++pi] = i; sum += i;
while (x % i == 0) x /= i,cnt++;
if (cnt > 1) return true;
}
if (x - 1) p[++pi] = x,sum += x;
return false;
}
bool init(){
if (Sp()) return true;
f[0] = 1;
int M = pi * S;
for (int i = 1; i <= pi; i++){
memcpy(g,f,sizeof(f));
for (int j = 0; j < p[i]; j++)
{
LL w = 0;
for (int k = j; k <= M; k += p[i])
{
w = (w + g[k]) % P;
if (k - S >= 0) w = ((w - g[k - S]) % P + P) % P;
f[k] = w;
}
}
}
inv[0] = inv[1] = 1;
for (int i = 2; i < 10; i++) inv[i] = 1ll * (P - P / i) * inv[P % i] % P;
return false;
}
int cal(LL x,int y){
LL n = x + y - 1,m = y - 1;
int re = 1;
for (int i = 1; i <= m; i++)
re = 1ll * re * ((n - i + 1) % P) % P * inv[i] % P;
return re;
}
int main(){
S = read(); int T = read();
if (init()){
while (T--) puts("0");
return 0;
}
while (T--){
N = read();
N -= sum;
if (N < 0){puts("0"); continue;}
LL ans = 0,cnt = N / S;
for (int i = 0; i < pi && i <= cnt; i++)
ans = (ans + 1ll * f[N % S + i * S] * cal(cnt - i,pi) % P) % P;
printf("%lld\n",ans);
}
return 0;
}

BZOJ3462 DZY Loves Math II 【多重背包 + 组合数】的更多相关文章

  1. DZY Loves Math II:多重背包dp+组合数

    Description Input 第一行,两个正整数 S 和 q,q 表示询问数量.接下来 q 行,每行一个正整数 n. Output 输出共 q 行,分别为每个询问的答案. Sample Inpu ...

  2. [bzoj3462]DZY Loves Math II (美妙数学+背包dp)

    Description Input 第一行,两个正整数 S 和 q,q 表示询问数量. 接下来 q 行,每行一个正整数 n. Output 输出共 q 行,分别为每个询问的答案. Sample Inp ...

  3. BZOJ3462 DZY Loves Math II(动态规划+组合数学)

    容易发现这是一个有各种玄妙性质的完全背包计数. 对于每个质数,将其选取个数写成ax+b的形式,其中x=S/pi,0<b<x.那么可以枚举b的部分提供了多少贡献,多重背包计算,a的部分直接组 ...

  4. bzoj3462: DZY Loves Math II

    状态很差脑子不清醒了,柿子一直在推错.... ... 不难发现这个题实际上是一个完全背包 问题在于n太大了,相应的有质数的数量不会超过7个 假设要求sigema(1~plen)i pi*ci=n 的方 ...

  5. bzoj 3462: DZY Loves Math II

    3462: DZY Loves Math II Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 211  Solved: 103[Submit][Sta ...

  6. DZY Loves Math II

    简要题面 对于正整数 \(S, n\),求满足如下条件的素数数列 \((p_1,p_2,\cdots,p_k)\)(\(k\) 为任意正整数) 的个数: \(p_1\le p_2\le\cdots\l ...

  7. BZOJ 3462 DZY Loves Math II ——动态规划 组合数

    好题. 首先发现$p$是互质的数. 然后我们要求$\sum_{i=1}^{k} pi*xi=n$的方案数. 然后由于$p$不相同,可以而$S$比较小,都是$S$的质因数 可以考虑围绕$S$进行动态规划 ...

  8. DZY Loves Math系列

    link 好久没写数学题了,再这样下去吃枣药丸啊. 找一套应该还比较有意思的数学题来做. [bzoj3309]DZY Loves Math 简单推一下. \[\sum_{i=1}^n\sum_{j=1 ...

  9. [BZOJ] DZY Loves Math 系列 I && II

    为了让自己看起来有点事干 ,做个套题吧..不然老是东翻翻西翻翻也不知道在干嘛... \(\bf 3309: DZY \ Loves \ Math\) 令 \(h=f*\mu\) 很明显题目要求的就是\ ...

随机推荐

  1. Github的基本功能

    著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注明出处.作者:Fadeoc Khaos链接:http://www.zhihu.com/question/20070065/answer/30 ...

  2. Python判断一个数是否为小数

    一.判断一个数是否为小数 1.有且仅有一个小数点 2.小数点的左边可能为正数或负数 3.小数点的右边为正数 二.实现代码 def is_float(str): if str.count('.') == ...

  3. 阿里云服务器下安装LAMP环境(CentOS Linux 6.3) 安装与配置 Apache 服务

    想让我们的阿里云服务器成为一台 Web 服务器,我们需要安装一个 Web 服务器软件,比如 Apache ,或者 Nginx 等等.下面我们就一起来安装一个 Apache 服务. 我们可以使用 yum ...

  4. 高性能可扩展MySQL数据库设计及架构优化 电商项目(慕课)第3章 MySQL执行计划(explain)分析

    ID:相同就从上而下,不同数字越大越优先

  5. [BZOJ] 3875: [Ahoi2014&Jsoi2014]骑士游戏

    设\(f[x]\)为彻底杀死\(x\)号怪兽的代价 有转移方程 \[ f[x]=min\{k[x],s[x]+\sum f[v]\} \] 其中\(v\)是\(x\)通过普通攻击分裂出的小怪兽 这个东 ...

  6. flask-bootstrap

    pip install bootstarp 使用bower安装bootstrap的命令是: bash$ bower install bootstrap不过问题出在如何安装bower上. 官方网站上这样 ...

  7. 用户和用户组以及 Linux 权限管理

    1.从 /etc/passwd 说起 前面的基本命令学习中,我们介绍了使用 passwd 命令可以修改用户密码.对于操作系统来说,用户名和密码是存放在哪里的呢?我们都知道一个站点的用户名和密码是存放在 ...

  8. PHP 根据IP获取地理位置

    /** * 根据用户IP获取用户地理位置 * $ip 用户ip */ function get_position($ip){ if(empty($ip)){ return '缺少用户ip'; } $u ...

  9. python3 题目 有四个数字:1、2、3、4,能组成多少个互不相同且无重复数字的三位数?各是多少?

    方法一:for循环遍历 counter=0 for i in range(1,5): for j in range(1,5): for k in range(1,5): if i !=j and j ...

  10. 排列算法汇总(下一个排列,全排列,第K个排列)

    一.下一个排列 首先,STL提供了两个用来计算排列组合关系的算法,分别是next_permutation和prev_permutation. next_permutation(nums.begin() ...