题目描述

给出$n$个正整数$a_1,a_2...a_n$和一个质数mod。一个变量$x$初始为$1$。进行$m$次操作.每次在$n$个数中随机选一个$a_i$,然后$x=x\times a_i$.问$m$次操作之后$x$的取值的期望。答案一定可以表示成$\frac{a}{b}$的精确分数形式。$a$和$b$可能很大,所以只需要输出$a\times b^{{10}^9+5}$模${10}^9+7$的结果。


输入格式

第一行三个整数$n,m,mod$。
接下来一行$n$个空格隔开的正整数$a_1,a_2...a_n$。


输出格式

一行一个整数表示答案。


样例

样例输入

2 1 3
1 2

样例输出

500000005


数据范围与提示

第$1$个测试点:$mod=2$。
第$2$个测试点:$n=1$。
第$3,4,5$个测试点:$m\leqslant 1,000,1\leqslant mod \leqslant 300$。
第$6,7,8$个测试点:$1\leqslant mod \leqslant 300$。
对于全部测试点:$1\leqslant a_i <mod,mod$为质数$1\leqslant mod \leqslant 1,000,1\leqslant n\leqslant {10}^5,1\leqslant m\leqslant {10}^9$。


吕老师教你学数学
质数$P$的原根$g$满足$1\leqslant rt<P$,且$rt$的$1$次方,$2$次方$...(P-1)$次方在模$P$意义下可以取遍$1$到$(P-1)$的所有整数。
欧拉定理:对于质数$P,1\leqslant x<P$的任意$x$的$(P-1)$次方在模$P$意义下都为$1$。
显然,原根的$1$次方,$2$次方$...(P-2)$次方在模$P$意义下都不为$1$,只有$(P-1)$次方在模$P$意义下为$1$。
这也是一个数成为原根的充分必要条件。


题解

第$1$个测试点:

$mod=2$,不要慌,不用考虑$mod$完会有$0$的情况,因为$1\leqslant a_i <mod$,于是$a_i$一定是$1$,所以$x$就是$1$,直接$puts("1");return 0;$就好了。

期望得分:$10$分,共$10$分。

第$2$个测试点:

$n=1$,因为每次只能选这一个数,所以快速幂直接搞就好了。

期望得分:$10$分,共$20$分。

第$3,4,5$个测试点:

考虑$DP$,定义$dp[i][j]$表示第$i$次操作后$x$为$j$的概率。

那么我们能写出$DP$式子:$dp[i][j\times a[k]\% mod]=\sum \limits_{k=1}^ndp[i-1][[j]$。

时间复杂度:$\Theta (n\times m\times mod)$。

但是注意到$3,4,5$这三个测试点并没有对$n$的范围加以约束,所以显然会$T$飞起,那么我们考虑优化这个$DP$。

接着注意$1\leqslant a_i <mod$,所以一共最多会有$mod-1$种不同的$a_i$,试图从这里入手。

开一个数组$sum[i]$记录$i$这个数出现的次数。

那么$DP$式子即可转化为:$dp[i][j\times k\%mod]=\sum \limits_{k=0}^{mod-1}dp[i-1][j]\times sum[k]$。

时间复杂度:$\Theta (m\times {mod}^2)$。

期望得分:$30$分$(RE)$,共$50$分。

如果你说你不希望$RE$,那么你可以把上面$dp$的第一维滚起来。

期望得分:$30$分$(TLE)$,共$50$分。

第$6,7,8$个测试点:

考虑对上面的算法进行优化,$m$很大,考虑矩阵快速幂。

时间复杂度:$\Theta ({mod}^3\times \log m)$。

期望得分:$60$分,共$80$分。

第$9,10$个测试点:

考虑对第$3,4,5$个测试点的$DP$进行优化,还是从$m$下手。

根据上面的$DP$式子可得:$dp[2\times i][j\times k\% mod]=\sum \limits_{j=0}^{mod-1}\sum \limits_{k=0}^{mod-1}dp[i][j]\times dp[i][k]$。

接着使用滚动数组,快速幂求解即可。


代码时刻

$30$分解法(非滚动):

#include<bits/stdc++.h>
using namespace std;
int n,m,mod;
long long sum[1001];
long long fz,fm,ans;
long long dp[1001][1001];
long long qpow(long long x,long long y)
{
long long res=1;
while(y)
{
if(y&1)res=(res*x)%1000000007;
x=(x*x)%1000000007;
y>>=1;
}
return res;
}
int main()
{
scanf("%d%d%d",&n,&m,&mod);
for(int i=1;i<=n;i++)
{
int x;
scanf("%d",&x);
sum[x]++;
}
dp[0][1]=1;
for(int i=1;i<=m;i++)
for(int j=0;j<mod;j++)
for(int k=0;k<mod;k++)
dp[i][j*k%mod]=(dp[i][j*k%mod]+dp[i-1][j]*sum[k]%1000000007)%1000000007;
for(int i=0;i<mod;i++)
fz=(fz+dp[m][i]*i%1000000007)%1000000007;
fm=qpow(n,m);
fm=qpow(fm,1000000005);
ans=fz*fm%1000000007;
printf("%lld",ans);
return 0;
}

$30$分解法(滚动):

#include<bits/stdc++.h>
using namespace std;
int n,m,mod;
long long sum[1001];
long long ans;
long long dp[2][1001];
long long qpow(long long x,long long y)
{
long long res=1;
while(y)
{
if(y&1)res=(res*x)%1000000007;
x=(x*x)%1000000007;
y>>=1;
}
return res;
}
int main()
{
scanf("%d%d%d",&n,&m,&mod);
for(int i=1;i<=n;i++)
{
int x;
scanf("%d",&x);
sum[x]++;
}
dp[0][1]=1;
for(int i=1;i<=m;i++)
{
memset(dp[i&1],0,sizeof(dp[i&1]));
for(int j=0;j<mod;j++)
for(int k=0;k<mod;k++)
dp[i&1][j*k%mod]=(dp[i&1][j*k%mod]+dp[i&1^1][j]*sum[k]%1000000007)%1000000007;
}
for(int i=0;i<mod;i++)
ans=(ans+dp[m&1][i]*i%1000000007)%1000000007;
printf("%lld",ans*qpow(qpow(n,m),1000000005)%1000000007);
return 0;
}

$100$分解法:

#include<bits/stdc++.h>
using namespace std;
int n,m,mod;
long long dp[2][100001],dp03[2][100001];
bool p,q;
long long ans;
long long qpow(long long x,long long y)
{
long long res=1;
while(y)
{
if(y&1)res=(res*x)%1000000007;
x=(x*x)%1000000007;
y>>=1;
}
return res;
}
void judge()
{
q^=1;
memset(dp03[q],0,sizeof(dp03[q]));
for(int i=0;i<mod;i++)
for(int j=0;j<mod;j++)
dp03[q][1LL*i*j%mod]=(dp03[q][1LL*i*j%mod]+dp03[q^1][i]*dp03[q^1][j]%1000000007)%1000000007;
}
int main()
{
scanf("%d%d%d",&n,&m,&mod);
for(int i=1;i<=n;i++)
{
int x;
scanf("%d",&x);
dp03[0][x]++;
}
dp[0][1]=1;
long long inv=qpow(qpow(n,m),1000000005)%1000000007;
while(m)
{
if(m&1)
{
p^=1;
memset(dp[p],0,sizeof(dp[p]));
for(int i=0;i<mod;i++)
for(int j=0;j<mod;j++)
dp[p][1LL*i*j%mod]=(dp[p][1LL*i*j%mod]+dp[p^1][i]*dp03[q][j]%1000000007)%1000000007;
}
m>>=1;
judge();
}
for(int i=0;i<mod;i++)
ans=(ans+i*dp[p][i])%1000000007;
ans=(ans*inv)%1000000007;
printf("%lld",ans);
return 0;
}

rp++

[CSP-S模拟测试]:随(快速幂+数学)的更多相关文章

  1. Rightmost Digit(快速幂+数学知识OR位运算) 分类: 数学 2015-07-03 14:56 4人阅读 评论(0) 收藏

    C - Rightmost Digit Time Limit:1000MS Memory Limit:32768KB 64bit IO Format:%I64d & %I64u Submit ...

  2. 1282 - Leading and Trailing ---LightOj1282(快速幂 + 数学)

    http://lightoj.com/volume_showproblem.php?problem=1282 题目大意: 求n的k次方的前三位和后三位数然后输出 后三位是用快速幂做的,我刚开始还是不会 ...

  3. zhx's contest (矩阵快速幂 + 数学推论)

    zhx's contest Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) To ...

  4. lightoj1281快速幂+数学知识

    https://vjudge.net/contest/70017#problem/E 后半部分快速幂就能求出来,要注意03lld不然001是输出错误为1.前半部分用log10() 对于给定的一个数n, ...

  5. noip-2006普及组-数列- 【模拟-找规律-快速幂】

    链接:https://ac.nowcoder.com/acm/contest/153/1047 来源:牛客网 题目描述 给定一个正整数k( ≤ k ≤ ),把所有k的方幂及所有有限个互不相等的k的方幂 ...

  6. LightOJ 1070 Algebraic Problem:矩阵快速幂 + 数学推导

    题目链接:http://lightoj.com/volume_showproblem.php?problem=1070 题意: 给你a+b和ab的值,给定一个n,让你求a^n + b^n的值(MOD ...

  7. HDU 3117 Fibonacci Numbers( 矩阵快速幂 + 数学推导 )

    链接:传送门 题意:给一个 n ,输出 Fibonacci 数列第 n 项,如果第 n 项的位数 >= 8 位则按照 前4位 + ... + 后4位的格式输出 思路: n < 40时位数不 ...

  8. [CSP-S模拟测试]:次芝麻(数学)

    题目描述 小$K$和小$X$都是小次货.身为小次货,最重要的事情就是次啦!所以他们正在纠结如何分芝麻次.一开始,小$K$有$n$个芝麻,小$X$有$m$个芝麻.因为他们都想次更多芝麻,所以每次手中芝麻 ...

  9. 【洛谷P1962 斐波那契数列】矩阵快速幂+数学推导

    来提供两个正确的做法: 斐波那契数列双倍项的做法(附加证明) 矩阵快速幂 一.双倍项做法 在偶然之中,在百度中翻到了有关于斐波那契数列的词条(传送门),那么我们可以发现一个这个规律$ \frac{F_ ...

随机推荐

  1. 递归法求组合数C(m,n)

    假设这样一个数组: 1 2 3 4 5 n=5 若 m=3 也就是要求C(3,5) 首先先选第一个数 1 那么剩下的工作就是在2-5之间选择2个数 如果我们没有选择第一个数 选第二个数2 那么剩下的工 ...

  2. 蚁群算法解决TSP问题

    代码实现 运行结果及参数展示 alpha=1beta=5 rho=0.1  alpha=1beta=1rho=0.1 alpha=0.5beta=1rho=0.1 概念蚁群算法(AG)是一种模拟蚂蚁觅 ...

  3. 一分钟安装mysql

    学数据库的人都知道,MySQL数据库是比较基本的掌握要求,不仅开源而且社区版本是免费使用的.由于工作上或者经常更换系统的原因,有时候会需要安装MySQL数据库.为了不至于每次安装都要查阅资料,现把安装 ...

  4. Centos7 搭建pptp服务器

    1.检查是否支持pptp 返回ok即表示支持 modprobe ppp-compress-18 && echo ok 2.安装ppp yum install -y ppp 3.安装pp ...

  5. [BZOJ 4025]二分图(线段树分治+带边权并查集)

    [BZOJ 4025]二分图(线段树分治+带边权并查集) 题面 给出一个n个点m条边的图,每条边会在时间s到t出现,问每个时间的图是否为一个二分图 \(n,m,\max(t_i) \leq 10^5\ ...

  6. [LeetCode] 109. 有序链表转换二叉搜索树

    题目链接 : https://leetcode-cn.com/problems/convert-sorted-list-to-binary-search-tree/ 题目描述: 给定一个单链表,其中的 ...

  7. 服务性能指标:PV、UV、TPS、QPS

    名词解释 PV Page View,网页浏览量.网页被读者调用浏览的次数.网页每次打开或刷新一次页面,记录一次.用户对同一页面的多次访问,访问量累计. UV Unique Visitor,独立访问者. ...

  8. P2586 [ZJOI2008]杀蚂蚁

    传送门 快乐模拟,修身养性 代码长度其实还好,主要是细节多 只要知道一些计算几何基础知识即可快乐模拟,按着题目要求一步步实现就行啦 注意仔细读题,蚂蚁每 $5$ 秒乱走一次的时候是只要能走就走了,不一 ...

  9. kNN分类算法实现

    kNN算法就是计算每个点到其他所有点的距离,选出距离最小的k个点.在这k个点里,哪个类别的最多,就把待分类的点归到哪类. kNN.py: from numpy import * import oper ...

  10. python初步学习

    一.字符编码 ASCII表是是基于拉丁字母的一套电脑编码系统,主要用于显示现代英语和其他西欧语言,其最多只能用 8 位来表示(一个字节),即:2**8 = 256-1,所以,ASCII码最多只能表示 ...