【题解】导游-C++

Description
宁波市的中小学生们在镇海中学参加程序设计比赛之余,热情的主办方邀请同学们参观镇海中学内的各处景点,已
知镇海中学内共有n处景点。现在有n位该校的学生志愿承担导游和讲解任务。每个学生志愿者对各个景点的熟悉程
度是不同的,如何将n位导游分配至n处景点,使得总的熟悉程度最大呢?要求每个景点处都有一个学生导游。
Input
有若干行:
第一行只有一个正整数n(1≤n≤17),表示有n个景点和n个学生导游。
第二行至第n+1行共n行,每行有n个以空格分隔的正整数。
第i+1行的第j个数k(1≤k≤1000),表示第i个学生导游对景点j的熟悉程度为k。
Output
只有一行,该行只有一个正整数,表示求得的熟悉程度之和的最大值。
Sample Input
3
10 6 8
9 2 3
1 7 2
Sample Output
24
//第1个学生负责第3个景点,第2个学生负责第1个景点,第3个学生负责第2个景点时,熟悉程度总和为24,达到最大值

这道题目是一道极大化剪枝。
很明显,这道题目用我们的爆搜大法好 是AC不了的。
为什么呢?
每个景点的选择情况有n种,共n个景点,爆搜的时间复杂度到了O(n^n),而n最大为17,这尼玛绝对TLE.
我们开挂 把可能得到的最大熟悉度算出来(因为开了挂所以一个人可以负责很多景点)
然后在搜索过程中!运用这个开挂值awa,如果当前状态,加上后面的开挂值如果比总开挂值还要大(我也不知道为什么会有这种情况)那这种情况就要剪枝。
虽然不知道为什么但是这原理非常简单,如果当前值以及以后的极大值的和比全局极大值还要大,这种情况就应该剪掉
其他的和爆搜没什么区别,爆搜的实现也还算简单,下面就直接贴代码,摸鱼酱的代码没有注释觉得好请点赞.

#include<bits/stdc++.h>
using namespace std;
int n,d[20][20],dis[20];
bool flag[20];
int ans=-1;
void dfs(int dep,int sum)
{
	if(dep==n+1)
	{
		ans=max(ans,sum);
		return;
	}
	if(ans>=sum+dis[n]-dis[dep-1])return;
	for(int i=1;i<=n;i++)
	{
		if(!flag[i])
		{
			flag[i]=1;
			dfs(dep+1,sum+d[dep][i]);
			flag[i]=0;
		}
	}
	return;
}
int main()
{
	cin>>n;
	for(int i=1;i<=n;i++)
	{
		int m=-1;
		for(int j=1;j<=n;j++)
		{
			cin>>d[i][j];
			m=max(m,d[i][j]);
		}
		dis[i]=dis[i-1]+m;
	}
	dfs(1,0);
	cout<<ans<<endl;
	return 0; 
}

ov.

转载于:https://www.cnblogs.com/moyujiang/p/11167724.html

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Main()函数模:主要打印开始界面和提示相关的操作。用穷举的方法把学校相关景点之间的路径和权值(距离)初始化,用cost[][]二维数组实现。打印:输入”s”进行查询最短路径,输入“b”进入主菜单页面,输入“e”退出程序。 switch(k) { case'b':star();break; case 's': printf("the shortest way searching:"); shortestdistance(); break; case 'e': exit(0); default:printf("Error!\nplease press b or s or e.\n");getchar();break; } Star()函数:主要是显示一个界面,功能是提供查询路径选择和查看进入景点的主页面和退出该页面返回主界面。 switch(choose) {case '0':instruction();getchar();break; case '1':jingdian();break; case '2': map();break; case '9':end();break; default:star(); } Jingdian()函数:显示校园的10个景点和代号,提示输入相应的代号进入该景点的函数模块。在这里也可以查询10个景点中任意两个之间的最短路径,也可以选择退出程序。 switch(point) { case '1': office();break; case '2': library();break; case '3': lake();break; case '4': classroom();break; case '5': apartment();break; case 's':shortestdistance();break; case '0': star();getchar();break; case 'e':exit(0); default:jingdian(); } Floyed()函数:求两个景点之间的最短距离。 for(i=1;i<=n;i++) for(j=1;j<=n;j++) { shortest[j]=cost[j]; path[j]=0; } for(k=1;k<=n;k++) for(i=1;i<=n;i++) for(j=1;j<=n;j++) if(shortest[j]>(shortest[k]+shortest[k][j])) {/*用path[][]记录从i到j的最短路径上点j的前驱景点的序号*/ shortest[j]=shortest[k]+shortest[k][j]; path[j]=k; path[j]=k; display(int i,int j)函数:打印两个景点之间的最短路径。 if(shortest[j]!=INT_MAX) { if(i<j) { printf("%d",b); while(path[j]!=0) {/*把i到j的路径上所有经过的景点按逆序打印出来*/ printf("<-%d",path[j]); if(i<j) j=path[j]; else i=path[j]; } printf("<-%d",a); printf("\n\n"); printf("(%d->%d)the distance is:%dm\n\n",a,b,shortest[a]); } else { printf("%d",a); while(path[j]!=0) {/*把i到j的路径上所有经过的景点按顺序打印出来*/ printf("->%d",path[j]); if(i<j) j=path[j]; else i=path[j]; } printf("->%d",b); printf("\n\n"); printf("(%d->%d)the shortest distance is:]m\n\n",a,b,shortest[a]); } 其余几个函数是学校景点的相应介绍。调用后都可以回到jingdian()函数的主界面。 End()函数是一个结束函数,打印结束语,执行退出命令。 void end() { char r; clrscr(); printf("THANK YOU AND YOUR WELCOME AGAIN\n"); printf("PLEASE INPUT ANY KEY EXIT !\n"); getchar(); scanf("%c",&r); exit(0); }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值