Codeforces 题面传送门 & 洛谷题面传送门

感谢此题教会我一个东西叫做同余最短路(大雾

首先这个不同 \(k\) 的个数 \(\le 50\) 这个条件显然是让我们对每个 \(k\) 进行一遍预处理并快速求出答案。怎么预处理呢?首先考虑一个非常 trivial 的性质,那就是所有 \(k\) 的非质数因子显然可以表示成质因子的和对吧,所以一个数能够表示成 \(k\) 的若干个质因子的和,当且仅当它能够表示成 \(k\) 的若干个质因子的和。因此考虑先对 \(k\) 进行一遍质因数分解——可以使用 PR 算法,但是我不会,每次 \(\sqrt{k}\) 地分解会超时,这里有一个简单的技巧,之前暑假某场模拟赛出现过,就是显然我们只会用到 \(\le\sqrt{k}\) 的所有质因子,因此可以考虑预处理出 \(\le \sqrt{10^{15}}\) 的所有质数,然后每次分解遍历一遍这个质数集合即可,这样分解的复杂度可以降到 \(\sqrt{10^{15}}·\dfrac{1}{\ln 10^{15}}·50\)。

考虑怎样计算答案,如果 \(k\) 本身就是质数那显然只有 \(k\) 的倍数能够表示成 \(k\) 的 \(\ne 1\) 的质因子之和。直接判断一下 \(n\bmod k=0\) 是否成立即可。如果 \(k\) 有两个质因子,那么假设 \(k\) 两个质因子分别为 \(x,y\),那么我们需检验是否 \(\exists a,b\in[0,\infty)\cap\mathbb{Z},s.t.ax+by=n\),显然必须有 \(by\equiv n\pmod{x}\),那么 \(b\) 的最小值 \(b_{\min}=n·y^{-1}\bmod x\),检验一下是否有 \(b_{\min}·y\le n\) 即可。

如果 \(k\) 质因子个数 \(\ge 3\),那么假设 \(k\) 的质因子分别为 \(p_1,p_2,\cdots,p_m\),那么我们需检验是否 \(\exists x_1,x_2,\cdots,x_m\in[0,\infty)\cap\mathbb{Z},s.t.\sum\limits_{i=1}^mp_ix_i=n\),直接检验貌似有点棘手,不过不难发现有一个性质,那就是对于任意 \(x_i\) 及 \(n\),如果 \(n\) 可行,那么 \(n+zp_i\) 也可行,也就是说对于所有 \(j\in[0,p_i)\),必然存在一个界 \(lim_j\),满足所有 \(\bmod x_i=j\) 的数当中,所有 \(\ge lim_j\) 的数都符合要求,所有 \(<lim_j\) 的数都符合要求,考虑怎样求 \(lim_j\),显然 \(lim_0=0\),而对于所有 \(j\in[0,x_i),t\in[1,m]\),显然 \(lim_{(j+p_t)\bmod p_1}\le lim_j+p_t\),这是一个最短路的形式,用 dijkstra 转移即可。并且不难发现对于 \(k\le 10^{15}\),这种情况下 \(k\) 最小的质因子必然 \(\le 10^5\),因此我们对 \(k\) 最小质因子跑同余最短路即可实现 \(10^5\log 10^5·50\) 的复杂度。

话说这场的 D、E,尤其是那个恶心无比的 D……勾起了远古回忆

const int MAXN=1e4;
const int MAXK=1e5;
const int MAXV=3.163e7;
int qpow(int x,int e,int mod){
int ret=1;
for(;e;e>>=1,x=1ll*x*x%mod) if(e&1) ret=1ll*ret*x%mod;
return ret;
}
int qu,pr[MAXV/8+5],prcnt,vis[MAXV+5];
void sieve(int n){
for(int i=2;i<=n;i++){
if(!vis[i]) pr[++prcnt]=i;
for(int j=1;j<=prcnt&&pr[j]*i<=n;j++){
vis[pr[j]*i]=1;if(i%pr[j]==0) break;
}
}
}
int res[MAXN+5];
map<ll,vector<pair<ll,int> > > mpv;
vector<ll> decomp(ll v){
vector<ll> ret;
for(int i=1;i<=prcnt;i++) if(v%pr[i]==0){
ret.pb(pr[i]);
while(v%pr[i]==0) v/=pr[i];
} if(v>1) ret.pb(v);
return ret;
}
ll dis[MAXK+5];
int main(){
sieve(MAXV);scanf("%d",&qu);
for(int i=1;i<=qu;i++){
ll n,k;scanf("%lld%lld",&n,&k);
if(k!=1) mpv[k].pb(mp(n,i));
}
for(auto it:mpv){
ll v=it.fi;vector<pair<ll,int> > qv=it.se;
vector<ll> pr=decomp(v);
if(pr.size()==1){
for(pair<ll,int> p:qv) res[p.se]=(p.fi%pr[0]==0);
} else if(pr.size()==2){
ll x=pr[0],y=pr[1],ivy=qpow(y%x,x-2,x);
for(pair<ll,int> p:qv){
int bmin=1ll*(p.fi%x)*ivy%x;
res[p.se]=(p.fi>=1ll*bmin*y);
}
} else {
int x=pr[0];memset(dis,63,sizeof(dis));dis[0]=0;
priority_queue<pair<ll,int>,vector<pair<ll,int> >,greater<pair<ll,int> > > q;
q.push(mp(0,0));
while(!q.empty()){
pair<ll,int> p=q.top();q.pop();
int cur=p.se;
for(int i=1;i<pr.size();i++){
if(dis[(cur+pr[i])%x]>dis[cur]+pr[i]){
dis[(cur+pr[i])%x]=dis[cur]+pr[i];
q.push(mp(dis[(cur+pr[i])%x],(cur+pr[i])%x));
}
}
}
for(pair<ll,int> p:qv) res[p.se]=(p.fi>=dis[p.fi%x]);
}
}
for(int i=1;i<=qu;i++) printf("%s\n",(res[i])?"YES":"NO");
return 0;
}

Codeforces 986F - Oppa Funcan Style Remastered(同余最短路)的更多相关文章

  1. [Codeforces 485F] Oppa Funcan Style Remastered

    [题目链接] https://codeforces.com/contest/986/problem/F [算法] 不难发现 , 每个人都在且仅在一个简单环中 , 设这些环长的长度分别为 A1, A2 ...

  2. CF986F Oppa Funcan Style Remastered

    CF986F Oppa Funcan Style Remastered 不错的图论转化题! 题目首先转化成:能否用若干个k的非1因数的和=n 其次,因数太多,由于只是可行性,不妨直接都用质因子来填充! ...

  3. 「CF986F」 Oppa Funcan Style Remastered

    「CF986F」 Oppa Funcan Style Remastered Link 首先发现分解成若干个 \(k\) 的因数很蠢,事实上每个因数都是由某个质因子的若干倍组成的,所以可以将问题转换为分 ...

  4. [CF986F]Oppa Funcan Style Remastered[exgcd+同余最短路]

    题意 给你 \(n\) 和 \(k\) ,问能否用 \(k\) 的所有 \(>1\) 的因子凑出 \(n\) .多组数据,但保证不同的 \(k\) 不超过 50 个. \(n\leq 10^{1 ...

  5. codeforces986F Oppa Funcan Style Remastered【线性筛+最短路】

    容易看出是用质因数凑n 首先01个因数的情况可以特判,2个的情况就是ap1+bp2=n,b=n/p2(mod p1),这里的b是最小的特解,求出来看bp2<=n则有解,否则无解 然后剩下的情况最 ...

  6. Codeforces 516E - Drazil and His Happy Friends(同余最短路)

    Codeforces 题面传送门 & 洛谷题面传送门 首先思考一个非常简单的性质:记 \(d=\gcd(n,m)\),那么每次在一起吃完饭的男女孩编号必定与 \(d\) 同余,而根据斐蜀定理可 ...

  7. 【66测试20161115】【树】【DP_LIS】【SPFA】【同余最短路】【递推】【矩阵快速幂】

    还有3天,今天考试又崩了.状态还没有调整过来... 第一题:小L的二叉树 勤奋又善于思考的小L接触了信息学竞赛,开始的学习十分顺利.但是,小L对数据结构的掌握实在十分渣渣.所以,小L当时卡在了二叉树. ...

  8. HDU 6071 Lazy Running (同余最短路 dij)

    Lazy Running Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 524288/524288 K (Java/Others)To ...

  9. BZOJ 2118 墨墨的等式 (同余最短路)

    题目大意:已知B的范围,求a1x1+a2x2+...+anxn==B存在非负正整数解的B的数量,N<=12,ai<=1e5,B<=1e12 同余最短路裸题 思想大概是这样的,我们选定 ...

随机推荐

  1. JS最简单的定时累加计数器

    js代码: 1 var timer , k = 0; 2 function star() { 3 k += 1; 4 document.getElementById("num"). ...

  2. 深入理解Java虚拟机之类加载机制篇

    概述 ​ 虚拟机把描述类的数据从 Class 文件加载到内存中,并对数据进行校验.转换解析和初始化,最终形成可以被虚拟机直接使用的Java类型,就是虚拟机的类加载机制. ​ 在Java语言里面,类型的 ...

  3. 【数据结构与算法Python版学习笔记】树——利用二叉堆实现优先级队列

    概念 队列有一个重要的变体,叫作优先级队列. 和队列一样,优先级队列从头部移除元素,不过元素的逻辑顺序是由优先级决定的. 优先级最高的元素在最前,优先级最低的元素在最后. 实现优先级队列的经典方法是使 ...

  4. Convolutional Neural Network-week2编程题2(Residual Networks)

    1. Residual Networks(残差网络) 残差网络 就是为了解决深网络的难以训练的问题的. In this assignment, you will: Implement the basi ...

  5. Canal的简单使用

    Canal的简单实用 一.背景 二.canal的工作原理 三.安装canal 1.mysql配置相关 1.检测binlog是否开启 2.mysql开启binlog 3.创建canal用户 2.cana ...

  6. netty中使用protobuf实现多协议的消息

    在我们使用 netty 的过程中,有时候为了高效的传输数据,经常使用 protobuf 进行数据的传输,netty默认情况下为我们实现的 protobuf 的编解码,但是默认的只能实现单个对象的编解码 ...

  7. mongodb的索引操作

    在mongodb中,当我们一个集合中的数据量非常大时,比如几百万条数据,如果不使用索引,对数据的查询就会进行全表扫描,这个时候查询的速度就会非常的慢,此时我们就需要为集合建立上索引,从而加快查询的速度 ...

  8. Py高级函数和方法

    Map() Redece() Dir() __len__   ---->>>  len() getattr().setattr() 以及   hasattr() 参考廖雪峰----- ...

  9. 【Java】 List和Array转换

    List转Array toArray 首先展示初学者容易犯的错误示例 List<String> strList = new ArrayList<>(); strList.add ...

  10. map2bean & bean2map

    1,自己实现: /** * @author xx * @since 2020/7/8 */ @Slf4j public class JavaBeanUtils { /** * 实体类转map * 效率 ...