神奇的小石头

1 篇文章 0 订阅

题目描述

斌斌翻水水的家乡有很多个水井,每个水井 i 和其他水井 j 都有一个管子连通,这样一个水井没水的时候其他的水井可以通过水管直接输送水过来,而这个管子的容量是vij,
有一天斌斌翻水水在野外烧烤的时候被一个奇异的小石头砸中,当时斌斌翻水水就觉的这个石头很特别就带回了家,在多年的研究之下他发现这个石头有个神奇的功能:
它能够将多个水井划分成两个区域A和B,在同一个区域的水井之间不再需要水管连接,神奇的小石头可以直接让他们的水流通,但是在不同区域的水井却不行,仍然需要水管相连接。
本来正常人直接把所有水井都放到A或者B,这样所有水井都互通了,但是斌斌翻水水不一样,他觉的这些水管放着不用可惜,他想让水井之间用到的水管的容量总数最大(每一个水井不是在A区域就是在B区域
不能独立在外),但是斌斌翻水水又很懒不想自己去想怎么进行划分,所以他把这个难题抛给了聪明的你,希望你能帮住到他。

Input

输入一共N+1行,第一行为一个数字N(2 <= N <= 20),代表一共有几个水井,接下去有N行数字,每行数字有N个数字,第i行数字的第j列数字代表Vij.(注意 vij = vji , vii = 0, 0 <= vij <= 10000)(例如样例中的 0 50 30 的 50代表第一个水井和第二个水井之间的水管容量为50)

Output

一共一行,那一行包含一个数字,代表斌斌翻水水让你求的利用水管的最大容量总和。

Sample Input

3
0 50 30
50 0 40
30 40 0

Sample Output

90

提示

90的答案是这样来的,将第一个水井和第三个水井放在A,将第二个水井放到B,那么第一个水井和第三个水井和第二个水井之间的水管要连接那么便是50+40 = 90。
虽然神奇的石头拥有着超自然的力量,但是其他物质和现实生活中一样,所以你无需考虑水管容量是负数的情况,只需要考虑水管是非负数的情况就行了。

解题思路

题目分析

这题题目比较难读懂,这里说一下,题目意思就相当于有两个区域,让你把n个水桶放在这两个区域中,可以用一根水管把不同区域的水桶连接,要使得两个区域间的管子体积达到最大。

解题思路

这里不难想到,可以用dfs的思维写这一题,把所有的水井放在A区域,dfs每一次让一个水井移动到B区域去。不过我一开始的dfs超时了,只能过75的样例,经过高人指点,发现可以换种dfs来搜索,这也是我从来没用过的方法,学到了!

超时解法

这里引以为戒,不做过多注释

void dfs(int ith, int total)
{
	ans = max(total, ans);
	if (ith > (n >> 1))
		return;
	for (int i = 1; i <= n; i++)
	{
		int num = total;
		if (vis[i] == 0)
		{
			vis[i] = 1;
			for (int j = 1; j <= n; j++)
			{
				if (vis[j] == 0)
					num += lake[i][j];
				else
					num -= lake[i][j];
			}
			dfs(ith + 1, num);
			vis[i] = 0;
		}
	}
	return;
}

AC代码

#include <algorithm>
#include <queue>
#include <iostream>
#include <cstdio>
#include <stack>
#include <string>

using namespace std;
const int N = 25;

int lake[N][N];
int vis[N];
int n, ans;

//这里与上面超时代码不同的是,直接用ith来充当遍历过程,减少了一重循环
void dfs(int ith, int total)
{
	ans = max(total, ans);
	if (ith == n + 1)
		return;
	vis[ith] = 1;//这里直接判断当前水井放在另一区域的情况
	int num = total;//用num来进行后面的运算而不直接使用total,是因为避免回溯的时候还要再用一层循环来减去加上的体积,较为方便
	for (int i = 1; i <= n; i++)
	{
		if (vis[i] == 0)
			num += lake[ith][i];//现在ith水井到了另一区域,和他不同区域的水井连接水管,加上体积
		else
			num -= lake[ith][i];//同一个区域的水井,因为之前区域不同,所以要减去这部分的体积
	}
	dfs(ith + 1, num);//递归遍历下一层
	vis[ith] = 0;//回溯清零
	dfs(ith + 1, total);//这里就相当于是不移动ith号水井了
	return;
}

int main()
{
	cin >> n;
	for (int i = 1; i <= n; i++)
	{
		for (int j = 1; j <= n; j++)
			scanf("%d", &lake[i][j]);
	}
	dfs(1, 0);
	cout << ans;
	return 0;
}
  • 6
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 6
    评论
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值