题目大意

给出数a0, a1, b0, b1,求满足gcd(a0, x)=a1, lcm(b0, x)=b1的x的个数

解法一

枚举b1的因数,看看是否满足上述条件。

怎样枚举因数

试除法。对于1~sqrt(b1)的所有数i,若其能被b1整除,则i和b1/i便是b1的质因数。

注意

  • 是sqrt(b1),而不是sqrt(b1)+1。不要擅自主张,因为这样sqrt(b1)~sqrt(b1+1)中的能被b1整除的数就被算重了。
  • 求lcm时,应当为a/gcd(a,b)*b,而不是a*b/gcd(a,b)。因为a*b可能会爆int。
#include <cstdio>
#include <cstring>
#include <cstdarg>
#include <cmath>
#include <cassert>
using namespace std; const int MAX_P = 45000, MAX_P_CNT = 30000;
int Primes[MAX_P_CNT];
int A, B, Gcd, Lcm, AnsCnt, N; void GetPrime(int *prime, int n)
{
static bool NotPrime[MAX_P];
memset(NotPrime, false, sizeof(NotPrime));
int primeCnt = 0;
for (int i = 2; i <= n; i++)
{
if (!NotPrime[i])
prime[primeCnt++] = i;
for (int j = 0; j < primeCnt; j++)
{
if (prime[j] * i > n)
break;
NotPrime[prime[j] * i] = true;
if (i%prime[j] == 0)
break;
}
}
} int GetGcd(int a, int b)
{
return b ? GetGcd(b, a%b) : a;
} int GetLcm(int a, int b)
{
return a / GetGcd(a, b) * b;//易忘点:Not a * b / GetGcd(a, b)
} void Dfs(int pos, int prod, int curTime)
{
if (pos>=0 && Primes[pos] > N)
return;
assert(Lcm%prod == 0);
if (curTime > 0)
{
if (GetGcd(prod, A) == Gcd && GetLcm(prod, B) == Lcm)
AnsCnt++;
int d = Lcm / prod;
if (d != prod && GetGcd(d, A) == Gcd && GetLcm(d, B) == Lcm)
AnsCnt++;
}
for (int time = 0; prod <= N && Lcm%prod==0; time++)
{
Dfs(pos + 1, prod, time);
prod *= Primes[pos + 1];
}
} int main()
{
GetPrime(Primes, MAX_P);
int caseCnt;
scanf("%d", &caseCnt);
while (caseCnt--)
{
scanf("%d%d%d%d", &A, &Gcd, &B, &Lcm);
AnsCnt = 0;
N = sqrt(Lcm);//易忘点:Don't plus 1
Dfs(-1, 1, 1);
printf("%d\n", AnsCnt);
}
return 0;
}

解法二

每个数都可以用Πp[i]^m[i](p[i]是质数)表示,对于两个数a,b的最大公因数,它每一个p[i]的m[i]是a,b的m[i]中的较小值,最小公倍数是较大值。最终的结果便是每个p[i]计算出的满足较大较小关系的x的取值范围的大小。

注意

枚举n的质因数时,可能存在一个质数p>sqrt(n)。此时不要忘了它呀!

#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cstdarg>
using namespace std; const int MAX_P = 45000, MAX_P_CNT = 3000;
int Primes[MAX_P_CNT];
int TotPrime; int GetPrime(int *prime, int n)
{
static bool NotPrime[MAX_P];
memset(NotPrime, false, sizeof(NotPrime));
int primeCnt = 0;
for(int i=2; i<=n; i++)
{
if (!NotPrime[i])
prime[primeCnt++] = i;
for (int j = 0; j < primeCnt; j++)
{
if (i*prime[j] > n)
break;
NotPrime[i*prime[j]] = true;
if (i%prime[j] == 0)
break;
}
}
return primeCnt;
} int GetM(int prime, int &x)
{
int cnt = 0;
while (x%prime == 0)
{
cnt++;
x /= prime;
}
return cnt;
} void DoPrime(int prime, int &ans, int &a, int &gcd, int &b, int &lcm)
{
int mA = GetM(prime, a), mGcd = GetM(prime, gcd), mB = GetM(prime, b), mLcm = GetM(prime, lcm);
if (mA == mGcd && mB == mLcm && mGcd <= mLcm)
ans *= mLcm - mGcd + 1;//x *= prime ^ (mGcd ~ mLcm)
else if (mA == mGcd && mB < mLcm && mGcd <= mLcm)
ans *= 1;//x *= prime ^ (mLcm)
else if (mA > mGcd && mB == mLcm && mGcd <= mLcm)
ans *= 1;//x *= prime ^ (mGcd)
else if (mA > mGcd && mB < mLcm && mGcd == mLcm)
ans *= 1;//x *= (mGcd == mLcm)
else
ans *= 0;
} int Proceed(int a, int gcd, int b, int lcm)
{
int ans = 1, end = max(a, lcm);
for (int p = 0; Primes[p] <= end && ans && p<TotPrime; p++)
DoPrime(Primes[p], ans, a, gcd, b, lcm);
if (a > 1)
DoPrime(a, ans, a, gcd, b, lcm);
else if (lcm > 1 && lcm != a)
DoPrime(lcm, ans, a, gcd, b, lcm);
return ans;
} int main()
{
TotPrime = GetPrime(Primes, MAX_P);
int caseCnt;
scanf("%d", &caseCnt);
while (caseCnt--)
{
int a, gcd, b, lcm;
scanf("%d%d%d%d", &a, &gcd, &b, &lcm);
printf("%d\n", Proceed(a, gcd, b, lcm));
}
return 0;
}

  

luogu1072 Hankson的趣味题的更多相关文章

  1. 算法训练 Hankson的趣味题

    算法训练 Hankson的趣味题   时间限制:1.0s   内存限制:64.0MB        问题描述 Hanks 博士是BT (Bio-Tech,生物技术) 领域的知名专家,他的儿子名叫Han ...

  2. 1172 Hankson 的趣味题[数论]

    1172 Hankson 的趣味题 2009年NOIP全国联赛提高组  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 黄金 Gold 题解       题目描述 Descrip ...

  3. 1172 Hankson 的趣味题

    1172 Hankson 的趣味题 2009年NOIP全国联赛提高组  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 黄金 Gold 题解       题目描述 Descrip ...

  4. Codevs 1172 Hankson 的趣味题 2009年NOIP全国联赛提高组

    1172 Hankson 的趣味题 2009年NOIP全国联赛提高组 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 黄金 Gold 题目描述 Description Hanks 博 ...

  5. 一本通1626【例 2】Hankson 的趣味题

    1626:[例 2]Hankson 的趣味题 题目描述 Hanks 博士是BT (Bio-Tech,生物技术) 领域的知名专家,他的儿子名叫Hankson.现在,刚刚放学回家的Hankson 正在思考 ...

  6. 洛谷 P1072 Hankson 的趣味题 解题报告

    P1072 \(Hankson\)的趣味题 题目大意:已知有\(n\)组\(a0,a1,b0,b1\),求满足\((x,a0)=a1\),\([x,b0]=b1\)的\(x\)的个数. 数据范围:\( ...

  7. CH3201 Hankson的趣味题

    题意 3201 Hankson的趣味题 0x30「数学知识」例题 描述 Hanks博士是BT(Bio-Tech,生物技术)领域的知名专家,他的儿子名叫Hankson.现在,刚刚放学回家的Hankson ...

  8. luogu P1072 Hankson的趣味题

    题目链接 luogu P1072 Hankson 的趣味题 题解 啊,还是noip的题好做 额,直接推式子就好了 \(gcd(x,a_0)=a_1=gcd(\frac{x}{a_1},\frac{a_ ...

  9. 洛谷P1072 Hankson 的趣味题

    P1072 Hankson 的趣味题 题目描述 Hanks 博士是 BT (Bio-Tech,生物技术) 领域的知名专家,他的儿子名叫 Hankson.现在,刚刚放学回家的 Hankson 正在思考一 ...

随机推荐

  1. JavaScript 函数 伪数组 arguments

    一.函数 函数:函数就是将一些语言进行封装,然后通过调用的形式,执行这些语句. 函数的作用: 1.将大量重复的语句写在函数里,以后需要这些语句的时候,可以直接调用函数,避免重复劳动 2.简化编程,让变 ...

  2. css 继承性和层叠性

    css有两大特性:继承性和层叠性 继承性 面向对象语言都会存在继承的概念,在面向对象语言中,继承的特点:继承了父类的属性和方法.那么我们现在主要研究css,css就是在设置属性的.不会牵扯到方法的层面 ...

  3. 【转载】cocos2dx 中 Android NDK 加载动态库的问题

     原文地址:http://blog.csdn.net/sozell/article/details/10551309 cocos2dx 中 Android NDK 加载动态库的问题 闲聊 最近在接入各 ...

  4. .net中的WebForm引人MVC的控制器

    当下.net中比较火的模式MVC模式,说实话对于菜鸟的我还没有遇到一个比较健全的MVC模式的项目也是比较遗憾.偶然间在网上看到WebForm实现MVC中的模式(主要是控制器...)就学习了一波,下面是 ...

  5. retrofit post请求多文件,partMap用法

    1. APIService 定义注解 @Multipart @POST("cyxx/Feedback/add.do") Observable<ResponseBody> ...

  6. Unity引擎GUI之Image

    UGUI的Image等价于NGUI的Sprite组件,用于显示图片. 一.Image组件: Source Image(图像源):纹理格式为Sprite(2D and UI)的图片资源(导入图片后选择T ...

  7. VM虚拟机中Ubuntu中执行apt-get update失败的解决方法(可能有效)

    首先确保虚拟机是连接网络的,可以用ping命令检测一下看是否连通网络.采用nat网络的时候确保服务是开的. 如果之前执行过apt-get update命令但是失败了,执行一下      rm -rf ...

  8. dhtmlxtree动态加载节点数据的小随笔

    最近做了一个这个东西,颇有些感触,随笔记录一下自己的过程. 首先特别感谢:https://blog.csdn.net/cfl20121314/article/details/46852591,对我的帮 ...

  9. python3 str类型

    python3 的str就是unicode,只有encode函数,调用encode返回的是bytes. bytes只有decode函数,调用decode返回的是str.

  10. ANN:DNN结构演进History—RNN

    前言: CNN在图像处理领域的极大成功源于CNN的二维递进映射结构,通过训练多层卷积核来进行特征提取函数训练,在二维图像的稀疏表达和语义关联分析方面有天生的结构优势.而涉及时序问题的逻辑序列分析-边长 ...