题意:定义F(a,0) = 0,F(a,b) = 1 + F(a,b - GCD(a,b)。给定 x 和 y (<=1e12)求F(x,y)。

题解:a=A*GCD(a,b) b=B*GCD(a,b),那么b-GCD(a,b) = (B-1)*GCD(a,b),如果此时A和B-1依然互质,那么GCD不变下一次还是要执行b-GCD(a,b)。那么GCD什么时候才会变化呢?就是说找到一个最小的S,使得(B-S)%T=0其中T是a的任意一个因子。变形得到:B%T=S于是我们知道S=min(B%T)。也就是说b剪掉了S次相同的一个GCD之后,ab有了新的GCD。新的GCD等于原来的GCD*T,可以把a、b都/T,同时GCD*T,这样问题化归为上述同样的问题,进行迭代

当B和A公约数不为1的时候(开始的时候,或者B减了一定次数1的时候),就相当于A和B同除以gcd(A,B),然后B继续一次减1。

这样只要每次计算出每次B要减多少次1才能和A有不为1的公约数。

那么预处理出A的质因数,然后每次对A的质因数判断一下,哪个最近(也就是模最小)即可。

#include <bits/stdc++.h>
using namespace std;
typedef long long LL;
int main() {
LL x, y;
cin >> x >> y;
LL g = __gcd(x, y);
x /= g, y /= g;
vector<LL> a;
for (LL i = ; i * i <= x; ++ i) {
while (x % i == ) {
x /= i;
a.push_back(i);
}
}
if (x > ) a.push_back(x);
LL ans = ;
while (y) {
LL g = y;
for (LL i : a) {
g = min(g, y % i);
}
ans += g;
y -= g;
vector<LL> b;
for (LL i : a) {
if (y % i == ) {
y /= i;
}
else {
b.push_back(i);
}
}
a.swap(b);
}
cout << ans << endl;
}

最新文章

  1. JAVA面向对象
  2. CentOS 6.5 下离线安装nginx
  3. DDD领域驱动设计基本理论知识总结
  4. 【整理】 JavaScript模块化规范AMD 和 CMD 的区别有哪些?
  5. [原创][LaTex]LaTex学习笔记之框架及宏包
  6. 【转载】gcc 使用中常用的参数及命令
  7. 使用C#和.NET 4编写的并行应用程序“多核并发编程的规则”
  8. 对RabbitMQ.Client进行一下小小的包装,绝对实用方便
  9. calendar中set方法和静态属性带来的坑
  10. seleium_元素定位
  11. ios高级开发之多线程(二)NSThread技术
  12. Spring AOP 的实现机制
  13. PHP中判断变量是否存在的方式
  14. 【pycharm 密钥】pycharm 2017 密钥
  15. ubuntu16安装MySQL
  16. SRTP讨论
  17. Mysqldump 参数大全
  18. 浅析Entity FrameWork性能优化
  19. Redis源代码剖析--对象object
  20. ASP.NET Web API编程——文件下载

热门文章

  1. Beego框架的一条神秘日志引发的思考
  2. 关于react native 路由传值及回调方法的理解
  3. &lt;转&gt;经典测试用例:电梯、杯子、桌子、洗衣机
  4. Smarty快速入门
  5. android webview 访问 https 页面
  6. 关联SecureCRT
  7. Linux学习笔记(16)Linux前后台进程切换(fg/bg/jobs/ctrl+z)
  8. 第一次编译ffmpeg
  9. react生态常用库分类
  10. Codeforces 1196D2. RGB Substring (hard version)