leecode/daily/2019-09-15.md

163 lines
5.7 KiB
Markdown
Raw Permalink Normal View History

2020-05-22 18:17:19 +08:00
# 毎日一题 - 水壶问题
## 信息卡片
* 时间2019-09-15
* 题目链接https://leetcode-cn.com/problems/water-and-jug-problem/
* tag`Math`
## 题目描述
```
给你一个装满水的 8 升满壶和两个分别是 5 升、3 升的空壶,请想个优雅的办法,使得其中一个水壶恰好装 4 升水,每一步的操作只能是倒空或倒满。
```
## 参考答案
1.数学分析解答
上面的问题是一个特例,我们可以抽象为[leetcode-365-水壶问题](https://leetcode-cn.com/problems/water-and-jug-problem/)。
```
有两个容量分别为 x升 和 y升 的水壶壶1壶2以及无限多的水。请判断能否通过使用这两个水壶从而可以得到恰好 z升 的水?
```
解题核心思路x < y即壶1容量小于壶2x == y的情况后面讨论
1. 将壶2倒满往壶1倒入至满。
2. 若壶1满记录当前壶2中新水量。壶1倒出将壶2中剩余的继续往壶1中倒入当壶1满继续此操作并记录当前壶2中新水量nw 若此新水量已被记录,则)。
3. 若出现壶1不满时即此时壶2必空重复操作1。
开辟一个新数组nws记录所有新水量对任意nws[i]可构造的水量为nws[i]nws[i]+xnws[i]+y。
其实不需要新数组因为数学上可以证明新水量的值循环周期呈现故可以使用一个临时变量cur当cur==x为终止条件
数学证明新水量nw值是循环周期的
![1111](<https://raw.githubusercontent.com/lvguofeng1303/markdownimage/master/daily/%E6%B0%B4%E5%A3%B6%E9%97%AE%E9%A2%98/math%20prove.jpg>)
个别特殊情况考虑:
- x == z || y == z; **true**
- x == 0 || x+y < z; **false**
- x+y == z || z == 0; **true**
```c++
class Solution {
public:
bool canMeasureWater(int x, int y, int z) {
if(x > y) return canMeasureWater(y, x, z);
if(x == z || y == z) return true;
if(x == 0 || x+y < z) return false;
if(x+y == z || z == 0) return true;
int cur = y - x;
while(cur != x){
if(cur == z) return true;
if(cur > x){
if(cur + x == z) return true;
cur = cur - x;
}
else{
if(cur + y == z || cur + x == z) return true;
cur = y - x + cur;
}
}
return false;
}
};
```
2.BFS
不仅可以计算是否能获取 z 升水,而且可以获得最少多少操作可获取 z 升水。(缺点,无法通过,因为需要太大的空间,需要申请一个三维数组记录状态)
核心思想就是状态转移问题:
壶0x+y壶1x壶2y壶0是本是无限大水池同理于定义为大小为x+y的壶。用bfs的思想使用一个队列记录所有新的状态。
对于任意状态cab状态转移就是
- 若c不为0将壶0倒水入壶1或壶2若a不为0将壶1倒水入壶0或壶2若b不为0将壶2倒水入壶0或壶1
- 记录每个新状态,并入队,若此状态访问过则不入队。
特殊情况考虑同1。
```c++
class Solution {
public:
struct state{
int nums[3];
state(int xy, int x, int y){
nums[0] = xy;
nums[1] = x;
nums[2] = y;
}
};
state pour_water(state cur, int src, int det, int size[]){
state ans = cur;
int need_w = size[det] - cur.nums[det];
if(need_w <= cur.nums[src]){
ans.nums[det] += need_w;
ans.nums[src] -= need_w;
}
else{
ans.nums[det] += ans.nums[src];
ans.nums[src] = 0;
}
return ans;
}
bool canMeasureWater(int x, int y, int z) {
if(x > y) return canMeasureWater(y, x, z); //
if(x == z || y == z) return true;
if(x == 0 || x+y < z) return false;
if(x+y == z || z == 0) return true;
int visited[x+y+1][x+1][y+1];
int water_size[3] = {x+y, x, y};
memset(visited, 0, sizeof(visited));
state cur(x+y, 0, 0);
queue<state> q;
q.push(cur);
int step = 0;
while(!q.empty()){
int size = q.size();
while(size){
state temp(0, 0, 0);
cur = q.front();
if(cur.nums[1] + cur.nums[2] == z) return true;
visited[cur.nums[0]][cur.nums[1]][cur.nums[2]] = 1;
q.pop();
if(cur.nums[0] != 0){
temp = pour_water(cur, 0, 1, water_size);
if(visited[temp.nums[0]][temp.nums[1]][temp.nums[2]] != 1)
q.push(temp);
temp = pour_water(cur, 0, 2, water_size);
if(visited[temp.nums[0]][temp.nums[1]][temp.nums[2]] != 1)
q.push(temp);
}
if(cur.nums[1] != 0){
temp = pour_water(cur, 1, 2, water_size);
if(visited[temp.nums[0]][temp.nums[1]][temp.nums[2]] != 1)
q.push(temp);
temp = pour_water(cur, 1, 0, water_size);
if(visited[temp.nums[0]][temp.nums[1]][temp.nums[2]] != 1)
q.push(temp);
}
if(cur.nums[2] != 0){
temp = pour_water(cur, 2, 1, water_size);
if(visited[temp.nums[0]][temp.nums[1]][temp.nums[2]] != 1)
q.push(temp);
temp = pour_water(cur, 2, 0, water_size);
if(visited[temp.nums[0]][temp.nums[1]][temp.nums[2]] != 1)
q.push(temp);
}
size--;
}
step++;
}
return false;
}
};
```
## 优秀解答
>暂缺