【BZOJ 2323】 2323: [ZJOI2011]细胞 (DP+矩阵乘法+快速幂*)

2323: [ZJOI2011]细胞

Description

2222年,人类在银河系外的某颗星球上发现了生命,并且携带了一个细胞回到了地球。经过反复研究,人类已经完全掌握了这类细胞的发展规律:

这种细胞最初的形态是“长条形”,一端是头,一端是尾,中间是躯干。细胞内部含有一列密码(你可以认为它是这种细胞的DNA)。密码是一个长度为n的数字串,且仅含有1~9这9种数字,沿着细胞的躯干从头到尾排列着。

首先,细胞会经历一次分裂。细胞将沿躯干方向分裂成若干个球体,躯干将退化成丝状物,连接着相邻的球体。在分裂过程中,质量是均匀分布的。换句话说,若分裂成k个球体,每个球体的质量为原来的1/k。然而,密码的分布是不确定的。若分割成k个球体,密码会被切割成k段(每段长度至少为1),并按从头到尾的顺序分布在各个球体中。如图,为其中一种合法的一次分裂:

接下来,细胞会经历二次分裂。对于每个球体,其中会含有一小段密码(注意他是有序的),我们把它看作一个十进制的数T。这个球体会被分割成T个小球体,并排成一排,之间用躯干退化成的丝状物相连接,并且质量仍然是均匀分布的,每个小球体的质量都是原球体的1/T。至此,密码已经发挥了它的作用,便消失了。如图,为二次分裂:

最后,细胞会进行变异。相邻小球体之间的丝状物可能会退化掉,这两个小球体便会以相切的方式直接连接。显然,二次分裂后,除两端外的每个小球体都有两段丝状物与其连接(头尾两端的小球体只有一段丝状物与其相连)。对于每个小球体,必须至少退化一段与其相连的丝状物,否则这个结构不稳定,会继续变异。如图,为一种稳定的变异:

现在,我们想知道,对于一个给定密码的细胞,总共有多少种稳定的结构。两种结构被认为相同,当且仅当他们拥有相同个数的小球体,从头到尾每个小球体的质量相同,并且从头到尾每对相邻小球体之间的连接方式相同(都是通过丝状物相连或都是通过相切直接相连)。你只需要回答这个结果 mod 1000000007即可。

Input

第一行为一个正整数n,表示细胞密码的长度。

第二行共n个数字,为给定的细胞密码,中间没有空格。

Output

只包含一个整数,为细胞的种数 mod 1000000007的结果。

Sample Input

【样例输入一】

1

1

【样例输入二】

1

5

【样例输入三】

2

11

Sample Output

【样例输出一】

0

【样例输出二】

3

【样例输出三】

56

HINT

【数据规模】

对于5%的数据满足,n ≤ 6;

对于25%的数据满足,n ≤ 25;

对于60%的数据满足,n ≤ 100;

对于70%的数据满足,n ≤ 300;

对于100%的数据满足,n ≤ 1 000。

Source

Day2

【分析】

  好题?

  感谢数学试卷让我发现了这个可爱的斐波那契数列。。【hhh

  然后【矩阵的指数不能mod phi[p]? 。。被奥爷爷d了。。。。

  首先,第一步或第三步不同,则出来的结果一定不同【自己想为什么吧、、我就是这样觉得的

  对于第三步,他说一个球两边一定有一边被缩了,其实就是n条边,让你选若干条不缩的边,他们不相邻(因为两端的点是有一条边,所以两端的边是不能选的)

  这个模型就是数学试卷上的经典模型?f[n]=f[n-1]+f[n-2],就是枚举第n个选还是不选,就是斐波那契数列。

  第三步的判断就搞定了。

  对于第一第二步,就是把这个大数字分成很多部分,套上第三步的方案加入答案中,这个可以用DP处理。

  f[x][0]表示x前面那条边没有选,

  f[x][1]表示x前面那条边选了。

  f[i][0]=f[i][0]+f[j][0]*F[nw-3]+f[j][1]*F[nw-2]

  f[i][1]=f[i][1]+f[j][0]*F[nw-2]+f[j][1]*F[nw-1]

  //F表示斐波那契数列, nw表示j+1~i表示的数。

  然后问题来了,F[nw???]怎么求,nw是个即大的数额。。

  【一开始我以为矩阵幂的质数可以mod phi 还乐呵呵地打了。。。

  其实可以预处理,但是传统的快速幂的话,貌似,挺慢??而且你要高精??

  传统的快速幂是以2为底的快速幂,他给你的高精数是10进制下的,不如把快速幂改成10为底的,那么你可以发现,有些过程可以合并,比如你算[l,r]的f,中间已经算了[l,r-1]的部分。

  所以预处理就是O(n^2*一个不知道多大的常数)

  好像别人打的都挺快的。。。可以看看其他解法:http://www.cnblogs.com/cghAndy/p/6594538.html

  总时间复杂度:O(n^2*一个不知道多大的常数)

  1 #include<cstdio>
  2 #include<cstdlib>
  3 #include<cstring>
  4 #include<iostream>
  5 #include<algorithm>
  6 using namespace std;
  7 #define LL long long
  8 #define Mod 1000000007
  9 #define Maxn 1010
 10
 11 int n;
 12
 13 struct node
 14 {
 15     LL w[3][3];
 16 }t[17],tt[3];
 17
 18 LL a[Maxn],f[Maxn][2];
 19 char s[Maxn];
 20
 21 void mul(int z,int x,int y)
 22 {
 23     t[2]=tt[0];
 24     for(int k=1;k<=2;k++)
 25      for(int i=1;i<=2;i++)
 26       for(int j=1;j<=2;j++)
 27         t[2].w[i][j]=(t[2].w[i][j]+t[x].w[i][k]*t[y].w[k][j])%Mod;
 28     t[z]=t[2];
 29 }
 30
 31 void qpow(int x,int b)
 32 {
 33     t[1]=tt[1];
 34     while(b)
 35     {
 36         if(b&1) mul(1,x,1);
 37         mul(x,x,x);
 38         b>>=1;
 39     }
 40     t[x]=t[1];
 41 }
 42
 43 LL B[Maxn][Maxn][3];
 44
 45 LL get_f(int x)
 46 {
 47     t[3]=tt[2];
 48     t[4]=tt[1];
 49     for(int i=x;i<=n;i++)
 50     {
 51         qpow(4,10);
 52         t[0]=t[3];
 53         qpow(0,a[i]);
 54         // mul(4,a[i]-10,4);
 55
 56         t[5]=t[4];
 57         B[x][i][0]=t[5].w[1][1];
 58         mul(5,5,3);
 59         B[x][i][1]=t[5].w[1][1];
 60         mul(5,5,3);
 61         B[x][i][2]=t[5].w[1][1];
 62     }
 63 }
 64
 65 void init()
 66 {
 67     tt[0].w[1][1]=0;tt[0].w[1][2]=0;
 68     tt[0].w[2][1]=0;tt[0].w[2][2]=0;
 69
 70     tt[1].w[1][1]=1;tt[1].w[1][2]=0;
 71     tt[1].w[2][1]=0;tt[1].w[2][2]=1;
 72
 73     tt[2].w[1][1]=0;tt[2].w[1][2]=1;
 74     tt[2].w[2][1]=1;tt[2].w[2][2]=1;
 75
 76     t[11]=tt[1];t[5]=tt[1];
 77     for(int i=2;i<=9;i++)
 78     {
 79         mul(i,i-1,5);
 80     }
 81 }
 82
 83 int main()
 84 {
 85     init();
 86     scanf("%d",&n);
 87     scanf("%s",s+1);
 88     for(int i=1;i<=n;i++) a[i]=s[i]-‘0‘;
 89
 90     memset(f,0,sizeof(f));
 91     f[0][0]=1;
 92
 93     int i,j,k;
 94     for(i=1;i<=n;i++) get_f(i);
 95
 96     for(i=1;i<=n;i++)
 97      for(j=0;j<i;j++)
 98       {
 99           f[i][0]=f[i][0]+f[j][0]*B[j+1][i][0];f[i][0]%=Mod;
100           f[i][0]=f[i][0]+f[j][1]*B[j+1][i][1];f[i][0]%=Mod;
101           f[i][1]=f[i][1]+f[j][0]*B[j+1][i][1];f[i][1]%=Mod;
102           f[i][1]=f[i][1]+f[j][1]*B[j+1][i][2];f[i][1]%=Mod;
103
104       }
105     printf("%d\n",f[n][0]);
106     return 0;
107 }

2017-03-21 21:16:25

时间: 2024-12-21 08:27:38

【BZOJ 2323】 2323: [ZJOI2011]细胞 (DP+矩阵乘法+快速幂*)的相关文章

ZOJ - 3216:Compositions (DP&amp;矩阵乘法&amp;快速幂)

We consider problems concerning the number of ways in which a number can be written as a sum. If the order of the terms in the sum is taken into account the sum is called a composition and the number of compositions of n is denoted by c(n). Thus, the

【BZOJ-1009】GT考试 KMP+DP+矩阵乘法+快速幂

1009: [HNOI2008]GT考试 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 2745  Solved: 1694[Submit][Status][Discuss] Description 阿申准备报名参加GT考试,准考证号为N位数X1X2....Xn(0<=Xi<=9),他不希望准考证号上出现不吉利的数字.他的不吉利数学A1A2...Am(0<=Ai<=9)有M位,不出现是指X1X2...Xn中没有恰好一段等于A1A2..

快速求斐波那契数列(矩阵乘法+快速幂)

斐波那契数列 给你一个n:f(n)=f(n-1)+f(n-2) 请求出 f(f(n)),由于结果很大请 对答案 mod 10^9+7; 1<=n<=10^100; 用矩阵乘法+快速幂求斐波那契数列是经典应用: 矩阵公式 C i j=C i k *C k j; 根据递推式 构造2*2矩阵: 原始矩阵 1 0 0 1 矩阵 2 1 1 1 0 原始矩阵与矩阵 2相乘达到转化状态效果: 对矩阵二进行快速幂 乘法:达到快速转化矩阵的效果: 即使达到快速转化状态:那么大的数据范围也很难求解: 高精?这有

codevs1281 矩阵乘法 快速幂 !!!手写乘法取模!!! 练习struct的构造函数和成员函数

对于这道题目以及我的快速幂以及我的一节半晚自习我表示无力吐槽,, 首先矩阵乘法和快速幂没必要太多说吧,,嗯没必要,,我相信没必要,,实在做不出来写两个矩阵手推一下也就能理解矩阵的顺序了,要格外注意一些细节,比如快速幂时ans矩阵的初始化方式,快速幂的次数,矩阵乘法过程中对临时矩阵的清零,最后输出结果时的初始矩阵...矩阵快速幂好理解但是细节还是有点小坑的.. 下面就是满满的槽点,,高能慎入!!! 对于这个题目要求矩阵过程中对m取模,结果对g取模,我表示难以接受,,上来没看清题直接wa19个点,另

矩阵乘法快速幂 cojs 1717. 数学序列

矩阵乘法模板: 1 #define N 801 2 #include<iostream> 3 using namespace std; 4 #include<cstdio> 5 int a[N][N],b[N][N],c[N][N]; 6 int n,m,p; 7 int read() 8 { 9 int ans=0,ff=1;char s; 10 s=getchar(); 11 while(s<'0'||s>'9') 12 { 13 if(s=='-') ff=-1;

矩阵乘法快速幂 codevs 1732 Fibonacci数列 2

1732 Fibonacci数列 2 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题解 查看运行结果 题目描述 Description 在“1250 Fibonacci数列”中,我们求出了第n个Fibonacci数列的值.但是1250中,n<=109.现在,你的任务仍然是求出第n个Fibonacci数列的值,但是注意:n为整数,且1 <= n <= 100000000000000 输入描述 Input Description 输入有多组数据,每

矩阵乘法快速幂 codevs 1250 Fibonacci数列

codevs 1250 Fibonacci数列 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题目描述 Description 定义:f0=f1=1, fn=fn-1+fn-2(n>=2).{fi}称为Fibonacci数列. 输入n,求fn mod q.其中1<=q<=30000. 输入描述 Input Description 第一行一个数T(1<=T<=10000). 以下T行,每行两个数,n,q(n<=109, 1<

【矩阵乘法快速幂】

快速入门视频: av56433157 1> p1926 斐波那契 #include<cstdio> #include<cstdlib> #include<cstring> using namespace std; long long n; const int mod=1000000007; long long nw[2][2],ans[2][2]; long long t[2][2]; void mul1() { memset(t,0,sizeof(t)); fo

【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,