http://hi.baidu.com/5l2_/blog/item/a0381951b5738a1e367abef0.html 这个文章写的很好。。。我只是照办人家的。。。。TT 自己想不到的。。。。。
大概意思 给定 a b k找到满足ax+by=k 的令|x|+|y|最小(等时令a|x|+b|y|最小)不妨a 〉b
先用扩展欧几里得算法求出 一组解 x0,y0,通解可以表示为x=x0+b/d *t y=y0-a/d *t
|x|+|y|=|x0+b/d *t |+|y0-a/d *t| 这个关于t的函数的最小值在 t = y0*d/a 附近的两整点里取。故直接验证这两点即可。
因为 设a>b之后
|x0+b/d *t| 单调递增,|y0-a/d*t| 先递减再递增。因斜率a/d>b/d,所以总的|x0+b/d *t |+|y0-a/d *t| 先递减再递增,使y0-a/d*t0=0 的t0附近有最小值。
#include<iostream>
using namespace std;
int x, y, a1, b1;
int ex_gcd(int a, int b, int &x, int &y) {
if (b == 0) {
x = 1, y = 0;
return a;
}
int d = ex_gcd(b, a % b, x, y), t = x;
x = y;
y = t - a / b*y;
return d;
}
int calx(int t) {
return abs(x + a1 * t);
}
int caly(int t) {
return abs(y - b1 * t);
}
int main() {
int a, b, d, k;
int tx1, tx2, ty1, ty2, r1, r2, r, ansx, ansy;
bool flag = 1;
while (scanf("%d%d%d", &a, &b, &k) && (a || b || k)) {
flag = 1;
if (a < b) flag = 0, swap(a, b);
d = ex_gcd(a, b, x, y);
x = x * (k / d);
y = y * (k / d);
a1 = b / d, b1 = a / d, r1 = y / b1;
if (r1 * b1 - y >= 0) r1--;
r2 = r1 + 1;
tx1 = calx(r1),tx2 = caly(r1);
ty1 = calx(r2),ty2 = caly(r2);
if ((tx1 + tx2) < (ty1 + ty2)) r = r1;
else if ((tx1 + tx2) > (ty1 + ty2)) r = r2;
else {
if ((tx1 * a + tx2 * b) <= (ty1 * a + ty2 * b)) r = r1;
else r = r2;
}
ansx = calx(r);
ansy = caly(r);
if (!flag)
printf("%d %d\n", ansy, ansx);
else
printf("%d %d\n", ansx, ansy);
}
return 0;
}