原文链接 https://www.cnblogs.com/cly-none/p/SRM701Div1C.html

题意:定义"Fibonacci string"为没有连续1的01串。现在,给出\(a,b\),定义一个"Fibonacci string"的权值为\(x^a y^b\),其中\(x\)为0的个数,\(y\)为1的个数。

要求对所有长度为\(n\)的"Fibonacci string"的权值求和,对\(10^9 + 7\)取模。

\(n \leq 10^9, \ a, b \leq 25\)

显然第一反应就是矩阵存下所有\((a,b)\)做快速幂。然而,这样的话矩阵的边长是\(O(a^2)\)的,不能通过本题。

考虑最终答案的式子:
\[ \sum_{k=0}^n {n-k+1 \choose k} k^b (n-k)^a \]
我们尝试化简:
\[ \begin{aligned} & \sum_{k=0}^n {n-k+1 \choose k} k^b (n-k)^a \\ = & \sum_{k=0}^n \sum_{j=0}^a {n-k+1 \choose k} k^b {a\choose j} n^{a-j} (-k)^j \\ = & \sum_{j=0}^a {a \choose j} (-1)^j \sum_{k=0}^n {n-k+1 \choose k}k^{b+j} \end{aligned} \]
注意到后面的\(\sum_{k=0}^n {n-k+1 \choose k} k^{b+j}\)就是当\(a' = 0, \ b' = b+j\)时,所有长度为\(n\)的"Fibonacci string"的权值和。这时,我们再建矩阵,边长就只有\(O(a)\)了。最后\(O(a)\)枚举\(j\)就能计算出答案。

时间复杂度\(O(a^3 \log n)\)。

#include <bits/stdc++.h>
using namespace std;const int MOD = (int)(1e9 + 7), N = 110;
struct matrix {int n,m,mat[N][N];matrix(int n=0,int m=0): n(n), m(m) {memset(mat,0,sizeof mat);}matrix operator * (const matrix& a) const {assert(m == a.n);matrix ret = matrix(n, a.m);for (int k = 0 ; k < m ; ++ k)for (int i = 0 ; i < n ; ++ i)for (int j = 0 ; j < a.m ; ++ j)(ret.mat[i][j] += 1ll * mat[i][k] * a.mat[k][j] % MOD) %= MOD;return ret;}
};
matrix power(matrix a,int b) {assert(a.n == a.m);matrix ret = matrix(a.n, a.m);for (int k = 0 ; k < a.n ; ++ k)ret.mat[k][k] = 1;while (b) {if (b&1) ret = ret * a;a = a * a;b >>= 1;}return ret;
}
int power(int a,int b) {int ret = 1;while (b) {if (b&1) ret = 1ll * ret * a % MOD;a = 1ll * a * a % MOD;b >>= 1;}return ret;
}
class FibonacciStringSum {
public:int get( int n, int a, int b ) ;
};
int val[N],cmb[N][N];
int FibonacciStringSum::get(int n, int a, int b) {memset(cmb,0,sizeof cmb);for (int i = 0 ; i <= a + b ; ++ i)cmb[i][0] = 1;for (int i = 1 ; i <= a + b ; ++ i)for (int j = 1 ; j <= i ; ++ j)cmb[i][j] = (cmb[i-1][j] + cmb[i-1][j-1]) % MOD;matrix sta = matrix(1, 2 * (a + b + 1));matrix tran = matrix(2 * (a + b + 1), 2 * (a + b + 1));sta.mat[0][0] = 1;for (int i = 0 ; i <= a + b ; ++ i) {tran.mat[i][i] = 1;tran.mat[i + a + b + 1][i] = 1;for (int j = 0 ; j <= i ; ++ j)tran.mat[j][a + b + 1 + i] += cmb[i][j];}tran = power(tran, n);sta = sta * tran;for (int i = 0 ; i <= a + b ; ++ i)val[i] = (sta.mat[0][i] + sta.mat[0][i + a + b + 1]) % MOD;int ans = 0;for (int i = 0, t = 1 ; i <= a ; ++ i, t = -t)(ans += 1ll * t * cmb[a][i] * power(n, a - i) % MOD * val[b + i] % MOD) %= MOD;ans = (ans % MOD + MOD) % MOD;return ans;
}

小结:这个问题的特殊之处在于,既可以直接矩阵快速幂,也可以写成数学和式。然而,二者都不能直接解决这个问题。把两种方法相结合一直是常用的技巧(如分块),在这里也启示我们对于一个问题不能死板地但从一个方向来思考。

转载于:https://www.cnblogs.com/cly-none/p/SRM701Div1C.html

【做题】SRM701 Div1 Hard - FibonacciStringSum——数学和式&矩阵快速幂相关推荐

  1. BZOJ 2326: [HNOI2011]数学作业( 矩阵快速幂 )

    BZOJ先剧透了是矩阵乘法...这道题显然可以f(x) = f(x-1)*10t+x ,其中t表示x有多少位. 这个递推式可以变成这样的矩阵...(不会用公式编辑器...), 我们把位数相同的一起处理 ...

  2. 做题记录 Newcoder 逆序对 (组合、快速幂、快速乘

    题目描述 链接:https://ac.nowcoder.com/acm/contest/21763/1018 来源:牛客网 求所有长度为n的01串中满足如下条件的二元组个数: 设第i位和第j位分别位a ...

  3. NOIP模拟题 2016.11.15 [LIS] [spfa] [同余最短路] [矩阵快速幂] [容斥原理] [数学]

    小L的二叉树 [题目描述] 勤奋又善于思考的小L接触了信息学竞赛,开始的学习十分顺利.但是,小L对数据结构的掌握实在十分渣渣. 所以,小L当时卡在了二叉树. 在计算机科学中,二叉树是每个结点最多有两个 ...

  4. POJ3070矩阵快速幂简单题

    题意:       求斐波那契后四位,n <= 1,000,000,000. 思路:        简单矩阵快速幂,好久没刷矩阵题了,先找个最简单的练练手,总结下矩阵推理过程,其实比较简单,关键 ...

  5. HDU - 4990 Reading comprehension(矩阵快速幂,水题)

    题目链接:点击查看 题目大意:给出一段程序,进行优化后提交 题目分析:其实就是找规律,大水题一个,偶尔也是需要做做水题找找自信(逃) 先将题目中的程序拿下来,跑上100项,然后拿到oeis里找一下规律 ...

  6. [3.3训练赛]One-Dimensional(矩阵快速幂),Freda的迷宫(无向图强连通分量+并查集),一道防AK好题

    文章目录 T1:One-Dimensional title solution code T2:[NOIP模拟赛]Freda的迷宫 title solution code T3:[NOIP模拟赛]一道防 ...

  7. HDU 5863 cjj's string game ( 16年多校10 G 题、矩阵快速幂优化线性递推DP )

    题目链接 题意 : 有种不同的字符,每种字符有无限个,要求用这k种字符构造两个长度为n的字符串a和b,使得a串和b串的最长公共部分长度恰为m,问方案数 分析 : 直觉是DP 不过当时看到 n 很大.但 ...

  8. 15年第六届蓝桥杯第九题_(矩阵快速幂优化的动态规划)

    垒骰子 赌圣atm晚年迷恋上了垒骰子,就是把骰子一个垒在另一个上边,不能歪歪扭扭,要垒成方柱体. 经过长期观察,atm 发现了稳定骰子的奥秘:有些数字的面贴着会互相排斥! 我们先来规范一下骰子:1 的 ...

  9. 又见斐波那契~矩阵快速幂入门题

    链接:https://www.nowcoder.com/acm/contest/105/G 来源:牛客网 题目描述 这是一个加强版的斐波那契数列. 给定递推式 求F(n)的值,由于这个值可能太大,请对 ...

最新文章

  1. mysql每组随机一条_MySql分组后随机获取每组一条数据的操作
  2. 学号:201621123032 《Java程序设计》第6周学习总结
  3. VisualStudioCode 中设置中文语言【图文教程】
  4. Ubuntu 启动自动登录
  5. github的基础使用
  6. 关于使用pdf.js预览pdf的一些问题
  7. 《Windows PowerShell实战指南(第2版)》——1.4 搭建自己的实验环境
  8. C++笔记-基于邻接矩阵的BFS(宽度优先遍历)
  9. 实现CStack类遇到的问题
  10. 精度高、模型小、速度快!梯形DenseNets结构实现语义分割新高度!
  11. UML系列——OO Unit4分析和学期总结
  12. 用python画玫瑰花-python绘制玫瑰的实现代码
  13. 超酷Loading进度条
  14. Matlab中函数imnoise使用浅谈
  15. java语言介绍及特点分析(萌新入门须知内容)
  16. IPTABLES中SNAT和MASQUERADE的区别
  17. 【CSDN|每日一练】Longest Continuous Increasing Subsequence
  18. Java9中使用jpa,如何正确地在Spring Data JPA和Jackson中用上Java 8的时间相关API(即JSR 310也即java.time包下的众神器)...
  19. Java网络编程——基本网络支持
  20. 水果店的开业活动怎么做,水果店开业活动朋友圈怎么发

热门文章

  1. awk处理文件内容格式
  2. 接口有个电池标志_USB接口上的小标签有啥用?从“+”号到闪电的奥秘
  3. cron 工具 每分钟_计划任务 cron和crontab
  4. 1164: 分治 逆序对
  5. indesign图片规定在左下角_详解InDesign基本使用方法
  6. 软件自动测试框架,软件自动化测试框架的研究和实现
  7. mysql copy pending_mysql 案例 ~ 主从复制延迟之并行复制
  8. 电脑记事本在哪_【锦囊站第002期】电脑一秒内完成文件搜索是如何实现的?
  9. java intent 传递集合对象_Intent之对象传递(Parcelable传递对象和对象集合)
  10. java服务器gc停顿_如何避免后台IO高负载造成的长时间JVM GC停顿(转)