POJ 2429 GCD & LCM Inverse (Pollard rho整数分解+dfs枚举)
题意:给出a和b的gcd和lcm,让你求a和b。按升序输出a和b。若有多组满足条件的a和b,那么输出a+b最小的。
思路:lcm=a*b/gcd
lcm/gcd=a/gcd*b/gcd
可知a/gcd与b/gcd互质,由此我们可以先用Pollard_rho法对lcm/gcd进行整数分解,
然后对其因子进行深搜找出符合条件的两个互质的因数,然后再都乘以gcd即为输出答案。
#include <iostream>
#include <stdio.h>
#include <algorithm>
#include <string.h>
#include <cstdlib>
#include <cmath> using namespace std;
const int maxn=;
long long lcm,gcd,n;
long long factor[maxn],fac[maxn];
long long Max;
long long aa,bb;
int cnt=,len=;
long long multi(long long a,long long b,long long mod){
long long ret=;
while(b){
if(b&){
ret=(ret+a)%mod;
}
a=a<<;
a=a%mod;
b=b>>;
}
return ret;
}
long long quickPow(long long a,long long b,long long mod){
long long ret=;
while(b){
if(b&)
ret=multi(ret,a,mod);
a=multi(a,a,mod);
b=b>>;
}
return ret;
}
bool witness(long long a,long long n){
long long m=n-;
int j=;
while(!(m&)){
j++;
m=m>>;
}
long long x=quickPow(a,m,n);
if(x==||x==n-)
return false;
while(j--){
x=multi(x,x,n);
if(x==n-)
return false;
}
return true;
}
//判定n是否为素数,若是,返回true
bool Miller_Rabin(long long n){
if(n==)
return true;
if(n< || !(n&))
return false;
for(int i=;i<;i++){
long long a=rand()%(n-)+;
if(witness(a,n))
return false;
}
return true;
}
long long Gcd(long long a,long long b){
return b==?a:Gcd(b,a%b);
} long long pollard_rho(long long n,int c){
long long x,y,d,i=,k=;
x=rand()%(n-)+;
y=x;
while(){
i++;
x=(multi(x,x,n)+c)%n;
d=Gcd((y-x+n)%n,n);
if(<d && d<n)
return d;
if(y==x)
return n;
if(i==k){
y=x;
k=k<<;
}
}
}
//对n进行质因数分解
void factorFind(long long n,int k){
if(n==)
return;
if(Miller_Rabin(n)){
factor[cnt++]=n;
return;
}
long long p=n;
while(p>=n)
p=pollard_rho(p,k--);
factorFind(p,k);
factorFind(n/p,k);
}
//dfs枚举a和b的值
void dfs(long long a,long long b,int k){
//如果目前a+b的值已经大于Max了,那么就直接return
if(a+b>=Max){
return;
}
if(k==len+){
if(a+b<Max){
Max=a+b;
aa=a;
bb=b;
}
return;
}
dfs(a*fac[k],b,k+);
dfs(a,b*fac[k],k+);
} int main()
{
long long a,b;
while(scanf("%I64d%I64d",&gcd,&lcm)!=EOF){
n=lcm/gcd;
//若n为素数的话,那aa=1,bb=lcm/gcd。
if(Miller_Rabin(n)){
//一开始WA的原因是,当gcd=1的时候,我是输出1,lcm
//后来才发现,比如1 40,输出5 8
aa=;
bb=n;
printf("%I64d %I64d\n",aa*gcd,bb*gcd);
}
else if(lcm==gcd){
printf("%I64d %I64d\n",gcd,gcd);
}
else{
cnt=;
factorFind(n,);
sort(factor,factor+cnt);
len=;
fac[]=factor[];
//这里由于对n分解质因数时,a/gcd 和 b/gcd肯定两两互质,所以先将相同的素因子先乘起来
//这样后面dfs时aa和bb肯定是两两互质的
for(int i=;i<cnt;i++){
if(factor[i]==factor[i-]){
fac[len]=fac[len]*factor[i];
}
else{
len++;
fac[len]=factor[i];
}
} //一开始RE,原因如下:
//当lcm=gcd时,n=lcm/gcd=1,那么a=fac[0]=0,n/a的时候会导致RE。。。
a=fac[];
b=;
Max=n/a+a;
aa=a;
bb=n/a;
dfs(a,b,);
if(aa>bb){
long long tmp=aa;
aa=bb;
bb=tmp;
}
printf("%I64d %I64d\n",aa*gcd,bb*gcd);
}
}
return ;
}
POJ 2429 GCD & LCM Inverse (Pollard rho整数分解+dfs枚举)的更多相关文章
- [POJ 2429] GCD & LCM Inverse
GCD & LCM Inverse Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 10621 Accepted: ...
- POJ 2429 GCD & LCM Inverse(Pollard_Rho+dfs)
[题目链接] http://poj.org/problem?id=2429 [题目大意] 给出最大公约数和最小公倍数,满足要求的x和y,且x+y最小 [题解] 我们发现,(x/gcd)*(y/gcd) ...
- POJ2429 GCD & LCM Inverse pollard_rho大整数分解
Given two positive integers a and b, we can easily calculate the greatest common divisor (GCD) and t ...
- POJ 2429 GCD & LCM Inverse(Miller-Rabbin素性测试,Pollard rho质因子分解)
x = lcm/gcd,假设答案为a,b,那么a*b = x且gcd(a,b) = 1,因为均值不等式所以当a越接近sqrt(x),a+b越小. x的范围是int64的,所以要用Pollard_rho ...
- POJ:2429-GCD & LCM Inverse(素数判断神题)(Millar-Rabin素性判断和Pollard-rho因子分解)
原题链接:http://poj.org/problem?id=2429 GCD & LCM Inverse Time Limit: 2000MS Memory Limit: 65536K To ...
- poj 2429 GCD & LCM Inverse 【java】+【数学】
GCD & LCM Inverse Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 9928 Accepted: ...
- POJ2429_GCD & LCM Inverse【Miller Rabin素数測试】【Pollar Rho整数分解】
GCD & LCM Inverse Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 9756Accepted: 1819 ...
- POJ1811_Prime Test【Miller Rabin素数测试】【Pollar Rho整数分解】
Prime Test Time Limit: 6000MS Memory Limit: 65536K Total Submissions: 29193 Accepted: 7392 Case Time ...
- POJ1811_Prime Test【Miller Rabin素数測试】【Pollar Rho整数分解】
Prime Test Time Limit: 6000MS Memory Limit: 65536K Total Submissions: 29193 Accepted: 7392 Case Time ...
随机推荐
- Cocos2d-X3.0 刨根问底(二)----- 从HelloWorld开始
小鱼习惯直接从代码实例来学习一套成型的引擎库. 运行cpp-empty-test 一个典型的HelloWorld程序翻看代码结构 看到了 main.h与main.cpp文件就从这里开始 #ifndef ...
- rqnoj71 拔河比赛
题目描述 superwyh的学校要举行拔河比赛,为了在赛前锻炼大家,老师决定把班里所有人分为两拨,进行拔河因为为锻炼所以为了避免其中一方的实力过强老师决定以体重来划分队伍,尽 量保持两个队伍的体重差最 ...
- bzoj1670 Usaco2006 Building the Moat护城河的挖掘 [凸包模板题]
Description 为了防止口渴的食蚁兽进入他的农场,Farmer John决定在他的农场周围挖一条护城河.农场里一共有N(8<=N<=5,000)股泉水,并且,护城河总是笔直地连接在 ...
- Linux System Log Collection、Log Integration、Log Analysis System Building Learning
目录 . 为什么要构建日志系统 . 通用日志系统的总体架构 . 日志系统的元数据来源:data source . 日志系统的子安全域日志收集系统:client Agent . 日志系统的中心日志整合系 ...
- C/C++ 跨平台交叉编译、静态库/动态库编译、MinGW、Cygwin、CodeBlocks使用原理及链接参数选项
目录 . 引言 . 交叉编译 . Cygwin简介 . 静态库编译及使用 . 动态库编译及使用 . MinGW简介 . CodeBlocks简介 0. 引言 UNIX是一个注册商标,是要满足一大堆条件 ...
- javascript自动转换大小写
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN"> <HTML> <HEAD ...
- 如何在 Ubuntu 14.04 里面配置 chroot 环境
你可能会有很多理由想要把一个应用.一个用户或者一个环境与你的 Linux 系统隔离开来.不同的操作系统有不同的实现方式,而在 Linux 中,一个典型的方式就是 chroot 环境. 在这份教程中,我 ...
- Linux 内核通知链机制的原理及实现
一.概念: 大多数内核子系统都是相互独立的,因此某个子系统可能对其它子系统产生的事件感兴趣.为了满足这个需求,也即是让某个子系统在发生某个事件时通知其它的子 系统,Linux内核提供了通知链的机制.通 ...
- POJ3744Scout YYF I(求概率 + 矩阵快速幂)
Scout YYF I Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 6757 Accepted: 1960 Descr ...
- poj1733Parity game
Parity game Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 7288 Accepted: 2833 Descr ...