Встреча у фонтана
Ограничение по времени: 0.5 секунды
Маша и Паша живут на одной улице, и их дома разделены только парком, в котором друзья любят гулять. В центре парка есть красивый фонтан, у которого Маша и Паша хотят сегодня встретиться. Известно, что Маша идёт до фонтана m минут, Паша — p минут.
Выйти из домов они договорились одновременно, также друзья решили приходить к фонтану и, если там никого нет, идти обратно к дому, а затем снова разворачиваться, пока в итоге не случится встреча у фонтана.
Помогите друзьям понять, смогут ли они встретиться в парке у фонтана, и если да, то сколько минут пройдёт с момента выхода из домов до их встречи.
Формат входных данных
Первая строка содержит целое число m (1⩽m⩽109) — время в минутах, которое требуется Маше, чтобы дойти от дома до фонтана.
Вторая строка содержит целое число p (1⩽p⩽109) — время в минутах, которое требуется Паше, чтобы дойти от дома до фонтана.
Формат выходных данных
Выведите одно целое число — время, через которое Маша и Паша смогут встретиться у фонтана, если выйдут из домов одновременно, или −1, если этого никогда не случится.
Обратите внимание на то, что значение ответа в этой задаче может превышать возможное значение 32‑битной целочисленной переменной, поэтому необходимо использовать 64‑битные целочисленные типы данных (тип int64 в языке Pascal, тип long long в C++, тип long в Java и C#).
Система оценки
Решения, правильно работающие, когда m и p не превосходят 103, будут оцениваться в 30 баллов.
Решения, правильно работающие, когда ответ не превосходит 109, будут оцениваться в 60 баллов.
Замечание
Здесь и далее все временные отметки даются относительно начала движения, т. е. выхода из дома.
В первом примере из условия Маша придёт к фонтану через 3 минуты, развернётся и пойдёт назад. Паша придёт к фонтану через 5 минут и отправится домой. Через 6 минут Маша доберётся до дома, вновь окажется у фонтана через 9 минут, опять не найдёт Пашу, развернётся и пойдёт домой. В следующий раз она будет у фонтана через 15 минут. Паша же дойдёт до дома через 10 минут и вернётся к фонтану через 15 минут, где он и встретится с Машей.
Во втором примере Маша успеет дойти до фонтана и вернуться домой, пока Паша идёт до фонтана.
Пока Паша возвращается домой, Маша опять проделывает путь до фонтана и обратно. Каждый раз, когда Паша оказывается у фонтана или у своего дома, Маша находится у своего дома, поэтому они не смогут встретиться.
Ответ нейросети
21 октября 2025 13:58
Кратко понятие задачи
Маше и Паше нужно одновременно оказаться у фонтана.
Маше до фонтана нужно m минут, Паше — p минут.
Они отправляются одновременно, затем идут к фонтану, если никого там нет — возвращаются домой, и так повторяют путь туда/обратно, пока не встретятся у фонтана.
Машина времени на встречу — минимальное t > 0 такое, что оба одновременно находятся у фонтана.
Ключевая идея
Маша оказывается у фонтана в моменты времени t ∈ {m, 3m, 5m, ...} то есть t ≡ m (mod 2m).
Паша оказывается у фонтана в моменты времени t ∈ {p, 3p, 5p, ...} то есть t ≡ p (mod 2p).
Нужно найти наименьшее t, удовлетворяющее системе конгруэнций:
t ≡ m (mod 2m)
t ≡ p (mod 2p)
Такого t может не быть. Необходимо проверить совместимость по китайской теореме о остатках:
существуют решения iff m ≡ p (mod gcd(2m, 2p)), то есть iff (p - m) делится на d = gcd(2m, 2p) = 2 * gcd(m, p).
Если решения нет — ответ -1. Если есть — вычисляем наименьшее t через CRT.
Пошаговое решение (алгоритм)
g0 = gcd(m, p)
d = 2 * g0
если (p - m) % d != 0, вывести -1
Пусть a = 2m, b = 2p. Разделим:
ap = a / d, bp = b / d, c = (p - m) / d
Нужно найти k такое, что ap * k ≡ c (mod bp). Так как gcd(ap, bp) = 1, существует обратная к ап по bp.
inv = модульный обратный ап по bp
k = (c mod bp) * inv mod bp
искомое время t = m + a * k
вывести t
Пояснение по корректности
В шаге 3 условие говорит: конгруэнции несовместны, значит встреча никогда не произойдёт.
В противном случае, после приведения к канонической форме, мы находим минимальный k в диапазоне [0, bp-1], что даёт минимальное t, удовлетворяющее обе конгруэнции. Так мы получаем первую встречу у фонтана.
Пример
Пример 1: m = 3, p = 5
g0 = gcd(3,5) = 1, d = 2
(p - m) = 2 делится на 2 → решение есть
a = 6, b = 10, ap = 3, bp = 5, c = 1
inv(3 mod 5) = 2 (потому что 32 ≡ 1 (mod 5))
k = (1 * 2) mod 5 = 2
t = m + ak = 3 + 6*2 = 15
Ответ: 15
Пример 2: m = 1, p = 2
g0 = gcd(1,2) = 1, d = 2
(p - m) = 1 не делится на 2 → встреча не произойдёт
Ответ: -1
Временная сложность
Основные операции — вычисление gcd и единичной modular inverse, всё это делает за O(log min(m, p)).
Используемые типы: 64-битные целые (long long). Значение может достигать порядка 2m·bp, но вписывается в 64 бит.
Пример кода (C++17, 64-битные целые)
#include <bits/stdc++.h>
using namespace std;
using int64 = long long;
int64 extgcd(int64 a, int64 b, int64 &x, int64 &y) {
if (b == 0) { x = 1; y = 0; return a; }
int64 x1, y1;
int64 g = extgcd(b, a % b, x1, y1);
x = y1;
y = x1 - (a / b) * y1;
return g;
}
int64 modinv(int64 a, int64 m) {
int64 x, y;
int64 g = extgcd(a, m, x, y);
if (g != 1) return -1; // inverse doesn't exist, but в нашем случае он всегда существует
x %= m;
if (x < 0) x += m;
return x;
}
int main() {
ios::sync_with_stdio(false);
cin.tie(nullptr);
int64 m, p;
if (!(cin >> m)) return 0;
cin >> p;
int64 g0 = std::gcd(m, p);
int64 d = 2 * g0;
int64 diff = p - m;
if (diff % d != 0) {
cout << -1 << "\n";
return 0;
}
int64 a = 2 * m;
int64 b = 2 * p;
int64 ap = a / d;
int64 bp = b / d;
int64 c = diff / d;
int64 inv = modinv((ap % bp + bp) % bp, bp);
// bp >= 1 и gcd(ap, bp) = 1, так что inv существует
int64 k = (( (c % bp + bp) % bp) * inv) % bp;
int64 t = m + a * k;
cout << t << "\n";
return 0;
}
Как применять на реальных задачах
Введите m и p, получите ответ:
если -1, значит никогда не встретятся у фонтана
иначе — минимальное время в минутах с момента выхода из домов, когда они встретятся на фонтане.
ПО распространяется в виде интернет-сервиса, специальные действия по установке ПО на стороне пользователя не требуются
Встреча у фонтана
Ограничение по времени: 0.5 секунды
Маша и Паша живут на одной улице, и их дома разделены только парком, в котором друзья любят гулять. В центре парка есть красивый фонтан, у которого Маша и Паша хотят сегодня встретиться. Известно, что Маша идёт до фонтана m минут, Паша — p минут.
Выйти из домов они договорились одновременно, также друзья решили приходить к фонтану и, если там никого нет, идти обратно к дому, а затем снова разворачиваться, пока в итоге не случится встреча у фонтана.
Помогите друзьям понять, смогут ли они встретиться в парке у фонтана, и если да, то сколько минут пройдёт с момента выхода из домов до их встречи.
Формат входных данных
Первая строка содержит целое число m (1⩽m⩽109) — время в минутах, которое требуется Маше, чтобы дойти от дома до фонтана.
Вторая строка содержит целое число p (1⩽p⩽109) — время в минутах, которое требуется Паше, чтобы дойти от дома до фонтана.
Формат выходных данных
Выведите одно целое число — время, через которое Маша и Паша смогут встретиться у фонтана, если выйдут из домов одновременно, или −1, если этого никогда не случится.
Обратите внимание на то, что значение ответа в этой задаче может превышать возможное значение 32‑битной целочисленной переменной, поэтому необходимо использовать 64‑битные целочисленные типы данных (тип int64 в языке Pascal, тип long long в C++, тип long в Java и C#).
Система оценки
Решения, правильно работающие, когда m и p не превосходят 103, будут оцениваться в 30 баллов.
Решения, правильно работающие, когда ответ не превосходит 109, будут оцениваться в 60 баллов.
Замечание
Здесь и далее все временные отметки даются относительно начала движения, т. е. выхода из дома.
В первом примере из условия Маша придёт к фонтану через 3 минуты, развернётся и пойдёт назад. Паша придёт к фонтану через 5 минут и отправится домой. Через 6 минут Маша доберётся до дома, вновь окажется у фонтана через 9 минут, опять не найдёт Пашу, развернётся и пойдёт домой. В следующий раз она будет у фонтана через 15 минут. Паша же дойдёт до дома через 10 минут и вернётся к фонтану через 15 минут, где он и встретится с Машей.
Во втором примере Маша успеет дойти до фонтана и вернуться домой, пока Паша идёт до фонтана.
Пока Паша возвращается домой, Маша опять проделывает путь до фонтана и обратно. Каждый раз, когда Паша оказывается у фонтана или у своего дома, Маша находится у своего дома, поэтому они не смогут встретиться.
Кратко понятие задачи
- Маше и Паше нужно одновременно оказаться у фонтана.
- Маше до фонтана нужно m минут, Паше — p минут.
- Они отправляются одновременно, затем идут к фонтану, если никого там нет — возвращаются домой, и так повторяют путь туда/обратно, пока не встретятся у фонтана.
- Машина времени на встречу — минимальное t > 0 такое, что оба одновременно находятся у фонтана.
Ключевая идея
- Маша оказывается у фонтана в моменты времени t ∈ {m, 3m, 5m, ...} то есть t ≡ m (mod 2m).
- Паша оказывается у фонтана в моменты времени t ∈ {p, 3p, 5p, ...} то есть t ≡ p (mod 2p).
- Нужно найти наименьшее t, удовлетворяющее системе конгруэнций:
t ≡ m (mod 2m)
t ≡ p (mod 2p)
- Такого t может не быть. Необходимо проверить совместимость по китайской теореме о остатках:
существуют решения iff m ≡ p (mod gcd(2m, 2p)), то есть iff (p - m) делится на d = gcd(2m, 2p) = 2 * gcd(m, p).
- Если решения нет — ответ -1. Если есть — вычисляем наименьшее t через CRT.
Пошаговое решение (алгоритм)
1) g0 = gcd(m, p)
2) d = 2 * g0
3) если (p - m) % d != 0, вывести -1
4) Пусть a = 2m, b = 2p. Разделим:
ap = a / d, bp = b / d, c = (p - m) / d
5) Нужно найти k такое, что ap * k ≡ c (mod bp). Так как gcd(ap, bp) = 1, существует обратная к ап по bp.
6) inv = модульный обратный ап по bp
7) k = (c mod bp) * inv mod bp
8) искомое время t = m + a * k
9) вывести t
Пояснение по корректности
- В шаге 3 условие говорит: конгруэнции несовместны, значит встреча никогда не произойдёт.
- В противном случае, после приведения к канонической форме, мы находим минимальный k в диапазоне [0, bp-1], что даёт минимальное t, удовлетворяющее обе конгруэнции. Так мы получаем первую встречу у фонтана.
Пример
- Пример 1: m = 3, p = 5
g0 = gcd(3,5) = 1, d = 2
(p - m) = 2 делится на 2 → решение есть
a = 6, b = 10, ap = 3, bp = 5, c = 1
inv(3 mod 5) = 2 (потому что 3*2 ≡ 1 (mod 5))
k = (1 * 2) mod 5 = 2
t = m + a*k = 3 + 6*2 = 15
Ответ: 15
- Пример 2: m = 1, p = 2
g0 = gcd(1,2) = 1, d = 2
(p - m) = 1 не делится на 2 → встреча не произойдёт
Ответ: -1
Временная сложность
- Основные операции — вычисление gcd и единичной modular inverse, всё это делает за O(log min(m, p)).
- Используемые типы: 64-битные целые (long long). Значение может достигать порядка 2m·bp, но вписывается в 64 бит.
Пример кода (C++17, 64-битные целые)
#include
using namespace std;
using int64 = long long;
int64 extgcd(int64 a, int64 b, int64 &x, int64 &y) {
if (b == 0) { x = 1; y = 0; return a; }
int64 x1, y1;
int64 g = extgcd(b, a % b, x1, y1);
x = y1;
y = x1 - (a / b) * y1;
return g;
}
int64 modinv(int64 a, int64 m) {
int64 x, y;
int64 g = extgcd(a, m, x, y);
if (g != 1) return -1; // inverse doesn't exist, but в нашем случае он всегда существует
x %= m;
if (x < 0) x += m;
return x;
}
int main() {
ios::sync_with_stdio(false);
cin.tie(nullptr);
int64 m, p;
if (!(cin >> m)) return 0;
cin >> p;
int64 g0 = std::gcd(m, p);
int64 d = 2 * g0;
int64 diff = p - m;
if (diff % d != 0) {
cout << -1 << "\n";
return 0;
}
int64 a = 2 * m;
int64 b = 2 * p;
int64 ap = a / d;
int64 bp = b / d;
int64 c = diff / d;
int64 inv = modinv((ap % bp + bp) % bp, bp);
// bp >= 1 и gcd(ap, bp) = 1, так что inv существует
int64 k = (( (c % bp + bp) % bp) * inv) % bp;
int64 t = m + a * k;
cout << t << "\n";
return 0;
}
Как применять на реальных задачах
- Введите m и p, получите ответ:
- если -1, значит никогда не встретятся у фонтана
- иначе — минимальное время в минутах с момента выхода из домов, когда они встретятся на фонтане.