【UOJ#62】怎样跑得更快

题面

这个题让人有高斯消元的冲动,但肯定是不行的。

这个题算是莫比乌斯反演的一个非常巧妙的应用(不看题解不会做)。

套路1:

因为\(b(i)\)能表达成一系列\(x(i)\)的和,所以我们尝试通过反演将\(x(i)\)表达成一系列\(b(i)\)的和的形式,那么就可以解出来了。

然后一个简单的化简:\(gcd(i,j)^c\cdot lcm(i,j)^d=i^d\cdot j^d\cdot gcd(i,j)c-d\)。

\[\displaystyle b_i=\sum_{j=1}^ni^dj^dgcd(i,j)^{c-d}x_j\\
\frac{b_i}{i^d}=\sum_{j=1}^ngcd(i,j)^{c-d}j^dx_j
\]

套路2:

看到gcd非常不爽,考虑干掉它。

我们设\(f(i)=i^{c-d}\),再构造函数\(fr(i)\),使得\(f(n)=\sum_{d|n}fr(d)\)。\(d|gcd(i,j)\Rightarrow d|i,d|j\)。

由莫比乌斯反演可以得到\(fr(n)=\sum_{d|n}\mu(d)f(\frac{n}{d})\)

所以

\[\displaystyle\frac{b_i}{i^d}=\sum_{j=1}^n\sum_{d|i,d|j}fr(d) j^dx_j\\
\]

然后我们要将右边的式子交换求和符号(因为我们要得到套路1的形式)。

\[\displaystyle\frac{b_i}{i^d}=\sum_{j=1}^n\sum_{d|i,d|j}fr(d) j^dx_j\\
=\sum_{d|i}fr(d)\sum_{d|j}j^dx_j\\
\]

套路3:

我们设\(Z_d=\sum_{d|j}j^dx_j\)。

于是\(\displaystyle\frac{b_i}{i^d}=\sum_{d|i}fr(d)Z_d\),然后我们呢将\(fr(d)\)解出来了过后就可以将\(Z_d\)解出来。

又是莫比乌斯反演

\[\displaystyle Z_d=\sum_{d|j}j^dx_j\\
j^dx_j=\sum_{j|d}\mu(\frac{d}{j})Z_j
\]

三个反演,就将这道好题(毒瘤题)解决了。

总结:

反演在两个函数直接构建了转换的桥梁,遇到难以求解的函数时,通常将其反演成一个好求的函数再反演回来。

代码:

#include<bits/stdc++.h>
#define ll long long
#define N 100005 using namespace std;
inline int Get() {int x=0,f=1;char ch=getchar();while(ch<'0'||ch>'9') {if(ch=='-') f=-1;ch=getchar();}while('0'<=ch&&ch<='9') {x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}return x*f;} int n,c,d,q;
ll f[N],b[N],g[N],h[N];
ll fr[N],z[N];
ll s[N],ans[N];
const ll mod=998244353;
ll ksm(ll t,ll x) {
ll ans=1;
for(;x;x>>=1,t=t*t%mod)
if(x&1) ans=ans*t%mod;
return ans;
}
bool vis[N];
int pri[N],mu[N]; void pre(int n) {
mu[1]=1;
for(int i=2;i<=n;i++) {
if(!vis[i]) {
pri[++pri[0]]=i;
mu[i]=-1;
}
for(int j=1;j<=pri[0]&&1ll*i*pri[j]<=n;j++) {
vis[i*pri[j]]=1;
if(i%pri[j]==0) {
break;
}
mu[i*pri[j]]=-mu[i];
}
}
} ll tem[N];
void solve() {
for(int i=1;i<=n;i++) {
if(fr[i]==0&&s[i]) {cout<<-1<<"\n";return ;}
z[i]=s[i]*ksm(fr[i],mod-2)%mod;
}
for(int i=1;i<=n;i++) tem[i]=0;
for(int i=1;i<=n;i++) {
for(int j=i;j<=n;j+=i) {
(tem[i]+=mu[j/i]*z[j]+mod)%=mod;
}
}
for(int i=1;i<=n;i++) {
if(!h[i]&&tem[i]) {cout<<-1<<"\n";return ;}
ans[i]=tem[i]*ksm(h[i],mod-2)%mod;
}
for(int i=1;i<=n;i++) cout<<ans[i]<<" ";cout<<"\n";
} int main() {
n=Get(),c=Get(),d=Get(),q=Get();
pre(n);
int t=c-d;
t=(t%(mod-1)+mod-1)%(mod-1);
for(int i=0;i<=n;i++) f[i]=ksm(i,t);
for(int i=1;i<=n;i++) {
h[i]=g[i]=ksm(i,d);
}
for(int i=1;i<=n;i++)
for(int j=i;j<=n;j+=i)
(fr[j]+=mu[j/i]*f[i]+mod)%=mod;
while(q--) {
for(int i=1;i<=n;i++) {
s[i]=0;
b[i]=Get();
b[i]=b[i]*ksm(g[i],mod-2)%mod;
}
for(int i=1;i<=n;i++)
for(int j=i;j<=n;j+=i)
(s[j]+=mu[j/i]*b[i]+mod)%=mod;
solve();
}
return 0;
}

UOJ 【UR #5】怎样跑得更快的更多相关文章

  1. 【UOJ#62】【UR #5】怎样跑得更快(莫比乌斯反演)

    [UOJ#62][UR #5]怎样跑得更快(莫比乌斯反演) 题面 UOJ 题解 众所周知,\(lcm(i,j)=\frac{ij}{gcd(i,j)}\),于是原式就变成了: \[\sum_{j=1} ...

  2. 「UR#5」怎样跑得更快

    「UR#5」怎样跑得更快 膜这个您就会了 下面是复读机mangoyang 我们要求 \[ \sum_{j=1}^n \gcd(i,j)^{c-d} j^d x_j=\frac{b_i}{i^d} \] ...

  3. 让DB2跑得更快——DB2内部解析与性能优化

    让DB2跑得更快——DB2内部解析与性能优化 (DB2数据库领域的精彩强音,DB2技巧精髓的热心分享,资深数据库专家牛新庄.干毅民.成孜论.唐志刚联袂推荐!)  洪烨著 2013年10月出版 定价:7 ...

  4. 面试官:如何写出让 CPU 跑得更快的代码?

    前言 代码都是由 CPU 跑起来的,我们代码写的好与坏就决定了 CPU 的执行效率,特别是在编写计算密集型的程序,更要注重 CPU 的执行效率,否则将会大大影响系统性能. CPU 内部嵌入了 CPU ...

  5. [翻译] 5点建议,让iOS程序跑得更快

      [文章原地址]http://mobile.tutsplus.com/tutorials/iphone/ios-quick-tip-5-tips-to-increase-app-performanc ...

  6. 让你的 Node.js 应用跑得更快的 10 个技巧(转)

    Node.js 受益于它的事件驱动和异步的特征,已经很快了.但是,在现代网络中只是快是不行的.如果你打算用 Node.js 开发你的下一个Web 应用的话,那么你就应该无所不用其极,让你的应用更快,异 ...

  7. 安装好Windows 8后必做的几件事情,让你的Win8跑的更快更流畅。

    1.关闭家庭组,因为这功能会导致硬盘和CPU处于高负荷状态. 关闭方法:Win+C-设置-更改电脑设置-家庭组-离开 如果用不到家庭组可以直接把家庭组服务也给关闭了:控制面板-管理工具-服务-Home ...

  8. 让你的 Node.js 应用跑得更快的 10 个技巧

    Node.js 受益于它的事件驱动和异步的特征,已经很快了.但是,在现代网络中只是快是不行的.如果你打算用 Node.js 开发你的下一个Web 应用的话,那么你就应该无所不用其极,让你的应用更快,异 ...

  9. UOJ#62. 【UR #5】怎样跑得更快 数论 莫比乌斯反演

    原文链接https://www.cnblogs.com/zhouzhendong/p/UOJ62.html 题解 太久没更博客了,该拯救我的博客了. $$\sum_{1\leq j \leq n} \ ...

随机推荐

  1. JavaWeb学习 (二十六)————监听器(Listener)学习(二)

    一.监听域对象中属性的变更的监听器 域对象中属性的变更的事件监听器就是用来监听 ServletContext, HttpSession, HttpServletRequest 这三个对象中的属性变更信 ...

  2. SpringBoot JPA + H2增删改查示例

    下面的例子是基于SpringBoot JPA以及H2数据库来实现的,下面就开始搭建项目吧. 首先看下项目的整体结构: 具体操作步骤: 打开IDEA,创建一个新的Spring Initializr项目, ...

  3. MySQL常用的备份方式与备份工具简介

    一.MySQL备份方式与备份类型 1.备份的必要性 再生产环境中,为了防止硬件故障.软件故障.自然灾害.误操作等各种原因导致的数据库数据丢失后能恢复到事故之前的状态,我们需要对数据库进行备份和恢复操作 ...

  4. yarn 工作原理(2)

    1.client向yarn提交job,首先找ResourceManager分配资源, 2.ResourceManager开启一个Container,在Container中运行一个Application ...

  5. JPA与EJB3的关系

    JPA是基于Java持久化的解决方案,主要是为了解决ORM框架的差异,它的出现在某种程度上能够解决目前ORM框架之间不能够兼容的问题,对开发人员来说,能够更好的在JPA规范下进行系统开发. JPA全称 ...

  6. API网关【gateway 】- 3

    最近在公司进行API网关重写,公司内采用serverMesh进行服务注册,调用,这里结合之前学习对API网关服务进行简单的总结与分析. 由于采用了大量的nginx相关的东西,所以在此记录一下: 在ng ...

  7. linux下vscode的c++工程配置

    准备 安装vscode,可直接下载deb包进行安装,完成后安装C/C++ for Visual Studio Code插件,安装后重启(最新1.3版本以后不需要重启). 生成目录和文件 新建文件夹[t ...

  8. 【pygame游戏编程】第四篇-----打字测速游戏

    下面我们一起用pygame编写一个打字测速游戏 这是一个很实用的有趣的小游戏: 开始之前先来学习几个小函数: 1. ord(ch) python内置函数,传入一个字符,返回字符的ascii码 2.ch ...

  9. a标签禁止跳转或者不跳转的几种实现方式

    1.onclick事件中返回false <a href="http://www.baidu.com"  onclick="return false" &g ...

  10. [转]Serif和Sans-serif字体的区别

    在西方国家罗马字母阵营中,字体分为两大种类:Sans Serif和Serif,打字机体虽然也属于Sans Serif,但由于是等宽字体,所以另外独立出Monospace这一种类,例如在Web中,表示代 ...