AcWing 222. 青蛙的约会

一.题目链接

二.思路

a:青蛙a的起点
b:青蛙b的起点
m:青蛙a一次能跳多远
n:青蛙b一次能跳多远
L:一圈的距离
(b-a):a要追b多少米
(m-n):每跳一次,a能追b多少米

x是总共跳了多少次
y是a追b不一定会在一圈内追完,而是追了y圈
          (m - n)*x = b - a + y*L
    (m - n)*x - y*L = b - a
    ———————       —   —————
    已知         已知 已知

扩展欧几里得求的是
    ax+by=d
    a已知,b已知,d是a和b的最大公约数,求x,y
因此把上式的a替换乘(m-n),b替换成L。
式子变成(m-n)*x+(-y)*L=d
如果(b-a)%d不等于0,两只青蛙永远不会碰面
如果(b-a)%d等于零,把(m-n)*x+(-y)*L=d扩大(b-a)/d倍后,x就是结果。

三.代码

#include <iostream>
#include <cstring>
#include <algorithm>

using namespace std;

typedef long long LL;
//扩展欧几里得
LL exgcd(LL a, LL b, LL &x, LL &y)
{
    if(b == 0)
    {
        x = 1;
        y = 0;
        return a;
    }
    LL d = exgcd(b, a % b, y, x);
    y -= a / b * x;
    return d;
}
int main()
{
    LL x, y, m, n, L;
    cin >> x >> y >> m >> n >> L;
    LL a, b;
    LL d = exgcd(m - n, L, a, b);
    if((y - x) % d) puts("Impossible");//ax + by = d 只能求出来d和d的倍数。
    else
    {
        a *= (y - x) / d;//将答案扩大
        LL t = abs(L / d);//因为要求正整数解,此时L/d不一定是正数 所以取绝对值。
        cout << (a % t + t) % t << endl;//求最小正整数解
    }
    return 0;
}

四.总结

  • a x + b y = d ax + by = d ax+by=d只能求出来最大公约数和最大公约数的倍数。
  • 要求正整数解的情况下,记得取abs,因为不一定为正数。
  • exgcd求出来的是d,因此一般要扩大答案。
青蛙约会”通常是一个经典的算法题或者是程序设计题目,在C语言中可以用于练习循环、条件判断以及简单的数学运算能力。 ### 题目背景: 两只青蛙在网上相识了,它们约定在线下见面。但是由于这两只青蛙分别位于两个不同的池塘边,并且只能按照一定的步长跳跃前进,因此需要计算出它们何时能够相遇。 假设第一只青蛙A从坐标X开始,每次跳跃M的距离;第二只青蛙B从坐标Y开始,每次跳跃N的距离。两条跑道长度都为L(即两蛙在一个环形轨道上运动)。求解两只青蛙经过多少次跳跃之后会首次相遇? --- #### 算法思路: 这是一个基于同余定理的问题。我们可以将其转化为寻找最小正整数t使得下面等式成立: ``` (X + t * M) % L == (Y + t * N) % L ``` 整理得到: ``` (t * M - t * N) % L = Y - X ``` 设 `D = M - N` 和 `R = Y - X`, 则问题变成找到满足 `(t * D) % L = R` 的最小非负整数`t`. 为了有效解决此方程,我们需要使用**扩展欧几里得算法**(Extended Euclidean Algorithm),通过它不仅可以验证是否有解还可以直接获得解答。 如果 gcd(D,L)|R (gcd代表最大公约数),则存在解决方案。否则无解,表示这两只青蛙永远无法相遇。 --- ##### C语言代码示例 ```c #include <stdio.h> // 扩展欧几里德算法 int ex_gcd(int a, int b, int *x, int *y){ if(b == 0){ *x = 1; *y = 0; return a; } else{ int r = ex_gcd(b, a%b, x, y); int temp = *x; *x = *y; *y = temp - a/b*(*y); return r; } } void frogMeeting(int x,int y,int m,int n,int l){ int d=m-n,r=y-x,g,x0,y0,k,t; g=ex_gcd(d,l,&x0,&y0); // 调用扩展欧几里德 if(r%g!=0){ // 如果r不是g的倍数,则无解 printf("They cannot meet.\n"); }else{ // 否则有解 k=r/g*x0; // 计算特解k*r/g=x0*(d/l)*l+r t=(long long)(l/g)*(k%(l/g)); // 最小正整数解 printf("Frogs will first meet after jumping for %lld times.",(unsigned long long)t); } } int main(){ int x=1,y=2,m=3,n=4,l=5; frogMeeting(x,y,m,n,l); return 0; } ``` 上述代码实现了利用扩展欧几里德算法求解两只青蛙在给定条件下多久能第一次相遇的功能。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值