【SDOI2008】【BZOJ3231】递归数列

Description

一个由自然数组成的数列按下式定义:

对于i <= k:ai = bi

对于i > k: ai = c1ai-1 + c2ai-2 + … + ckai-k

其中bj和 cj (1<=j<=k)是给定的自然数。写一个程序,给定自然数m <= n, 计算am + am+1 + am+2 + … + an, 并输出它除以给定自然数p的余数的值。

Input

由四行组成。

第一行是一个自然数k。

第二行包含k个自然数b1, b2,…,bk。

第三行包含k个自然数c1, c2,…,ck。

第四行包含三个自然数m, n, p。

Output

仅包含一行:一个正整数,表示(am + am+1 + am+2 + … + an) mod p的值。

Sample Input

2

1 1

1 1

2 10 1000003

Sample Output

142

HINT

对于100%的测试数据:

1<= k<=15

1 <= m <= n <= 1018

Source

矩乘+快速幂

把那一段和式看成两个前缀和相减就可以用矩乘了.

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
#define LL long long
#define MAXN 20
using namespace std;
int K;
LL b[20],c[20];
LL m,n,p;
LL ans;
struct matrix
{
    LL a[MAXN][MAXN];
    matrix()
    {
        memset(a,0,sizeof(a));
    }
    friend matrix operator *(matrix A,matrix B)
    {
        matrix ret;
        for (int i=1;i<=K+1;i++)
            for (int j=1;j<=K+1;j++)
                for (int k=1;k<=K+1;k++)
                    ret.a[i][j]=(ret.a[i][j]+A.a[i][k]*B.a[k][j])%p;
        return ret;
    }
    friend matrix operator ^(matrix x,LL k)
    {
        matrix ret;
        for (int i=1;i<=K+1;i++)    ret.a[i][i]=1;
        for (LL i=k;i;i>>=1,x=x*x)
        {
            if (i&1)    ret=ret*x;
        }
        return ret;
    }
}tmp1,tmp2;
LL solve(LL x)
{
    if (!x) return tmp2.a[1][K+1];
    matrix T=tmp1^x;
    T=tmp2*T;
    return T.a[1][K+1];
}
int main()
{
    scanf("%d",&K);
    for (int i=1;i<=K;i++)  scanf("%lld",&b[i]);
    for (int i=1;i<=K;i++)  scanf("%lld",&c[i]);
    scanf("%lld%lld%lld",&m,&n,&p);
    for (int i=1;i<=K;i++)  b[i]%=p,c[i]%=p,b[K+1]+=b[i],b[K+1]%=p;
    for (int i=1;i<=K;i++)  tmp1.a[i][1]=tmp1.a[i][K+1]=c[i];
    for (int i=2;i<=K;i++)  tmp1.a[i-1][i]=1;
    tmp1.a[K+1][K+1]=1;
    for (int i=1;i<=K;i++)  tmp2.a[1][i]=b[K-i+1];
    tmp2.a[1][K+1]=b[K+1];
    if (n<=K)
    {
        for (int i=m;i<=n;i++)  ans+=b[i],ans%=p;
        printf("%lld\n",ans);
        return 0;
    }
    ans=solve(n-K);
    if (m<=K)
        for (int i=1;i<m;i++)   ans-=b[i];
    else    ans-=solve(m-K-1);
    ans=(ans+p)%p;
    cout<<ans<<endl;
}

版权声明:本文为博主原创文章,未经博主允许不得转载。

时间: 2024-08-01 03:00:01

【SDOI2008】【BZOJ3231】递归数列的相关文章

review 9.29 viv 逃命 &amp; 递归数列

T1 逃命 survive Time Limit: 1 Sec  Memory Limit: 32768 K Description 糟糕的事情发生啦,现在大家都忙着逃命.但是逃命的通道很窄,大家只能排成一行. 现在有n个人,从1标号到n.同时有一些奇怪的约束条件,每个都形如:a必须在b之前. 同时,社会是不平等的,这些人有的穷有的富.1号最富,2号第二富,以此类推.有钱人就贿赂负责人,所以他们有一些好处. 负责人现在可以安排大家排队的顺序,由于收了好处,所以他要让1号尽量靠前,如果此时还有多种

【bzoj3231】[Sdoi2008]递归数列 矩阵乘法+快速幂

题目描述 一个由自然数组成的数列按下式定义: 对于i <= k:ai = bi 对于i > k: ai = c1ai-1 + c2ai-2 + ... + ckai-k 其中bj和 cj (1<=j<=k)是给定的自然数.写一个程序,给定自然数m <= n, 计算am + am+1 + am+2 + ... + an, 并输出它除以给定自然数p的余数的值. 输入 由四行组成. 第一行是一个自然数k. 第二行包含k个自然数b1, b2,...,bk. 第三行包含k个自然数c1,

[luogu2461 SDOI2008] 递归数列 (矩阵乘法)

传送门 Description 一个由自然数组成的数列按下式定义: 对于i <= k:ai = bi 对于i > k: ai = c1ai-1 + c2ai-2 + ... + ckai-k 其中bj 和 cj (1<=j<=k)是给定的自然数.写一个程序,给定自然数m <= n, 计算am + am+1 + am+2 + ... + an, 并输出它除以给定自然数p的余数的值. Input 输入文件spp.in由四行组成. 第一行是一个自然数k. 第二行包含k个自然数b1,

bzoj 3231: [Sdoi2008]递归数列【矩阵乘法】

今天真是莫名石乐志 一眼矩阵乘法,但是这个矩阵的建立还是挺有意思的,就是把sum再开一列,建成大概这样 然后记!得!开!long!long!! #include<iostream> #include<cstdio> using namespace std; const int N=20; long long n,b[N],c[N],sum,l,r,mod; struct jz { long long a[N][N]; jz operator * (const jz &b)

[SDOI2008]递归数列

嘟嘟嘟 裸的矩阵快速幂,构造一个\((k + 1) * (k + 1)\)的矩阵,把sum[n]也放到矩阵里面就行了. #include<cstdio> #include<iostream> #include<cmath> #include<algorithm> #include<cstring> #include<cstdlib> #include<cctype> #include<vector> #incl

证明一个递归数列极限的存在

If ${x_{n + 1}} = \cos {x_n}$, prove that $\mathop {\lim }\limits_{n \to \infty } {x_n}$ exists. Note 1: As we want to prove that a limit of sequence exist, some methods can be used. $\left\{ {{x_n}} \right\}$ is bounded and monotonic, then $\left\{

递归法

递归法(Recursion)是一种在函数或方法中调用自身的编程技术,在计算机方法中,使用递归技术往往使函数的定义和算法的描述简洁且易于理解.任何可以用计算机求解的问题所需要的计算时间都与其规模有关.而且规模越小,解题所需要的计算时间通常越小,从而比较容易处理. 简而言之,递归思想就是用与自身问题相似但规模较小的问题来描述自己. 例如,兔子出生两个月后就有繁殖能力,一对兔子每个月能生出一对兔子来.如果所有兔子都不死,那么一年以后可以繁殖多少对兔子? 第一个月小兔子没有繁殖能力,所有还是一对:两个月

Python yield 使用浅析 ----以裴波那契数列生成为例

转载请注明出处,http://blog.csdn.net/suool/article/details/38388681谢谢! 之前转过一篇讲Python生成器的博文,不过觉得不是很清晰,前几天见到有讲yield的文章,感觉还不错,遂整理于此,与诸君共享.愿共同进步. 我们先抛开 generator,以一个常见的编程题目来展示 yield 的概念. 如何生成斐波那契數列 斐波那契(Fibonacci)數列是一个非常简单的递归数列,除第一个和第二个数外,任意一个数都可由前两个数相加得到.用计算机程序

python学习之函数(四)--递归

1.什么是递归:递归比迭代高级: 在递归函数中调用自己,如下,会陷入死循环: def recursion(): return recursion()小知识:python3的递归层深是100层,可以更改递归层深import syssys.setrecursionlimit(10000) 用非递归方式实现阶乘: def factorail(n): result = n for i in range(1,n): result *= i return resultprint (factorail(5))