题目链接:公式题 (2)

比赛链接:华东交通大学2018年ACM“双基”程序设计竞赛

题目描述

令f(n)=2f(n-1)+3f(n-2)+n,f(1)=1,f(2)=2

令g(n)=g(n-1)+f(n)+n*n,g(1)=2

告诉你n,输出g(n)的结果,结果对1e9+7取模

输入描述:

多组输入,每行一个整数n(1<=n<=1e9),如果输入为0,停止程序。

输出描述:

在一行中输出对应g(n)的值,结果对1e9+7取模。

示例1

输入

1
5
9
456
0

输出

2
193
11956
634021561

说明

多组输入,输入为0时,终止程序

备注:

项数极大,朴素算法无法在规定时间内得出结果

题解

矩阵快速幂

\[\left[
\begin{matrix}
g(n) \\
f(n) \\
f(n - 1) \\
n ^ 2 \\
n \\
1 \\
\end{matrix}
\right]
=
\left[
\begin{matrix}
1 & 2 & 3 & 1 & 3 & 2 \\
0 & 2 & 3 & 0 & 1 & 1 \\
0 & 1 & 0 & 0 & 0 & 0 \\
0 & 0 & 0 & 1 & 2 & 1 \\
0 & 0 & 0 & 0 & 1 & 1 \\
0 & 0 & 0 & 0 & 0 & 1 \\
\end{matrix}
\right]
\left[
\begin{matrix}
g(n - 1) \\
f(n - 1) \\
f(n - 2) \\
(n - 1) ^ 2 \\
n - 1 \\
1 \\
\end{matrix}
\right]
\]

代码

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn = 10 + 5;
const ll mod = 1e9 + 7; struct Matrix {
int n, m;
ll a[maxn][maxn];
Matrix(int n = 0, int m = 0) : n(n), m(m) {}
void input() {
for(int i = 1; i <= n; ++i) {
for(int j = 1; j <= m; ++j) {
scanf("%lld", &a[i][j]);
}
}
}
void output() {
for(int i = 1; i <= n; ++i) {
for(int j = 1; j <= m; ++j) {
printf("%lld", a[i][j]);
printf("%s", j == m? "\n": " ");
}
}
}
void init() {
memset(a, 0, sizeof(a));
}
void unit() {
if(n == m) {
init();
for(int i = 1; i <= n; ++i) {
a[i][i] = 1;
}
}
}
Matrix operator *(const Matrix b) {
Matrix c(n, b.m);
c.init();
for(int i = 1; i <= c.n; ++i) {
for(int k = 1; k <= m; ++k) {
for(int j = 1; j <= c.m; ++j) {
c.a[i][j] = (c.a[i][j] + a[i][k] * b.a[k][j]) % mod;
}
}
}
return c;
}
Matrix qmod(ll b) {
if(n == m) {
Matrix a = *this;
Matrix ans = Matrix(n, n);
ans.unit();
if(!b) return ans;
while(b) {
if(b & 1) ans = ans * a;
a = a * a;
b >>= 1;
}
return ans;
}
}
}; int main() {
int n;
while(~scanf("%d", &n) && n) {
if(n == 1) printf("2\n");
else if(n == 2) printf("8\n");
else {
Matrix m(6, 6);
m.init();
m.a[1][1] = 1; m.a[1][2] = 2; m.a[1][3] = 3; m.a[1][4] = 1; m.a[1][5] = 3; m.a[1][6] = 2;
m.a[2][2] = 2; m.a[2][3] = 3; m.a[2][5] = 1; m.a[2][6] = 1;
m.a[3][2] = 1;
m.a[4][4] = 1; m.a[4][5] = 2; m.a[4][6] = 1;
m.a[5][5] = 1; m.a[5][6] = 1;
m.a[6][6] = 1;
Matrix ans(6, 1);
ans.a[1][1] = 8;
ans.a[2][1] = 2;
ans.a[3][1] = 1;
ans.a[4][1] = 4;
ans.a[5][1] = 2;
ans.a[6][1] = 1;
ans = (m.qmod(n - 2)) * ans;
printf("%lld\n", ans.a[1][1]);
}
}
return 0;
}

最新文章

  1. 学习笔记_springmvc返回值、数据写到页面、表单提交、ajax、重定向
  2. SVN配置管理(trunk、branches、tags)
  3. 程序设计第二次作业&lt;1&gt;
  4. c#部分---好题--顺便练练“类的知识”
  5. mysql约束(自己原先总结的有点不准)
  6. 高质量、处于持续更新的R包
  7. ASP.NET网站怎么发布 Web项目程序怎么发布部署(暂时收藏)
  8. javascript单例模式(懒汉 饿汉)
  9. 禁止root远程登录
  10. Android ART
  11. Loadrunner常见错误处理方法
  12. 开始你的第一个npm脚本工具
  13. c++中成员函数的参数名与成员变量名重合的问题
  14. RSAC 2018:人工智能成为驱动网络安全的新 “引擎”
  15. 关于陌生的依赖模块,如withStyles、react-apollo等
  16. c#异步编程(三)—ASP.NET MVC 异步控制器及EF异步操作
  17. CRM UI 打印PDF
  18. semi-consistent简介
  19. ubuntu中设置xampp开机启动
  20. 使用SQL语句将数据库中的两个表合并成一张表

热门文章

  1. python的代码块缓存机制,小数据池机制。
  2. 安装beanstalkd - centos
  3. mysql注入篇
  4. java做题笔记
  5. 洛谷 P2756 飞行员配对方案问题 (二分图匹配)
  6. Java随机访问文件
  7. linux 服务器修改密码
  8. linux 挂载iso文件,挂载ntfs文件系统
  9. jq-demo-点击改变图片
  10. java web应用用户上传图片的存储地址