HDU 5628 Clarke and math Dirichlet卷积+快速幂
题意:bc round 72 中文题面
分析(官方题解):
如果学过Dirichlet卷积的话知道这玩意就是g(n)=(f*1^k)(n),
由于有结合律,所以我们快速幂一下1^k就行了。
当然,强行正面刚和式也是能搞的(反正我不会)。
一次Dirichlet卷积复杂度是O(nlogn)的,所以总时间复杂度为O(nlognlogk).
注:普及Dirichlet卷积概念,设f,g为两个数论函数,
那么规定 (f∗g)=∑d|nf(d)g(n/d)为f,g的Dirichlet卷积
Dirichlet卷积有以下性质,有交换律和结合律,对加法有分配律
f∗(g∗h)=(f∗g)∗h
f∗(g+h)=f∗g+f∗h
f∗g=g∗f
然后对于这个题解上所说的一次Dirichlet卷积是O(nlogn)这个我不是很懂
但是看网上大神写的估计也差不多,感觉和埃氏筛的复杂度差不多甚至还要小应该是在O(nlognlogn)左右
所以下面的代码的复杂度大概应该是O(nlog^2nlogk)
然后顺带提一下,对于Dirichlet卷积来说,规定有一个幺元e的话
即:f*e=f,
这个幺元函数有一个性质e[1]=1,e[k]=0,k=1,2,3.....
这个在快速幂的时候会用到
然后上代码
#include <iostream>
#include <cstdio>
using namespace std;
typedef long long LL;
const int N=1e5+;
const LL mod=1e9+;
int n,k;
LL f[N],s[N],ans[N],tmp[N];
void solve()
{
while(k)
{
if(k&)
{
for(int i=; i<=n; ++i)tmp[i]=;
for(int i=; i*i<=n; ++i)
{
tmp[i*i]=(tmp[i*i]+ans[i]*s[i])%mod;
for(int j=i+; j*i<=n; ++j)
tmp[i*j]+=(ans[i]*s[j]%mod+ans[j]*s[i]%mod)%mod,tmp[i*j]%=mod;
}
for(int i=; i<=n; ++i)ans[i]=tmp[i];
}
for(int i=; i<=n; ++i)tmp[i]=;
for(int i=; i*i<=n; ++i)
{
tmp[i*i]=(tmp[i*i]+s[i]*s[i])%mod;
for(int j=i+; j*i<=n; ++j)
tmp[i*j]+=(s[i]*s[j]%mod+s[j]*s[i]%mod)%mod,tmp[i*j]%=mod;
}
for(int i=;i<=n;++i)s[i]=tmp[i];
k>>=;
}
for(int i=; i<=n; ++i)tmp[i]=;
for(int i=; i*i<=n; ++i)
{
tmp[i*i]=(tmp[i*i]+f[i]*ans[i])%mod;
for(int j=i+; j*i<=n; ++j)
tmp[i*j]+=(f[i]*ans[j]%mod+f[j]*ans[i]%mod)%mod,tmp[i*j]%=mod;
}
for(int i=;i<=n;++i)ans[i]=tmp[i];
}
int main()
{
int T;
scanf("%d",&T);
while(T--)
{
scanf("%d%d",&n,&k);
for(int i=; i<=n; ++i)
scanf("%I64d",&f[i]),s[i] = ,ans[i] = ;
ans[] = ;
solve();
for(int i=; i<n; ++i)
printf("%I64d ",ans[i]);
printf("%I64d\n",ans[n]);
}
return ;
}
HDU 5628 Clarke and math Dirichlet卷积+快速幂的更多相关文章
- HDU.5628.Clarke and math(狄利克雷卷积 快速幂)
\(Description\) \[g(i)=\sum_{i_1|i}\sum_{i_2|i_1}\sum_{i_3|i_2}\cdots\sum_{i_k|i_{k-1}}f(i_k)\ mod\ ...
- HDU 5628 Clarke and math——卷积,dp,组合
HDU 5628 Clarke and math 本文属于一个总结了一堆做法的玩意...... 题目 简单的一个式子:给定$n,k,f(i)$,求 然后数据范围不重要,重要的是如何优化这个做法. 这个 ...
- HDU 5628 Clarke and math dp+数学
Clarke and math 题目连接: http://acm.hdu.edu.cn/showproblem.php?pid=5628 Description Clarke is a patient ...
- HDU 1757 A Simple Math Problem (矩阵快速幂)
题目 A Simple Math Problem 解析 矩阵快速幂模板题 构造矩阵 \[\begin{bmatrix}a_0&a_1&a_2&a_3&a_4&a ...
- hdu 1757 A Simple Math Problem_矩阵快速幂
题意:略 简单的矩阵快速幂就行了 #include <iostream> #include <cstdio> #include <cstring> using na ...
- hdu 5187 zhx's contest [ 找规律 + 快速幂 + 快速乘法 || Java ]
传送门 zhx's contest Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others ...
- HDU 2855 斐波那契+矩阵快速幂
http://acm.hdu.edu.cn/showproblem.php?pid=2855 化简这个公式,多写出几组就会发现规律 d[n]=F[2*n] 后面的任务就是矩阵快速幂拍一个斐波那契模板出 ...
- HDU 5950:Recursive sequence(矩阵快速幂)
http://acm.hdu.edu.cn/showproblem.php?pid=5950 题意:给出 a,b,n,递推出 f(n) = f(n-1) + f(n-2) * 2 + n ^ 4. f ...
- HDU1757 A Simple Math Problem 矩阵快速幂
A Simple Math Problem Time Limit: 3000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Ot ...
随机推荐
- HTMLParser获取属性名
HTMLParser获取属性名方式: 原始网页文本: <a title="美军被曝虐尸" href="http://www.sogou.com/web?query= ...
- linux系统日志使用
FROM:http://blog.csdn.net/zzxian/article/details/7905964 Part I: syslogd & klogd ---------/etc ...
- Visual Studio 2013 之 Productivity Power Tools
http://blogs.msdn.com/b/visualstudio_cn/archive/2014/02/20/visual-studio-2013-productivity-power-too ...
- hadoop 错误处理机制
hadoop 错误处理机制 1.硬件故障 硬件故障是指jobtracker故障或TaskTracker 故障 jobtracker是单点,若发生故障,目前hadoop 还无法处理,唯有选择最牢靠的硬件 ...
- c++ 异常处理 assert | try
#include <iostream> #include <cassert> using namespace std; int main() { ; assert(i == ) ...
- hibernate多对一单向外键
hibernate多对一单向外键: 描述:
- 【概率】poj 2096:Collecting Bugs
Description Ivan is fond of collecting. Unlike other people who collect post stamps, coins or other ...
- js和HTML结合(补充知识:如何防止文件缓存的js代码)
来自<javascript高级程序设计 第三版:作者Nicholas C. Zakas>的学习笔记(二) 使用html标签<script>可以把js嵌入到html页面中,让脚本 ...
- [Akka]发送一条消息的内部流程
本想通过了解一下Akka-actor工程中主要的类的概念,来看下Akka内部运作的机制.无奈里边的类的确太多,注释中对每个类的功能也没有足够的解释.所以还是通过debug的方式,找个入手点,看一下互相 ...
- redis 参考
http://redis.readthedocs.org/en/2.4/index.html