计蒜客 - 无脑博士的试管们

时间限制 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;
}


posted @ 2017-10-26 16:28  不想吃WA的咸鱼  阅读(201)  评论(0编辑  收藏  举报