avatar

目录
LeetCode 365:水壶问题

0 智力题

如何使用5L和6L水桶,得到3L水?
解答:
先用5升的容器灌满水,倒入6升容器里,
再用5升容器灌满水,用这个水把6升容器灌满,
5升容器里只有4升水,把6升容器里的水倒掉,
把5升容器里的水倒入6升容器,这样6升容器里有4升水。
把5升容器灌满水,用这个水再把6升容器灌满,5升容器里就只有3升水了。


以前看技术岗的面试智力题看到了这道倒水的题目,当时想半天没想出来,
后面发现Leetcode里竟然有专门针对这类题的程序:

1 LeetCode 365:水壶问题 [中等] [DFS]

有两个容量分别为 x升 和 y升 的水壶以及无限多的水。请判断能否通过使用这两个水壶,从而可以得到恰好 z升 的水?
如果可以,最后请用以上水壶中的一或两个来盛放取得的 z升 水。

你允许:
装满任意一个水壶
清空任意一个水壶
从一个水壶向另外一个水壶倒水,直到装满或者倒空
示例 1:
输入: x = 3, y = 5, z = 4
输出: True

示例 2:
输入: x = 2, y = 6, z = 5
输出: False

解法1:深度优先搜索

明确倒水的操作有以下几种情况:

Code
1
2
3
4
5
6
把 X 壶的水灌进 Y 壶,直至灌满或倒空;
把 Y 壶的水灌进 X 壶,直至灌满或倒空;
把 X 壶灌满;
把 Y 壶灌满;
把 X 壶倒空;
把 Y 壶倒空

使用remain_x,remain_y记录当前X,Y壶中水量状态,每次都进行深度优先搜索操作。使用栈的数据结构,每次从栈中弹出状态进行深度优先遍历,把状态压入栈,直到栈空或者达到要求。

C++ 新的关联容器unordered_set内部实现是基于哈希表(hashtable)
深搜可能会导致陷入无止境的递归,因此我们还需要使用一个哈希结合(HashSet)存储所有已经搜索过的 remain_x, remain_y 状态,保证每个状态至多只被搜索一次

Code
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
using XY = pair<int,int>;

class Solution {
public:
bool canMeasureWater(int x, int y, int z) {
stack<XY> s;
s.emplace(0,0);
auto hash_function = [](const XY & o){return hash<int>()(o.first) ^hash<int>()(o.second);};
unordered_set<XY,decltype(hash_function)> seen(0, hash_function);
while(!s.empty()){
if(seen.count(s.top())){
s.pop();
continue;
}
seen.emplace(s.top());

auto[remain_x,remain_y] = s.top();

s.pop();

if(remain_x == z || remain_y == z || remain_x+remain_y == z){
return true;
}

s.emplace(x,remain_y); //把X壶灌满

s.emplace(remain_x,y); //把y壶灌满

s.emplace(0,remain_y); //把x壶倒空

s.emplace(remain_x,0); //把x壶倒空

//把X壶的水灌进Y壶,直至灌满或者倒空
s.emplace(remain_x - min(remain_x, y-remain_y), remain_y + min(remain_x, y-remain_y));

//把Y壶的水灌进X壶,直至灌满或者倒空
s.emplace(remain_x + min(remain_y, x-remain_x), remain_y - min(remain_y, x-remain_x));
}
return false;
}
};

解法2:数学思想,贝祖定理

每次操作只会让桶里的水总量
增加 x,
增加 y,
减少 x,
减少 y。

贝祖定理

对任何整数a、b和它们的最大公约数d,关于未知数x和y的线性不定方程(称为裴蜀等式):
若a,b是整数,且gcd(a,b)=d
那么对于任意的整数x,y,ax+by都一定是d的倍数,
特别地,一定存在整数x,y,使ax+by=d成立

总结起来我们可以认为每次操作只会给水量带来x或者y的变化。
我们的目标变为:找到一对整数a,b,使得
ax + by = z
而只要满足z<=x+y,且这样的a,b存在,则可以达到目标。

贝祖定理中,x+by=z有解当且仅当 zx,y的最大公约数的倍数。
因此我们只需要找到 x,y 的最大公约数并判断 z 是否是它的倍数即可。

Code
1
2
3
4
5
6
7
8
class Solution {
public:
bool canMeasureWater(int x, int y, int z) {
if (x + y < z) return false;
if (x == 0 || y == 0) return z == 0 || x + y == z;
return z % gcd(x, y) == 0;
}
};

数学方法真的简单很多,但是如果不看题解都不知道有这个定理(手动狗头

文章作者: Bellium
文章链接: https://belliumtang.github.io/2020/04/05/Leecode365/
版权声明: 本博客所有文章除特别声明外,均采用 CC BY-NC-SA 4.0 许可协议。转载请注明来自 Bellium
打赏
  • 微信Wechatpay
    微信Wechatpay
  • 支付宝Alipay
    支付宝Alipay