@codeforces - 717A@ Festival Organization
@description@
一个长度为 n 的 01 序列是好的,当且仅当该序列任意两个 0 不相邻。
求从 [l, r] 中选出 k 个长度相等的 01 序列的方案数。
1 ≤ k ≤ 200, 1 ≤ l ≤ r ≤ 10^18。
@solution@
定义 f[i] 表示合法 01 串数量,对最后一个是 1 还是 0 进行讨论得到 f[i] = f[i-1] + f[i-2]。
递推式有斐波那契数列形式,但注意 f[1] = 2 所以不完全是斐波那契,不过可以通过移位变成斐波那契。
以下都把它当作斐波那契数列来解决。
则问题相当于问 \(\sum_{i=l}^{r}C_{f_i}^k = \sum_{i=0}^{r}C_{f_i}^k - \sum_{i=0}^{l-1}C_{f_i}^k\)。
可以通过第一类斯特林数化成 \(\sum_{i=0}^{n}C_{f_i}^k = \frac{1}{k!}\sum_{i=0}^{n}(\sum_{j=0}^{k}S_{k}^{j}*f_{i}^j*(-1)^{k-j})\)。
问题可以等价于求 \(\sum_{i=0}^{n}f_{i}^p\)。当 p = 1, 2 的时候都可以搜到相应的式子。是否 p 增大也有相应的式子呢?
除了把递推式写成矩阵以外,斐波那契还有一个常见解决方法:通项公式。众所周知,斐波那契的通项公式可以写作 \(f[i] = a*A^i + b*B^i\)。
注意通项公式中含有 \(\sqrt{5}\),而该模数下 5 没有二次剩余。所以需要类似复数,将数存成 \((a + b*\sqrt{5})\) 的形式。
然后直接把通项公式往里面代:
\\ = \sum_{i=0}^{n}\sum_{j=0}^{p}C_p^j*(a*A^i)^j*(b*B^j)^{p-j}
\\ = \sum_{j=0}^{p}C_p^j*a^j*b^{p-j}\sum_{i=0}^{n}(A^j*B^{p-j})^i\]
等比数列求和解后面的式子即可。
最后复杂度 O(k^2logA)(还有等比数列快速幂的复杂度)。
@accepted code@
#include <cstdio>
#include <algorithm>
using namespace std;
typedef long long ll;
const int MAXK = 200;
const int MOD = int(1E9) + 7;
struct mint{
int x, y;
mint(int _x=0, int _y=0) : x(_x), y(_y) {}
friend mint operator + (mint a, mint b) {
int x = (a.x + b.x >= MOD ? a.x + b.x - MOD : a.x + b.x);
int y = (a.y + b.y >= MOD ? a.y + b.y - MOD : a.y + b.y);
return mint(x, y);
}
friend mint operator - (mint a, mint b) {
int x = (a.x - b.x < 0 ? a.x - b.x + MOD : a.x - b.x);
int y = (a.y - b.y < 0 ? a.y - b.y + MOD : a.y - b.y);
return mint(x, y);
}
friend mint operator * (mint a, mint b) {
int x = (1LL*a.x*b.x + 5LL*a.y*b.y) % MOD;
int y = (1LL*a.x*b.y + 1LL*a.y*b.x) % MOD;
return mint(x, y);
}
friend mint mpow(mint a, ll k) {
mint r = 1;
while( k ) {
if( k & 1 ) r = r * a;
a = a * a;
k >>= 1;
}
return r;
}
friend mint operator / (mint a, mint b) {
mint p = a * mint(b.x, (MOD - b.y)%MOD), q = mint((1LL*b.x*b.x%MOD + MOD - 5LL*b.y*b.y%MOD)%MOD, 0);
return p * mpow(q, MOD-2);
}
}s[MAXK + 5][MAXK + 5], c[MAXK + 5][MAXK + 5];
const mint A1 = 1/mint(0,1);
const mint A2 = (1 + mint(0,1))/2;
const mint B1 = 1/mint(0,MOD-1);
const mint B2 = (1 - mint(0,1))/2;
void init() {
for(int i=0;i<=MAXK;i++) {
c[i][0] = 1;
for(int j=1;j<=i;j++)
c[i][j] = c[i-1][j] + c[i-1][j-1];
}
for(int i=0;i<=MAXK;i++)
for(int j=0;j<=i;j++)
c[i][j] = c[i][j] * mpow(A1, j) * mpow(B1, i-j);
s[0][0] = 1;
for(int i=1;i<=MAXK;i++)
for(int j=1;j<=i;j++)
s[i][j] = s[i-1][j-1] + (i-1)*s[i-1][j];
for(int i=0;i<=MAXK;i++)
for(int j=0;j<=MAXK;j++)
if( (i + j) & 1 ) s[i][j] = 0 - s[i][j];
}
int k;
mint get(mint a, ll m) {
if( a.x == 1 && a.y == 0 ) return m%MOD + 1;
else return (mpow(a, m + 1) - 1) / (a - 1);
}
mint solve(ll m) {
mint ans = 0;
for(int j=0;j<=k;j++) {
mint del = 0;
for(int p=0;p<=j;p++) {
mint tmp = mpow(A2, p) * mpow(B2, j-p);
del = del + c[j][p] * get(tmp, m);
}
ans = ans + s[k][j] * del;
}
return ans;
}
int main() {
ll l, r; init();
scanf("%d%lld%lld", &k, &l, &r), l += 2, r += 2;
mint p = 1; for(int i=1;i<=k;i++) p = p*i;
printf("%d\n", ((solve(r) - solve(l-1))/p).x);
}
@details@
注意等比数列特判公比为 1 !!!(老是记不住)
貌似是 BJOI2019 的勘破神机?
@codeforces - 717A@ Festival Organization的更多相关文章
- CF 717A Festival Organization——斯特林数+递推求通项+扩域
题目:http://codeforces.com/contest/717/problem/A 是 BJOI2019 勘破神机 的弱化版. 令 \( g[i] \) 表示长为 i .以 1 结尾的方案数 ...
- CF717 Festival Organization
\(CF717\ Festival\ Organization\) Description 一个合法的串定义为:长度在 \([l,r]\) 之间,且只含 \(0,1\),并且不存在连续 \(2\) 个 ...
- CF717A Festival Organization(第一类斯特林数,斐波那契数列)
题目大意:求 $\sum\limits_{n=l}^{r}\dbinom{f_n}{k}\bmod 10^9+7$.其中 $f_n$ 是长度为 $n$ 的 $01$ 序列中,没有连续两个或超过两个 $ ...
- Codeforces Round #127 (Div. 1) E. Thoroughly Bureaucratic Organization 二分 数学
E. Thoroughly Bureaucratic Organization 题目连接: http://www.codeforces.com/contest/201/problem/E Descri ...
- codeforces 340C Tourist Problem
link:http://codeforces.com/problemset/problem/340/C 开始一点也没思路,赛后看别人写的代码那么短,可是不知道怎么推出来的啊! 后来明白了. 首先考虑第 ...
- codeforces magic five --快速幂模
题目链接:http://codeforces.com/contest/327/problem/C 首先先算出一个周期里面的值,保存在ans里面,就是平常的快速幂模m做法. 然后要计算一个公式,比如有k ...
- Codeforces Round #219 (Div. 2) E. Watching Fireworks is Fun
http://codeforces.com/contest/373/problem/E E. Watching Fireworks is Fun time limit per test 4 secon ...
- python爬虫学习(5) —— 扒一下codeforces题面
上一次我们拿学校的URP做了个小小的demo.... 其实我们还可以把每个学生的证件照爬下来做成一个证件照校花校草评比 另外也可以写一个物理实验自动选课... 但是出于多种原因,,还是绕开这些敏感话题 ...
- 【Codeforces 738D】Sea Battle(贪心)
http://codeforces.com/contest/738/problem/D Galya is playing one-dimensional Sea Battle on a 1 × n g ...
随机推荐
- Scrapy下载器中间件实现随机请求头和代理ip
一.设置随机请求头 class UAMiddleWare(object): UA_LIST = [ 'Mozilla/5.0 (Macintosh; U; Intel Mac OS X 10_6_8; ...
- js 打开app应用,如果没有安装就去下载
废话不多说,直接上代码 var APPCommon = { iphoneSchema: 'XingboTV://', iphoneDownUrl: 'https://itunes.apple.com/ ...
- --1.plsql中学习job
--1.plsql中学习job --学习job --建表 create table test_job(para_date date); commit; insert into test_job val ...
- C# 制作ActiveX控件并添加到网页
1.创建ActiveX控件——按钮 2.定义一个接口,并在控件中实现 3.部署安装 4.CAB打包 5.添加到网页中进行测试 一. 创建ActiveX控件——按钮 1.新建一个Window窗体控件库项 ...
- Ubuntu 链接ln的使用:创建和删除符号链接
一 . 使用方式 ln [option] source_file dist_file (source_file是待建立链接文件的文件,dist_file是新创建的链接文件) -f 建立时,将同档案名删 ...
- 【python之路40】Python 作用域
python的作用域与javaScript是一样的,参考:http://www.cnblogs.com/sunshuhai/p/9112578.html 一.python是以函数作为作用域的 if 1 ...
- 我的常用vs code 插件
换了台电脑重新装上了VS CODE,但是用起来后发现非常不顺手,突然醒悟原来还没有装上插件. 正动手装插件,但又一脸茫然了,我以前都装了些什么插件来着?因为平时根本不会去几插件的名字啊,只能靠搜搜一些 ...
- [转]js设计模式-策略模式
在程序设计中,常常遇到类似的情况,要实现某一个功能有多种方案可以选择.比如一个压缩文件的程序,既可以选择zip算法,也可以选择gzip算法.这些算法灵活多样,而且可以随意互相替换.这种解决方案就是本文 ...
- 51nod1947 栈的代价和
1947 栈的代价和 n是5e7 只能O(n)做 大力生成函数转形式幂级数再解方程 这个是广义二项式定理: https://baike.baidu.com/item/%E4%BA%8C%E9%A1%B ...
- 锋利的jQuery学习笔记之jQuery选择器
在介绍jQuery选择器之前,先简单介绍一下CSS选择器---> 一.CSS选择器 常见的CSS选择器有以下几种: 选择器 语法 描述 示例 标签选择器 E{CSS规则} 以文档元素为选择符 t ...