计蒜客 - 无脑博士的试管们
时间限制 1000ms 空间限制 65536K
题目描述
无脑博士有三个容量分别是 A,B,C 升的试管,A,B,C分别是三个从 1 到 20 的整数,最初,A 和 B 试管都是空的,而 C试管是装满硫酸铜溶液的。有时,无脑博士把硫酸铜溶液从一个试管倒到另一个试管中,直到被灌试管装满或原试管空了。当然每一次灌注都是完全的。由于无脑博士天天这么折腾,早已熟练,溶液在倒的过程中不会有丢失。
写一个程序去帮助无脑博士找出当 A试管是空的时候,C 试管中硫酸铜溶液所剩量的所有可能性。
输入格式
输入包括一行,为空格分隔开的三个数,分别为整数 A,B,C。
输出格式
输出包括一行,升序地列出当 A试管是空的时候,C 试管溶液所剩量的所有可能性。
样例输入
2 5 10
样例输出
5 6 7 8 9 10
【思路】
数据很小,考虑穷举的方法。容器里的水可以按要求倒过来倒过去,那么假设当前的三个容量为A,B,C的容器所装的水的体积为a,b,c那么无非有以下几种可能的倒水情况,把A中的水倒入B中或者把B中的水倒入A中,把A中的水倒入C中或者把C中的水倒入A中,把B中的水倒入C中或者把C中的水倒入B中,用dp数组记录每种状态,当状态重复时结束递归即可。当前状态为a=0时,c便是对应的一组解。
#include<cstdio>
#include<vector>
#include<cstring>
#include<algorithm>
using namespace std;
int A, B, C;
int dp[30][30][30];
vector<int> ans;
void solve(int a, int b, int c) {
if (dp[a][b][c]) return;
dp[a][b][c] = 1;
if (0 == a) ans.push_back(c);
if (a <= B - b) solve(0, a + b, c);
else solve(a + b - B, B, c);
if (b <= A - a) solve(a + b, 0, c);
else solve(A, a + b - A, c);
if (a <= C - c) solve(0, b, a + c);
else solve(a + c - C, b, C);
if (c <= A - a) solve(a + c, b, 0);
else solve(A, b, a + c - A);
if (b <= C - c) solve(a, 0, b + c);
else solve(a, b + c - C, C);
if (c <= B - b) solve(a, b + c, 0);
else solve(a, B, b + c - B);
}
int main() {
while (scanf("%d%d%d", &A, &B, &C) == 3) {
memset(dp, 0, sizeof(dp));
ans.clear();
solve(0, 0, C);
sort(ans.begin(), ans.end());
int len = ans.size();
for (int i = 0; i < len; i++)
printf("%d%c", ans[i], i + 1 == len ? '\n' : ' ');
}
return 0;
}