首先是惯例的吐槽。SDOI题目名称是一个循环,题目内容也是一个循环,基本上过几年就把之前的题目换成另一个名字出出来,喜大普奔亦可赛艇。学长说考SDOI可以考出联赛分数,%%%。

下面放解题报告。并不喜欢打莫比鸟斯的解题报告就是因为公式编辑太鬼。

不知道多少分算法:简单模拟不解释。

正解一眼是莫比鸟斯函数,话说上次考莫比鸟斯就是去年吧,好像题目名也叫数字表格,只不过多了一个前缀"Crash的"。

慢慢推吧,这里公式编辑器好像坏了?雾,贼慢。

假设n<=m;(if(n>m)swap(n,m);)

老套路,枚举(i,j),看被算了多少次。

//好像不是严格意义上的布尔表达式?差不多就是这个意思吧。

然后提前+替换,变成了

然后上面那一堆东西就是喜闻乐见的莫比鸟斯函数优化

变成这样一个鬼样子

上面那堆就是喜闻乐见的数论分块搞搞。

然后注意到其实下面这一段也可以分块... ...

还是要解释一下:

把指数那一堆设为Get(nn,mm),可以用数论分块算出来。

然后原式就变成了

又可以喜闻乐见数论分块。

现在大概是O(n1/2(n1/2)1/2)=O(n3/4)*T;

然后直接肛正面应该只有60分。

100分的话卡卡常数,加点记忆化就过去了...就过去了...(大雾)。

然后还要预处理前缀积什么的,还要用逆元和欧拉函数降幂大法... ...

这题出的还是不错的。

然后好像逆元预处理 比爆算 要慢一点?雾。

其实可以离线后再省一点预处理的时间(丧心病狂)。

无所谓了。在B站上应该稳定40s以内吧。

把long long 改成int 是最好的常数优化。

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <algorithm>
#include <cstring>
#define LL long long
using namespace std; const int N = 1000010;
const int Mod = 1000000007;
const int Nmod = Mod-1;
int n,m,f[N],miu[N],vis[N],P[N/10],tot,Ny[N];
int map[5010][5010];
LL Ans,ans; inline int gi()
{
int x=0,res=1;char ch=getchar();
while(ch<'0' || ch>'9'){if(ch=='-')res=-res;ch=getchar();}
while(ch>='0'&&ch<='9')x=x*10+ch-48,ch=getchar();
return x*res;
} inline int QPow(int d,int z)
{
int _ans=1;
for(;z;z>>=1,d=(LL)d*d%Mod)
if(z&1)_ans=(LL)_ans*d%Mod;
return _ans;
} inline void pre()
{
f[1]=1;
for(int i=2;i<N;++i){
f[i]=f[i-1]+f[i-2];
if(f[i]>=Mod)f[i]-=Mod;
}
f[0]=1;
for(int i=1;i<N;++i)
f[i]=(LL)f[i]*f[i-1]%Mod;
for(int i=0;i<N;++i)
Ny[i]=QPow(f[i],Mod-2);
} inline void shai()
{
miu[1]=1;
for(int i=2;i<N;++i){
if(!vis[i])P[++tot]=i,miu[i]=-1;
for(int j=1;j<=tot;++j){
int Inc=i*P[j];
if(Inc>=N)break;
vis[Inc]=1;
if(i%P[j]==0)break;
miu[Inc]=-miu[i];
}
}
for(int i=1;i<=N;++i)
miu[i]=miu[i-1]+miu[i];
} inline int Get(int nn,int mm)
{
if(nn<=5000 && mm<=5000)
if(map[nn][mm])return map[nn][mm];
ans=0;
for(int l=1,r;l<=nn;l=r+1){
r=min(nn/(nn/l),mm/(mm/l));
ans+=1ll*(miu[r]-miu[l-1])*(nn/l)*(mm/l);
}
ans%=Nmod;
if(nn<=5000 && mm<=5000)map[nn][mm]=ans;
return ans;
} int main()
{
pre();shai();int T=gi();
while(T--){
Ans=1;n=gi();m=gi();if(n>m)swap(n,m);
for(int l=1,r;l<=n;l=r+1){
r=min(n/(n/l),m/(m/l));
Ans=1ll*Ans*QPow(1ll*f[r]*Ny[l-1]%Mod,Get(n/l,m/l))%Mod;
}
printf("%lld\n",Ans);
}
return 0;
}

  

最新文章

  1. VMware Workstation 10.0 正式版官方简体中文下载(附序列号)
  2. Linux 硬盘分区生效命令partprobe
  3. LayaAir学习笔记
  4. C语言程序设计第七次作业
  5. 2014-2015 ACM-ICPC, NEERC, Moscow Subregional Contest A. Advanced 2048
  6. 闭包(closure)
  7. 【Android 开源】:最火的Android开源项目 第01期
  8. Axiom3D学习日记 3.Cameras, Lights, and Shadows
  9. 笔记1 linux 多线程 互斥锁
  10. Constructing Roads(最小生成树)
  11. awk之FIELDWIDTHS字段宽度
  12. Django 个性化管理员站点
  13. C#中is运算符
  14. Timus 1132 Square Root(二次剩余 解法2)
  15. 【BJOI2019】排兵布阵 DP
  16. The key of real time embedded system
  17. linux安装activemq
  18. Confluence的简单安装以及与jira链接(Confluence不知道有没有破解)
  19. vue路由跳转报错解决
  20. ava中有三种移位运算符

热门文章

  1. Spring框架——AOP代理
  2. Java计算字符串中字母出现的次数
  3. 学习爬虫的day03 (通过代理去爬去数据)
  4. 51Nod1136--欧拉函数
  5. PHP获取路径或目录实现
  6. ExtJs 带参数的MVC
  7. lua 中pairs 和 ipairs差别
  8. HTML学习笔记之三(localstorage的使用)
  9. 关于React组件之间如何优雅地传值的探讨
  10. Hibernate学习(二补充)关系映射----基于外键的双向一对一