水壶问题详解

想了解更多数据结构以及算法题,可以关注微信公众号“数据结构和算法”,每天一题为你精彩解答。也可以扫描下面的二维码关注
在这里插入图片描述
有两个容量分别为 x升 和 y升 的水壶以及无限多的水。请判断能否通过使用这两个水壶,从而可以得到恰好 z升 的水?

如果可以,最后请用以上水壶中的一或两个来盛放取得的 z升 水。

你允许:

  • 装满任意一个水壶
  • 清空任意一个水壶
  • 从一个水壶向另外一个水壶倒水,直到装满或者倒空

示例 1:

输入: x = 3, y = 5, z = 4
输出: True

示例 2:

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

答案:

public boolean canMeasureWater1(int x, int y, int z) {
    return z == 0 || (long) x + y >= z && z % gcd(x, y) == 0;
}

public int gcd(int x, int y) {//求x,y的最大公约数
    return y == 0 ? x : gcd(y, x % y);
}

解析:

这题估计大家都遇到过好多次了,即使没在面试中遇到过,但至少在书上也看到过。这题如果单从代码上来看基本上没什么难度,难的是对这题的理解,其实这里面涉及到一个定理叫裴蜀定理。需要理解他,这题才能看明白。下面再来看一种解法,

 public boolean canMeasureWater(int x, int y, int z) {
     if (z < 0 || z > x + y) {
         return false;
     }
     Set<Integer> set = new HashSet<>();
     Queue<Integer> q = new LinkedList<>();
     q.offer(0);
     while (!q.isEmpty()) {
         int n = q.poll();

        int top = n - y;
        if (top >= 0 && set.add(top)) {
            q.offer(top);
        }
        int down = n + y;
        if (down <= x + y && set.add(down)) {
            q.offer(down);
        }
        int left = n - x;
        if (left >= 0 && set.add(left)) {
            q.offer(left);
        }
        int right = n + x;
        if (right <= x + y && set.add(right)) {
            q.offer(right);
        }
        if (set.contains(z)) {
            return true;
        }
    }
    return false;
}

以原点0为中心,向他的上下左右4个方向发散,所以最终会满足一个方程ax+by=z,并且a,b都是整数,如果x,y,z都不为0的情况下,当且仅当x+y=z的时候,a=b=1,否则如果满足条件,a和b肯定是一个为正数一个为负数,也就是一个总共装了几桶水,一个总共倒了几桶水。

  • 2
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

数据结构和算法

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值