BZOJ 1876: [SDOI2009]SuperGCD
1876: [SDOI2009]SuperGCD
Time Limit: 4 Sec Memory Limit: 64 MB
Submit: 3060 Solved: 1036
[Submit][Status][Discuss]
Description
Input
Output
Sample Input
54
Sample Output
HINT
对于20%的数据,0 < A , B ≤ 10 ^ 18。
对于100%的数据,0 < A , B ≤ 10 ^ 10000。
Source
首先普通的GCD显然是不行的,而高精度又难以进行mod运算(至少我不会写),所以我们需要一个神奇的求最大公约数的方法。
好像是在60年代,某位前辈深感欧几里得算法求GCD的不便,对其进行了改进,适合大整数(大素数)的GCD过程,而且易于理解。
首先,我们需要了解以下知识:
1. GCD(a, b) = k * GCD(a / k, b / k) 其中,k是a和b的一个公因数。
2. GCD(a, b) = GCD(a / k, b) 其中,k仅为a的因数,而非b的因数。
3. GCD(a, b) = GCD(a - b, b) 其中,a大于等于b。
对于知识1和2,当k=2的时候,有:
如果a,b都是偶数,则GCD(a, b) = 2 * GCD(a / 2, b / 2)。
如果a,b中只有一个偶数,则GCD(a, b) = GCD(a / 2, b)。(假设a是偶数)
而*2和/2的运算可以通过位运算快速实现(当然,不这么做也是可以过的)。
#include <bits/stdc++.h> const int siz = ;
const int mod = ; struct Int
{
int num[siz], len; inline Int(void) {
memset(num, , sizeof(num)), len = ;
} inline Int(char *s) {
int l = strlen(s);
len = (l + ) >> ;
memset(num, , sizeof(num));
for (int i = ; i < l; ++i)
(num[(l - i + ) >> ] *= ) += s[i] - ;
} inline friend bool operator < (const Int &a, const Int &b) {
if (a.len != b.len)
return a.len < b.len;
for (int i = a.len; i; --i)
if (a.num[i] != b.num[i])
return a.num[i] < b.num[i];
return false;
} inline void sub(const Int &a) {
for (int i = ; i <= a.len; ++i)
num[i] -= a.num[i];
for (int i = ; i <= len; ++i)
if (num[i] < )
num[i] += mod, --num[i + ];
while (len > && !num[len])--len;
} inline void mul2(void) {
for (int i = ; i <= len; ++i)
num[i] <<= ;
for (int i = ; i <= len; ++i)
if (num[i] >= mod)
num[i] -= mod, ++num[i + ];
while (num[len + ])++len;
} inline void div2(void) {
for (int i = len; i; --i) {
if (i > && (num[i] & ))
num[i - ] += mod;
num[i] >>= ;
}
while (len > && !num[len])--len;
} inline bool even(void) {
return !(num[] & );
} inline bool zero(void) {
return len == && !num[];
} inline void print(void) {
printf("%d", num[len]);
for (int i = len - ; i; --i)
printf("%08d", num[i]);
putchar('\n');
}
}; inline Int gcd(Int a, Int b)
{
int bit = ;
while (true)
{
if (a.zero()) {
while (bit--)b.mul2();
return b;
}
if (b.zero()) {
while (bit--)a.mul2();
return a;
}
int flag = ;
if (a.even())++flag, a.div2();
if (b.even())++flag, b.div2();
if (flag == )
++bit;
else {
if (a < b)
b.sub(a);
else
a.sub(b);
}
}
} char a[];
char b[]; signed main(void)
{
scanf("%s", a);
scanf("%s", b);
gcd(Int(a), Int(b)).print();
}
@Author: YouSiki
BZOJ 1876: [SDOI2009]SuperGCD的更多相关文章
- bzoj 1876 [SDOI2009]SuperGCD(高精度+更相减损)
1876: [SDOI2009]SuperGCD Time Limit: 4 Sec Memory Limit: 64 MBSubmit: 2384 Solved: 806[Submit][Sta ...
- BZOJ 1876: [SDOI2009]SuperGCD( 更相减损 + 高精度 )
更相减损,要用高精度.... --------------------------------------------------------------- #include<cstdio> ...
- BZOJ 1876 [SDOI2009] SuperGcd | PY好题
题面就是让你求两个超级大整数,求GCD 题解: 题目本意应该是出题人想考考高精度取膜 但是可以通过一种神奇的Stein算法来做 由J. Stein 1961年提出的Stein算法很好的解决了欧几里德算 ...
- 【BZOJ1876】[SDOI2009]SuperGCD(数论,高精度)
[BZOJ1876][SDOI2009]SuperGCD(数论,高精度) 题面 BZOJ 洛谷 题解 那些说数论只会\(gcd\)的人呢?我现在连\(gcd\)都不会,谁来教教我啊? 显然\(gcd\ ...
- bzoj 1879: [Sdoi2009]Bill的挑战
题目链接 bzoj 1879: [Sdoi2009]Bill的挑战 题解 n<=15,装压吧 对所有字符串进行装压 可以预处理一个数组can[i][j]表示所有的字符串中,有哪些可以在第i位匹配 ...
- [SDOI2009][BZOJ 1876]SuperGCD
Description Sheng bill有着惊人的心算能力,甚至能用大脑计算出两个巨大的数的GCD(最大公约 数)!因此他经常和别人比 赛计算GCD.有一天Sheng bill很嚣张地找到了你,并 ...
- bzoj千题计划288:bzoj1876: [SDOI2009]SuperGCD
http://www.lydsy.com/JudgeOnline/problem.php?id=1876 高精压位GCD 对于 GCD(a, b) a>b 若 a 为奇数,b 为偶数,GCD ...
- BZOJ 1876 SuperGCD
Description Sheng bill有着惊人的心算能力,甚至能用大脑计算出两个巨大的数的GCD(最大公约数)!因此他经常和别人比赛计算GCD.有一天Sheng bill很嚣张地找到了你,并要求 ...
- 「BZOJ 1876」「SDOI 2009」SuperGCD「数论」
题意 求\(\gcd(a, b)\),其中\(a,b\leq10^{10000}\) 题解 使用\(\text{Stein}\)算法,其原理是不断筛除因子\(2\)然后使用更相减损法 如果不筛\(2\ ...
随机推荐
- Android 手机卫士1--实现splash页面
1.minSdkVersion.targetSdkVersion.maxSdkVersion.target API level四个数值到底有什么区别? minSdkVersion, maxSdkVer ...
- Hibernate入门详解
学习Hibernate ,我们首先要知道为什么要学习它?它有什么好处?也就是我们为什么要学习框架技术? 还要知道 什么是Hibernate? 为什么要使用Hibernate? Hib ...
- 学习廖雪峰的git教程
地址:http://www.liaoxuefeng.com/wiki/0013739516305929606dd18361248578c67b8067c8c017b000 1.git add:添加文件 ...
- react native初步常见问题
首先按照资料一步步搭建环境运行,然后成功了,很激动,可是,安卓就是没这么容易成功,还是太年轻了 could not get batchedbridge, make sure your bundle i ...
- 【搬砖】安卓入门(2)- Java开发编程基础--进制转换和运算符
02.01_Java语言基础(常量的概述和使用)(掌握) A:什么是常量 在程序执行的过程中其值不可以发生改变 B:Java中常量的分类 字面值常量 自定义常量(面向对象部分讲) C:字面值常量的分类 ...
- iOS 语音朗读
//判断版本大于7.0 if ([[[UIDevice currentDevice] systemVersion] integerValue] >= 7.0) { NSStr ...
- 在C语言中利用PCRE实现正则表达式
1. PCRE简介 2. 正则表达式定义 3. PCRE正则表达式的定义 4. PCRE的函数简介 5. 使用PCRE在C语言中实现正则表达式的解析 6. PCRE函数在C语言中的使用小例子 1. P ...
- 【代码笔记】iOS-一个tableView,两个section
一,效果图. 二,工程图. 三,代码. RootViewController.h #import <UIKit/UIKit.h> @interface RootViewController ...
- 软件工程随笔(1)--jetbrain在软件工程中的应用
接下来几天我要写半年的软件工程学习后的感想,今天从介绍IDE开始.首先,本人至今为止全部项目都是在mypclise上完成的.本人采用myeclipse唯一的原因就是它使用方便.但是,我也承认myecl ...
- 感受C# 的魅力,将一坨代码写成一行
摘自MSDN :https://msdn.microsoft.com/zh-cn/library/bb549151(v=vs.100).aspx 1.平时定义一个委托 using System; // ...