Brute-force Algorithm_矩阵快速幂&&欧拉公式*****

Any fool but Brute knows that the function “funny” will be called too many times. Brute wants to investigate the number of times the function will be called, but he is too lazy to do it.
Now your task is to calculate how many times the function “funny” will be called, for the given a, b and n. Because the answer may be too large, you should output the answer module by P.
For each test cases, there are four integers a, b, P and n in a single line.
You can assume that 1≤n≤1000000000, 1≤P≤1000000, 0≤a, b<1000000.
3 4 10 3
4 5 13 5
3 2 19 100
Case #2: 11
Case #3: 12
【参考】http://www.cnblogs.com/183zyz/archive/2012/05/11/2495401.html
【题意、思路】f[n]=f[n-1]*f[n-2];
f的前面几项可以罗列出来:
a^1*b^0,a^0*b^1,a^1*b^1,a^1*b^2,a^2*b^3....
可以发现a的指数和b的指数均类似于斐波那契数列。
用矩阵的快速幂可以很快的求出第n项a和b的指数分别是多少。
但是这个指数会非常大,存不下来,需要对一个数去模。
这里需要用到一个公式:
A^B%C=A^( B%Phi[C] + Phi[C] )%C (B>=Phi[C])
Phi[C]表示不大于C的数中与C互质的数的个数,可以用欧拉函数来求:
找到C的所有素因子。
Phi[C]=C*(1-1/q1)*(1-1/q2)*(1-1/q3)*....*(1-1-qk);
q1,q2,q3...qk表示C的素因子。
到这里基本上就能解决了。
#include<iostream>
#include<stdio.h>
#include<string.h>
using namespace std;
const int N=;
long long int a,b,p,n;
int vis[N],prime[N],K;
long long int phi;
struct mat
{
long long int a[][];
};
void init()
{
memset(vis,,sizeof(vis));
for(int i=;i<=;i++)
{
if(vis[i]==)
{
for(int j=i+i;j<=;j+=i)
{
vis[j]=;
}
}
}
K=;
for(int j=;j<=;j++)
if(vis[j]==) prime[++K]=j;
}
mat power(mat a1,mat b1)//矩阵乘法
{
mat c;
for(int i=;i<=;i++)
{
for(int j=;j<=;j++)
{
c.a[i][j]=;
for(int k=;k<=;k++)
{
c.a[i][j]+=a1.a[i][k]*b1.a[k][j];
if(c.a[i][j]>phi)
c.a[i][j]=c.a[i][j]%phi+phi;
}
}
}
return c;
}
mat mod(mat x,long long int k)
{
mat ans;
ans.a[][]=;
ans.a[][]=;
ans.a[][]=;
ans.a[][]=;
while(k)
{
if(k%) ans=power(ans,x);
x=power(x,x);
k/=;
}
return ans;
}
long long int mod1(long long int a,long long int k)
{
long long int ans;
ans=;
while(k)
{
if(k%)
{
ans=ans*a;
ans%=p;
}
a=a*a;
a%=p;
k/=;
}
return ans%p;
}
int main()
{
int t,cas=;
long long int aa,bb,num,num2,num1;
mat now,ans;
init();
scanf("%d",&t);
while(t--)
{
scanf("%lld%lld%lld%lld",&a,&b,&p,&n);
printf("Case #%d: ",cas++); if(n==) {printf("%lld\n",a%p);continue;}
else if(n==) {printf("%lld\n",b%p);continue;}
else if(n==) {printf("%lld\n",a*b%p);continue;}
if(p==) {printf("0\n");continue;}
else
{
now.a[][]=;
now.a[][]=;
now.a[][]=;
now.a[][]=;
// A^B % C = A ^ ( B % phi[C] + phi[C] ) %C ( B >= phi[C] ) ,phi[C]表示与C互质的数的个数
phi=;
num=p;
for(int i=; i<=K; i++)
{
if(prime[i]>p) break;
if(p%prime[i]==)
{
phi*=(prime[i]-);
num/=prime[i];
} }
//phi[C]=C*(1-1/p1)*(1-1/p2)*...*(1-1/pt);p1,p2,...pt表示C的素因子
phi*=num;//phi表示phi[c]
ans=mod(now,n-);//求指数 num1=ans.a[][];//a的指数
num2=ans.a[][]+ans.a[][];//b的指数
if(num2>phi) num2=num2%phi+phi; aa=mod1(a,num1);//a^num1%p
bb=mod1(b,num2);//b^num2%p
printf("%lld\n",aa*bb%p);
} }
return ;
}
Brute-force Algorithm_矩阵快速幂&&欧拉公式*****的更多相关文章
- BNU29139——PvZ once again——————【矩阵快速幂】
PvZ once again Time Limit: 2000ms Memory Limit: 65536KB 64-bit integer IO format: %lld Java cla ...
- 矩阵快速幂 HDU 4565 So Easy!(简单?才怪!)
题目链接 题意: 思路: 直接拿别人的图,自己写太麻烦了~ 然后就可以用矩阵快速幂套模板求递推式啦~ 另外: 这题想不到或者不会矩阵快速幂,根本没法做,还是2013年长沙邀请赛水题,也是2008年Go ...
- 51nod 算法马拉松18 B 非010串 矩阵快速幂
非010串 基准时间限制:1 秒 空间限制:131072 KB 分值: 80 如果一个01字符串满足不存在010这样的子串,那么称它为非010串. 求长度为n的非010串的个数.(对1e9+7取模) ...
- 51nod 1113 矩阵快速幂
题目链接:51nod 1113 矩阵快速幂 模板题,学习下. #include<cstdio> #include<cmath> #include<cstring> ...
- 【66测试20161115】【树】【DP_LIS】【SPFA】【同余最短路】【递推】【矩阵快速幂】
还有3天,今天考试又崩了.状态还没有调整过来... 第一题:小L的二叉树 勤奋又善于思考的小L接触了信息学竞赛,开始的学习十分顺利.但是,小L对数据结构的掌握实在十分渣渣.所以,小L当时卡在了二叉树. ...
- HDU5950(矩阵快速幂)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5950 题意:f(n) = f(n-1) + 2*f(n-2) + n^4,f(1) = a , f(2 ...
- 51nod 1126 矩阵快速幂 水
有一个序列是这样定义的:f(1) = 1, f(2) = 1, f(n) = (A * f(n - 1) + B * f(n - 2)) mod 7. 给出A,B和N,求f(n)的值. Input 输 ...
- hdu2604(递推,矩阵快速幂)
题目链接:hdu2604 这题重要的递推公式,找到公式就很easy了(这道题和hdu1757(题解)类似,只是这道题需要自己推公式) 可以直接找规律,推出递推公式,也有另一种找递推公式的方法:(PS: ...
- 矩阵乘法&矩阵快速幂&矩阵快速幂解决线性递推式
矩阵乘法,顾名思义矩阵与矩阵相乘, 两矩阵可相乘的前提:第一个矩阵的行与第二个矩阵的列相等 相乘原则: a b * A B = a*A+b*C a*c+b*D c d ...
随机推荐
- 2016年31款轻量高效的开源JavaScript插件和库
目前有很多网站设计师和开发者喜欢使用由JavaScript开发的插件和库,但同时面临一个苦恼的问题:它们中的大多数实在是太累赘而且常常降低网站的性能.其实,其中也有不少轻量级的插件和库,它们不仅轻巧有 ...
- lucene底层数据结构——底层filter bitset原理,时间序列数据压缩将同一时间数据压缩为一行
如何联合索引查询? 所以给定查询过滤条件 age=18 的过程就是先从term index找到18在term dictionary的大概位置,然后再从term dictionary里精确地找到18这个 ...
- PDF 补丁丁 0.4.1.820 测试版发布:统一PDF的页面尺寸
之前删掉了统一页面尺寸功能,在这个版本给搬回来了. 调整页面留白的功能还没实现,请无视相关选项.
- UVALive 6948 Jokewithpermutation dfs
题目链接:UVALive 6948 Jokewithpermutation 题意:给一串数字序列,没有空格,拆成从1到N的连续数列. dfs. 可以计算出N的值,也可以直接检验当前数组是否合法. # ...
- php安装出现的部分错误
在CentOS编译PHP5的时候有时会遇到以下的一些错误信息,基本上都可以通过yum安装相应的库来解决.以下是具体的一些解决办法: checking for BZip2 support… yes ch ...
- C#学习7.31判断体重是否超标
using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.T ...
- 各种常用函数 (SQL)
数学函数 1.绝对值 S:select abs(-1) value O:select abs(-1) value from dual 2.取整(大) S:select ceiling(-1.001 ...
- 二模 (13)day1
第一题: 题目大意: N个发射站排成一排,求每个发射站左右第一个比它高的发射站. N<=1000000 解题过程: 1.前几天做poj的时候刚好在discuss里看到有一个神奇的东东叫单调栈,正 ...
- Maven 玩 github上的项目
第一步,使用maven创建了一个项目"helloworld",cmd命令如下: @echo offecho [INFO] Generating project in ./gener ...
- 在shell脚本中进行条件控制以及使用循环
转载请标明:http://www.cnblogs.com/winifred-tang94/ if条件语句语法: if [ 条件表达式 ] then 代码 else 代码 fi 注意:在上面的if条件语 ...