传送门

Description

Sheng bill有着惊人的心算能力,甚至能用大脑计算出两个巨大的数的GCD(最大公约 数)!因此他经常和别人比赛计算GCD。有一天Sheng bill很嚣张地找到了你,并要求和你比 赛,但是输给Sheng bill岂不是很丢脸!所以你决定写一个程序来教训他。

Input

共两行: 第一行:一个数A。 第二行:一个数B。

Output

一行,表示A和B的最大公约数。

Sample Input

12
54

Sample Output

6

Hint

对于100%的数据,0 < A , B ≤ 10 ^ 10000。

Solution

如果你觉得这是个裸的gcd的话,你会发现高精度乘除取余在这么大的位数下就是找死。考虑使用高精度运算复杂度更低的更损相减法。但是朴素的更损相减法是\(O(n)\)的。需要进行优化。

考虑对于两个数\(a,b\)两数共可能出现以下情况:

(不妨设\(a>b\))

1、\(a\)是偶数,\(b\)不是。那么\(gcd(a,b)=gcd(\frac{a}{2},b)\)。

2、\(a\)不是偶数,\(b\)是。那么\(gcd(a,b)=gcd(a,\frac{b}{2})\)。

3、\(a,b\)都是偶数,那么\(gcd(a,b)=2~\times~gcd(\frac{a}{2},\frac{b}{2})\)。

4、\(a,b\)都不是偶数,那么应用更损相减法,\(gcd(a,b)=gcd(b,a-b)\)。

考虑这么做的复杂度。当两个数是奇数的时候,需要更损相减法,两个奇数做差的答案是一个偶数。每次其中一个数除二后最多做一次更损相减。

考虑一个数除二的最大次数是\(logn\)的。所以优化后更损相减部分的复杂度是\(O(logn)\)的。

Code

// luogu-judger-enable-o2
#include<cstdio>
#include<cstring>
#define rg register
#define ci const int
#define cl const long long int typedef long long int ll; namespace IO {
char buf[50];
} template<typename T>
inline void qr(T &x) {
char ch=getchar(),lst=' ';
while(ch>'9'||ch<'0') lst=ch,ch=getchar();
while(ch>='0'&&ch<='9') x=(x<<1)+(x<<3)+(ch^48),ch=getchar();
if (lst=='-') x=-x;
} template<typename T>
inline void write(T x,const char aft,const bool pt) {
if(x<0) {putchar('-');x=-x;}
int top=0;
do {
IO::buf[++top]=x%10+'0';
x/=10;
} while(x);
while(top) putchar(IO::buf[top--]);
if(pt) putchar(aft);
} template <typename T>
inline T mmax(const T a,const T b) {if(a>b) return a;return b;}
template <typename T>
inline T mmin(const T a,const T b) {if(a<b) return a;return b;}
template <typename T>
inline T mabs(const T a) {if(a<0) return -a;return a;} template <typename T>
inline void mswap(T &a,T &b) {T temp=a;a=b;b=temp;} struct Bignum {
short int num[10010],len;
void clear() {memset(num,0,sizeof num);len=0;}
void operator-=(const Bignum &others) {
for(rg int i=1;i<=this->len;++i) {
this->num[i]-=others.num[i];
while(this->num[i]<0) {
this->num[i]+=10,--this->num[i+1];
}
}
while(!this->num[this->len]) --this->len;
if(!this->len) ++this->len;
}
bool operator!=(const Bignum &others) {
if(this->len!=others.len) return true;
for(rg int i=1;i<=this->len;++i) if(this->num[i] != others.num[i]) return true;
return false;
}
bool operator<(const Bignum &others) {
if(this->len!=others.len) return this->len<others.len;
for(rg int i=len;i;--i) if(this->num[i]!=others.num[i]) return this->num[i]<others.num[i];
return false;
}
Bignum operator*(const Bignum &others) {
Bignum _ans;_ans.clear();
int llen=this->len+others.len+5;
for(rg int i=1;i<=this->len;++i) {
for(rg int j=1;j<=others.len;++j) {
_ans.num[i+j-1]+=this->num[i]*others.num[j];
}
}
for(rg int i=1;i<=llen;++i) {
_ans.num[i+1]+=_ans.num[i]/10;
_ans.num[i]%=10;
}
_ans.len=llen;
while(!_ans.num[_ans.len]) --_ans.len;
if(!_ans.len) _ans.len=1;
return _ans;
}
};
Bignum a,b,ans; char MU[10010];
int cnt; void dv(Bignum&);
void mul(Bignum&);
void init(Bignum&,int);
void print(Bignum&); int main() {
scanf("%s",MU+1);init(a,strlen(MU+1));
scanf("%s",MU+1);init(b,strlen(MU+1));
ans.num[1]=1;ans.len=1;
while(a != b) {
bool flag=false;
if(!((int(a.num[1])) & 1)) {dv(a);flag=true;}
if(!((int(b.num[1])) & 1)) {dv(b);if(flag) mul(ans);flag=true;}
if(flag) continue;
// print(a);puts("\nemm");print(b);putchar('\n');
if(a < b) {b-=a;}
else {a-=b;}
}
ans=ans*a;
print(ans);putchar('\n');
return 0;
} void init(Bignum &k,int l) {
for(rg int i=l;i;--i) k.num[++k.len]=MU[i]-'0';
} void dv(Bignum &k) {
int lst=0;
for(rg int i=k.len;i;--i) {
int _temp=k.num[i]+(lst<<1)+(lst<<3);
int _ans=_temp/2;
if(_temp & 1) lst=1;else lst=0;
k.num[i]=_ans;
}
while(!k.num[k.len]) --k.len;
return;
} void mul(Bignum &k) {
int lst=0;k.len+=2;
for(rg int i=1;i<=k.len;++i) {
k.num[i]*=2;
k.num[i]+=lst;
lst=k.num[i]/10;k.num[i]%=10;
}
while(!k.num[k.len]) --k.len;
if(!k.len) ++k.len;
return;
} void print(Bignum &k) {
// printf("%d\n",k.len);
for(rg int i=k.len;i>0;--i) putchar(k.num[i]+'0');
}

Summary

在高精度运算下求gcd,如果两数特大可以考虑使用优化后的更损相减。但是需要注意的是更损相减法的常数在极端情况下大概会达到\(4\)倍。而欧几里得算法的常数小于\(1\)。在取模计算量可以忽略的情况下应尽量选择欧几里得算法。

再说一句GCD真的不是什么党……

【数论数学】【P2152】【SDOI2009】Super GCD的更多相关文章

  1. [bzoj]2705: [SDOI2012]Longge的问题[数论][数学][欧拉函数][gcd]

    [bzoj]P2705 OR [luogu]P2303 Longge的问题 Description Longge的数学成绩非常好,并且他非常乐于挑战高难度的数学问题.现在问题来了:给定一个整数N,你需 ...

  2. UVA12716 GCD XOR 数论数学构造

    版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/u010682557/article/details/36204645 题目给你一个N,让你求 两个数 ...

  3. 数学--数论--HDU 5019 revenge of GCD

    Revenge of GCD Problem Description In mathematics, the greatest common divisor (gcd), also known as ...

  4. 数学--数论--HDU1792A New Change Problem(GCD规律推导)

    A New Change Problem Problem Description Now given two kinds of coins A and B,which satisfy that GCD ...

  5. 数学 赛码 1010 GCD

    题目传送门 /* 数学:官方题解 首先,数组中每个元素至少是1 然后对于任意一个询问Li, Ri, Ansi, 说明Li ~ Ri中的元素必定是Ansi的倍数,那么只需将其与Ansi取最小公倍数即可 ...

  6. 【数学】XMU 1597 GCD

    题目链接: http://acm.xmu.edu.cn/JudgeOnline/problem.php?id=1597 题目大意: 求(am-bm, an-bn),结果取模1000000007,a,b ...

  7. noip2017考前基础复习——数论数学

    ·最大公约数 gcd 辗转相除法  gcd(a,b)=gcd(b,a%b) int gcd(int x,int y){ ) return x; return gcd(y,x%y); } 效率O(log ...

  8. P2152 [SDOI2009]SuperGCD 未完成

    辗转相减求a,b的gcd其实可以优化的: 1.若a为偶数,b为奇数:gcd(a,b)=gcd(a/2,b) 2.若a为奇数,b为偶数:gcd(a,b)=gcd(a,b/2) 3.若a,b都是偶数:gc ...

  9. 数学:莫比乌斯反演-GCD计数

    Luogu3455:莫比乌斯反演进行GCD计数 莫比乌斯反演就是用来解决这一类问题的,通常f函数是要求的那个,F函数是显然的 这样利用F的结果就可以推出来f的结果 在计算结果的时候整除分快儿一下就可以 ...

随机推荐

  1. 银行系统ps:不太完善,蟹蟹评论

    # 主程序运行 import time from guanli import GuanLi from atm import ATM from user import User def main(): ...

  2. 利用nohup后台运行jar文件包程序

    Linux 运行jar包命令如下: 方式一: java -jar XXX.jar特点:当前ssh窗口被锁定,可按CTRL + C打断程序运行,或直接关闭窗口,程序退出 那如何让窗口不锁定? 方式二 j ...

  3. nginx web服务器的安装使用

    nginx是一个web服务器(高性能web服务器),类似于apache服务器和iis服务器,由于nginx服务器具有轻量级高并发的特点,目前nginx的使用已经超越了apache. nginx介绍:n ...

  4. 使用深度学习来破解 captcha 验证码(转)

    使用深度学习来破解 captcha 验证码 本项目会通过 Keras 搭建一个深度卷积神经网络来识别 captcha 验证码,建议使用显卡来运行该项目. 下面的可视化代码都是在 jupyter not ...

  5. hibernate 异常a different object with the same identifier value was already associated with the session

    在使用hibernate的时候发现了一个问题,记录一下解决方案. 前提开启了事务和事务间并无commit,进行两次save,第二次的时候爆出下面的异常a different object with t ...

  6. SpringCloud IDEA 教学 (四) 断路器(Hystrix)

    写在开始 在SpringCloud项目中,服务之间相互调用(RPC Remote Procedure Call —远程过程调用),处于调用链路底层的服务产生不可用情况时,请求会产生堆积使得服务器线程阻 ...

  7. php 安全方面面试题

    1 MySQL数据库作发布系统的存储,一天五万条以上的增量,预计运维三年,怎么优化? a. 设计良好的数据库结构,允许部分数据冗余,尽量避免join查询,提高效率.b. 选择合适的表字段数据类型和存储 ...

  8. 在Excel里面,单元格里输入公式后只显示公式本身,不显示结果,怎么办

    这种情况是对Excel进行了设置,设置的就是在单元格中只显示公式,不显示结果,解决的办法有两个: 1 用快捷键CTR+~ 2 点击"公式"选项卡,然后反选里面的"显示公式 ...

  9. 软件工程第六周psp

    1.psp表格 类别 任务 开始时间 结束时间 中断时间 delta时间 立会 讲技术文档,分配任务 10月20日16:17 10月20日16:50 0 33分钟 准备工作 根据任务查资料 10月20 ...

  10. C与C++,面向过程与面向对象

    C与C++在电梯处理上的不同 (注:个人理解) 对比区别: C语言程序制定具体流程,按流程逐步进行. C++程序将过程结构化,需要使用时利用接口访问与调用不同功能的类结构结构. 电梯类代码 电梯类定义 ...