week6 贪心算法

T1:三国游戏

题目描述:

# [NOIP2010 普及组] 三国游戏

小涵很喜欢电脑游戏,这些天他正在玩一个叫做《三国》的游戏。

在游戏中,小涵和计算机各执一方,组建各自的军队进行对战。游戏中共有 $N$ 位武将($N$为偶数且不小于$4$),任意两个武将之间有一个“默契值”,表示若此两位武将作为一对组合作战时,该组合的威力有多大。游戏开始前,所有武将都是自由的(称为自由武将,一旦某个自由武将被选中作为某方军队的一员,那么他就不再是自由武将了),换句话说,所谓的自由武将不属于任何一方。

游戏开始,小涵和计算机要从自由武将中挑选武将组成自己的军队,规则如下:小涵先从自由武将中选出一个加入自己的军队,然后计算机也从自由武将中选出一个加入计算机方的军队。接下来一直按照“小涵→计算机→小涵→……”的顺序选择武将,直到所有的武将被双方均分完。然后,程序自动从双方军队中各挑出一对默契值最高的武将组合代表自己的军队进行二对二比武,拥有更高默契值的一对武将组合获胜,表示两军交战,拥有获胜武将组合的一方获胜。

已知计算机一方选择武将的原则是尽量破坏对手下一步将形成的最强组合,它采取的具体策略如下:任何时刻,轮到计算机挑选时,它会尝试将对手军队中的每个武将与当前每个自由武将进行一一配对,找出所有配对中默契值最高的那对武将组合,并将该组合中的自由武将选入自己的军队。 下面举例说明计算机的选将策略,例如,游戏中一共有$6$个武将,他们相互之间的默契值如下表所示:

 ![](https://cdn.luogu.com.cn/upload/pic/54.png) 

双方选将过程如下所示:

 ![](https://cdn.luogu.com.cn/upload/pic/55.png) 

小涵想知道,如果计算机在一局游戏中始终坚持上面这个策略,那么自己有没有可能必胜?如果有,在所有可能的胜利结局中,自己那对用于比武的武将组合的默契值最大是多少?  

假设整个游戏过程中,对战双方任何时候均能看到自由武将队中的武将和对方军队的武将。为了简化问题,保证对于不同的武将组合,其默契值均不相同。

## 输入格式

共 N 行。

第一行为一个偶数 $N$,表示武将的个数。

第 $2 $行到第 $N $行里,第$i+1$行有$N_i$个非负整数,每两个数之间用一个空格隔开,表示$ i $号武将和$ i+1,i+2,…,N $号武将之间的默契值($0≤$默契值$≤1,000,000,000$)。

## 输出格式

共 $1$ 或 $2 $行。

若对于给定的游戏输入,存在可以让小涵获胜的选将顺序,则输出$ 1$,并另起一行输出所有获胜的情况中,小涵最终选出的武将组合的最大默契值。如果不存在可以让小涵获胜的选将顺序,则输出 $0$。

## 样例 #1

### 样例输入 #1

```

5 28 16 29 27 
23 3 20 1 
8 32 26 
33 11 
12
```

### 样例输出 #1

```
1
32
```

## 样例 #2

### 样例输入 #2

```

42 24 10 29 27 12 58 
31 8 16 26 80 6 
25 3 36 11 5 
33 20 17 13 
15 77 9 
4 50 
19
```

### 样例输出 #2

```
1
77
```

## 提示

【数据范围】

对于$ 40\%$的数据有 $N≤10$。

对于$ 70\%$的数据有$ N≤18$。

对于 $100\%$的数据有 $N≤500$。

思路:由给出的示例,可以看出题目的贪心原则与选取第二大武将默契值有关(除了选第一个),假设当前选择的武将为A,与他匹配的最大默契值的武将为B,第二大武将默契值为武将C。此时计算机会选择B武将,由于最大的武将默契值的两个武将被拆开了,所以小涵选择第二大武将C的默契值必然是最优解,即小涵必胜!!

简化后的问题就是:求一堆数里面的第二大值

#include<bits/stdc++.h>
using namespace std;
int n;
int hero[1005][1005];
int ans=-1;
int main()
{
	cin>>n;
	for(int i=1;i<n;i++)
	{
		for(int j=i+1;j<=n;j++)
		{
			cin>>hero[i][j];
			hero[j][i]=hero[i][j];//把数据做成一个n*n的矩阵
		}
	}
	for(int i=1;i<=n;i++)
	{
		sort(hero[i]+1,hero[i]+1+n);//对于每一行的数据进行排序,寻找到最大值
		ans=max(ans,hero[i][n-1]);
	} 
	cout<<"1"<<endl<<ans;
	return 0;
}

T2:独木桥

题目描述:

# 独木桥

## 题目背景

战争已经进入到紧要时间。你是运输小队长,正在率领运输部队向前线运送物资。运输任务像做题一样的无聊。你希望找些刺激,于是命令你的士兵们到前方的一座独木桥上欣赏风景,而你留在桥下欣赏士兵们。士兵们十分愤怒,因为这座独木桥十分狭窄,只能容纳 $1$ 个人通过。假如有 $2$ 个人相向而行在桥上相遇,那么他们 $2$ 个人将无法绕过对方,只能有 $1$ 个人回头下桥,让另一个人先通过。但是,可以有多个人同时呆在同一个位置。

## 题目描述

突然,你收到从指挥部发来的信息,敌军的轰炸机正朝着你所在的独木桥飞来!为了安全,你的部队必须撤下独木桥。独木桥的长度为 $L$,士兵们只能呆在坐标为整数的地方。所有士兵的速度都为 $1$,但一个士兵某一时刻来到了坐标为 $0$ 或 $L+1$ 的位置,他就离开了独木桥。

每个士兵都有一个初始面对的方向,他们会以匀速朝着这个方向行走,中途不会自己改变方向。但是,如果两个士兵面对面相遇,他们无法彼此通过对方,于是就分别转身,继续行走。转身不需要任何的时间。

由于先前的愤怒,你已不能控制你的士兵。甚至,你连每个士兵初始面对的方向都不知道。因此,你想要知道你的部队最少需要多少时间就可能全部撤离独木桥。另外,总部也在安排阻拦敌人的进攻,因此你还需要知道你的部队最多需要多少时间才能全部撤离独木桥。

## 输入格式

第一行共一个整数 $L$,表示独木桥的长度。桥上的坐标为 $1, 2, \cdots, L$。

第二行共一个整数 $N$,表示初始时留在桥上的士兵数目。

第三行共有 $N$ 个整数,分别表示每个士兵的初始坐标。

## 输出格式

共一行,输出 $2$ 个整数,分别表示部队撤离独木桥的最小时间和最大时间。$2$ 个整数由一个空格符分开。

## 样例 #1

### 样例输入 #1

```
4
2
1 3
```

### 样例输出 #1

```
2 4
```

## 提示

对于 $100\%$ 的数据,满足初始时,没有两个士兵同在一个坐标,$1\le  L\le5\times 10^3$,$0\le N\le5\times10^3$,且数据保证 $N\le L$。

思路:对于人P和人Q在桥中相遇然后转头问题,可以看成P,Q继续走,P,Q互相交换了身份,即这个碰头问题可以忽略。

首先是个人撤离最小时间,假设人到桥左端的距离为A,到右端的距离为B,这个最小时间必然是min(A,B),即A,B中较小值,然后其次是个人撤离最大时间为max(A,B),要想整个部队都从桥头撤离,必然要等最后一个人都离开桥,即部队离开最小的时间是撤离的士兵中最小时间的最大值,部队撤离最大时间同理。

#include<bits/stdc++.h>
using namespace std;
int L,N;
int main()
{
	cin>>L>>N;
	int ans1=0,ans2=0; 
	for(int i=1;i<=N;i++){
		int x;
		cin>>x;
		ans1=max(min(x,L-x+1),ans1);
		ans2=max(max(x,L-x+1),ans2);
	}
	cout<<ans1<<' '<<ans2;
	return 0;
 } 

T3:排队接水

题目描述:

# 排队接水

## 题目描述

有 $n$ 个人在一个水龙头前排队接水,假如每个人接水的时间为 $T_i$,请编程找出这 $n$ 个人排队的一种顺序,使得 $n$ 个人的平均等待时间最小。

## 输入格式

第一行为一个整数 $n$。

第二行 $n$ 个整数,第 $i$ 个整数 $T_i$ 表示第 $i$ 个人的等待时间 $T_i$。

## 输出格式

输出文件有两行,第一行为一种平均时间最短的排队顺序;第二行为这种排列方案下的平均等待时间(输出结果精确到小数点后两位)。

## 样例 #1

### 样例输入 #1

```
10 
56 12 1 99 1000 234 33 55 99 812
```

### 样例输出 #1

```
3 2 7 8 1 4 9 6 10 5
291.90
```

## 提示

$n \leq 1000,t_i \leq 10^6$,不保证 $t_i$ 不重复。

当 $t_i$ 重复时,按照输入顺序即可(sort 是可以的)

思路:对于整个队伍排队总时间T=t1*(n-1)+t2*(n-2)+...+t(n-1);若想要这个T最小,必然第一个接水的人时间必须最小,对于本题解决排序后,序号丢失问题,可以用一个结构体排序即可

#include<bits/stdc++.h>
using namespace std;
struct peo{
	int id,t;
};
int n;
bool cmp(peo a,peo b){
	return a.t<b.t;
}
int main()
{
	peo p[1005];
	cin>>n;
	for(int i=1;i<=n;i++)cin>>p[i].t,p[i].id=i;
	sort(p+1,p+1+n,cmp);
	double ans=0;
	for(int i=1;i<=n;i++){
		cout<<p[i].id<<' ';
		ans+=p[i].t*(n-i);
	}
	cout<<endl<<fixed<<setprecision(2)<<ans/n;
	return 0;
 } 

T4:合并果子

题目描述:

# [NOIP2004 提高组] 合并果子 / [USACO06NOV] Fence Repair G

## 题目描述

在一个果园里,多多已经将所有的果子打了下来,而且按果子的不同种类分成了不同的堆。多多决定把所有的果子合成一堆。

每一次合并,多多可以把两堆果子合并到一起,消耗的体力等于两堆果子的重量之和。可以看出,所有的果子经过 $n-1$ 次合并之后, 就只剩下一堆了。多多在合并果子时总共消耗的体力等于每次合并所耗体力之和。

因为还要花大力气把这些果子搬回家,所以多多在合并果子时要尽可能地节省体力。假定每个果子重量都为 $1$ ,并且已知果子的种类 数和每种果子的数目,你的任务是设计出合并的次序方案,使多多耗费的体力最少,并输出这个最小的体力耗费值。

例如有 $3$ 种果子,数目依次为 $1$ , $2$ , $9$ 。可以先将 $1$ 、 $2$ 堆合并,新堆数目为 $3$ ,耗费体力为 $3$ 。接着,将新堆与原先的第三堆合并,又得到新的堆,数目为 $12$ ,耗费体力为 $12$ 。所以多多总共耗费体力 $=3+12=15$ 。可以证明 $15$ 为最小的体力耗费值。

## 输入格式

共两行。  
第一行是一个整数 $n(1\leq n\leq 10000)$ ,表示果子的种类数。  

第二行包含 $n$ 个整数,用空格分隔,第 $i$ 个整数 $a_i(1\leq a_i\leq 20000)$ 是第 $i$ 种果子的数目。

## 输出格式

一个整数,也就是最小的体力耗费值。输入数据保证这个值小于 $2^{31}$ 。

## 样例 #1

### 样例输入 #1

```

1 2 9
```

### 样例输出 #1

```
15
```

## 提示

对于 $30\%$ 的数据,保证有 $n \le 1000$:

对于 $50\%$ 的数据,保证有 $n \le 5000$;

对于全部的数据,保证有 $n \le 10000$。

思路:堆叠果子,可以先从两堆开始堆起,现在有一堆果子A:10斤,另一堆果子B:一斤,现在要将他们放在一堆,显然会选择将B果子搬到A果子上,这样消耗的体力最小,最省事。对于n堆这样的果子也是一样的,将他们排序(优先队列priority_queue自动排序),然后依次将最小的一堆叠加到第二小的那一堆,循环到只有一堆果子。

优先队列的定义:

  1. 默认情况下是以降序为准:priority_queue<int> Q(等价于priority_queue<int,vector<int>,less<int> >)
  2. 若要是升序排列:priority_queue<int,vector<int>,greater<int> > Q;
#include<bits/stdc++.h>
using namespace std;
int n,ans;
priority_queue<int,vector<int>,greater<int> > Q;
int main()
{
	cin>>n;
	for(int i=1;i<=n;i++){
		int x;
		cin>>x;
		Q.push(x);
	}
	while(Q.size()!=1){
		int x,y;
		x=Q.top();Q.pop();
		y=Q.top();Q.pop();
		ans+=x+y;
		Q.push(x+y);
	}
	cout<<ans;
	return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值