定义:贪心算法是指在对问题求解时,总是做出在当前看来是最好的选择。
也就是说,不从整体最优上加以考虑,只做出在某种意义上的局部最优解。
贪心算法不是对所有问题都能得到整体最优解,关键是贪心策略的选择,选择的贪心策略必须具备
无后效性,即某个状态以前的过程不会影响以后的状态,只与当前状态有关。
因此贪心算法一般只考虑当前的情况
例题:
题目描述
小涵很喜欢电脑游戏,这些天他正在玩一个叫做《三国》的游戏。
在游戏中,小涵和计算机各执一方,组建各自的军队进行对战。游戏中共有 NN 位武将(NN为偶数且不小于44),任意两个武将之间有一个“默契值”,表示若此两位武将作为一对组合作战时,该组合的威力有多大。游戏开始前,所有武将都是自由的(称为自由武将,一旦某个自由武将被选中作为某方军队的一员,那么他就不再是自由武将了),换句话说,所谓的自由武将不属于任何一方。
游戏开始,小涵和计算机要从自由武将中挑选武将组成自己的军队,规则如下:小涵先从自由武将中选出一个加入自己的军队,然后计算机也从自由武将中选出一个加入计算机方的军队。接下来一直按照“小涵→计算机→小涵→……”的顺序选择武将,直到所有的武将被双方均分完。然后,程序自动从双方军队中各挑出一对默契值最高的武将组合代表自己的军队进行二对二比武,拥有更高默契值的一对武将组合获胜,表示两军交战,拥有获胜武将组合的一方获胜。
已知计算机一方选择武将的原则是尽量破坏对手下一步将形成的最强组合,它采取的具体策略如下:任何时刻,轮到计算机挑选时,它会尝试将对手军队中的每个武将与当前每个自由武将进行一一配对,找出所有配对中默契值最高的那对武将组合,并将该组合中的自由武将选入自己的军队。 下面举例说明计算机的选将策略,例如,游戏中一共有66个武将,他们相互之间的默契值如下表所示:
双方选将过程如下所示:
小涵想知道,如果计算机在一局游戏中始终坚持上面这个策略,那么自己有没有可能必胜?如果有,在所有可能的胜利结局中,自己那对用于比武的武将组合的默契值最大是多少?
假设整个游戏过程中,对战双方任何时候均能看到自由武将队中的武将和对方军队的武将。为了简化问题,保证对于不同的武将组合,其默契值均不相同。
输入格式
共 N 行。
第一行为一个偶数 NN,表示武将的个数。
第 22行到第 NN行里,第i+1i+1行有N_iNi个非负整数,每两个数之间用一个空格隔开,表示ii号武将和i+1,i+2,…,Ni+1,i+2,…,N号武将之间的默契值(0≤0≤默契值≤1,000,000,000≤1,000,000,000)。
输出格式
共 11 或 22行。
若对于给定的游戏输入,存在可以让小涵获胜的选将顺序,则输出11,并另起一行输出所有获胜的情况中,小涵最终选出的武将组合的最大默契值。如果不存在可以让小涵获胜的选将顺序,则输出 00。
输入输出样例
输入 #1复制
6 5 28 16 29 27 23 3 20 1 8 32 26 33 11 12输出 #1复制
1 32输入 #2复制
8 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\%40%的数据有 N≤10N≤10。
对于70\%70%的数据有N≤18N≤18。
对于 100\%100%的数据有 N≤500N≤500。
思路:本来是用了模拟 但是一直通不过,无奈看了题解
才发现是博弈论(我哭死)
由于机器一直回破坏你所选择的武将的最大默契值(最优解)
因此人和机器都拿不到最优解,人是先手,所以人一定可以拿到次优解
因此最大值即为所有武将默契值的次优解中最大的
AC代码
#include<bits/stdc++.h>
using namespace std;
int n;
int main()
{ int ans=0;
int mq[502][502];
cin>>n;
for(int i=1;i<=n;i++)
{
for(int j=i+1;j<=n;j++)
{
cin>>mq[i][j];
mq[j][i]=mq[i][j];
}
}
for(int i=1;i<=n;i++)
{
sort(mq[i]+1,mq[i]+n+1);
if(ans<mq[i][n-1])
ans=mq[i][n-1];
}
cout<<"1"<<endl<<ans;
}
例题:
题目背景
战争已经进入到紧要时间。你是运输小队长,正在率领运输部队向前线运送物资。运输任务像做题一样的无聊。你希望找些刺激,于是命令你的士兵们到前方的一座独木桥上欣赏风景,而你留在桥下欣赏士兵们。士兵们十分愤怒,因为这座独木桥十分狭窄,只能容纳 11 个人通过。假如有 22 个人相向而行在桥上相遇,那么他们 22 个人将无法绕过对方,只能有 11 个人回头下桥,让另一个人先通过。但是,可以有多个人同时呆在同一个位置。
题目描述
突然,你收到从指挥部发来的信息,敌军的轰炸机正朝着你所在的独木桥飞来!为了安全,你的部队必须撤下独木桥。独木桥的长度为 LL,士兵们只能呆在坐标为整数的地方。所有士兵的速度都为 11,但一个士兵某一时刻来到了坐标为 00 或 L+1L+1 的位置,他就离开了独木桥。
每个士兵都有一个初始面对的方向,他们会以匀速朝着这个方向行走,中途不会自己改变方向。但是,如果两个士兵面对面相遇,他们无法彼此通过对方,于是就分别转身,继续行走。转身不需要任何的时间。
由于先前的愤怒,你已不能控制你的士兵。甚至,你连每个士兵初始面对的方向都不知道。因此,你想要知道你的部队最少需要多少时间就可能全部撤离独木桥。另外,总部也在安排阻拦敌人的进攻,因此你还需要知道你的部队最多需要多少时间才能全部撤离独木桥。
输入格式
第一行共一个整数 LL,表示独木桥的长度。桥上的坐标为 1, 2, \cdots, L1,2,⋯,L。
第二行共一个整数 NN,表示初始时留在桥上的士兵数目。
第三行共有 NN 个整数,分别表示每个士兵的初始坐标。
输出格式
共一行,输出 22 个整数,分别表示部队撤离独木桥的最小时间和最大时间。22 个整数由一个空格符分开。
输入输出样例
输入 #1复制
4 2 1 3输出 #1复制
2 4说明/提示
对于 100\%100% 的数据,满足初始时,没有两个士兵同在一个坐标,1\le L\le5\times 10^31≤L≤5×103,0\le N\le5\times10^30≤N≤5×103,且数据保证 N\le LN≤L。
关键点:那么当两个士兵撞在一起时,这与他们相互穿过并没有任何区别。
因此找maxtime和mintime时
只需找到每个士兵的max和min时间
最小时间即为所有人的min时间中的最大值
最大时间即为所有人的max时间中的最大值
利用max函数和两个数组用sort排序即可
#include<bits/stdc++.h>
using namespace std;
int n;
int temp=0;
int main()
{int l;
cin>>l>>n;
int kkk,jjj;
int ans1[5005],ans2[5005];
for(int i=0;i<n;i++)
{ cin>>temp;
kkk=max(l+1-temp,temp);
ans1[i]=kkk;
jjj=min(l+1-temp,temp);
ans2[i]=jjj;
}
sort(ans1,ans1+n);
sort(ans2,ans2+n);
cout<<ans2[n-1];
cout<<" ";
cout<<ans1[n-1];
}
例题:
题目描述
有 nn 个人在一个水龙头前排队接水,假如每个人接水的时间为 T_iTi,请编程找出这 nn 个人排队的一种顺序,使得 nn 个人的平均等待时间最小。
输入格式
第一行为一个整数 nn。
第二行 nn 个整数,第 ii 个整数 T_iTi 表示第 ii 个人的等待时间 T_iTi。
输出格式
输出文件有两行,第一行为一种平均时间最短的排队顺序;第二行为这种排列方案下的平均等待时间(输出结果精确到小数点后两位)。
输入输出样例
输入 #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^6n≤1000,ti≤106,不保证 t_iti 不重复。
当 t_iti 重复时,按照输入顺序即可(sort 是可以的)
本题思路是最基础的贪心算法
实现上可以使用结构体,也可以使用pair结构
我使用的是pair,first是等待时间,second是序号
因为pair会先比较first'
最后计算sum和ave即可
代码如下:
#include<bits/stdc++.h>
using namespace std;
int main()
{ double sum=0.0;
int n;
cin>>n;
pair<int,int> p[1001];
for(int i=0;i<n;i++)
{
cin>>p[i].first;
p[i].second=i+1;
}
sort(p,p+n);
for(int i=0;i<n;i++)
{
cout<<p[i].second<<" ";
}
for(int i=0;i<n;i++)
{
sum=sum+p[i].first*(n-1-i);
}
sum=sum/n;
printf("\n%.2lf",sum);
}
例题:
题目描述
在一个果园里,多多已经将所有的果子打了下来,而且按果子的不同种类分成了不同的堆。多多决定把所有的果子合成一堆。
每一次合并,多多可以把两堆果子合并到一起,消耗的体力等于两堆果子的重量之和。可以看出,所有的果子经过 n-1n−1 次合并之后, 就只剩下一堆了。多多在合并果子时总共消耗的体力等于每次合并所耗体力之和。
因为还要花大力气把这些果子搬回家,所以多多在合并果子时要尽可能地节省体力。假定每个果子重量都为 11 ,并且已知果子的种类 数和每种果子的数目,你的任务是设计出合并的次序方案,使多多耗费的体力最少,并输出这个最小的体力耗费值。
例如有 33 种果子,数目依次为 11 , 22 , 99 。可以先将 11 、 22 堆合并,新堆数目为 33 ,耗费体力为 33 。接着,将新堆与原先的第三堆合并,又得到新的堆,数目为 1212 ,耗费体力为 1212 。所以多多总共耗费体力 =3+12=15=3+12=15 。可以证明 1515 为最小的体力耗费值。
输入格式
共两行。
第一行是一个整数 n(1\leq n\leq 10000)n(1≤n≤10000) ,表示果子的种类数。第二行包含 nn 个整数,用空格分隔,第 ii 个整数 a_i(1\leq a_i\leq 20000)ai(1≤ai≤20000) 是第 ii 种果子的数目。
输出格式
一个整数,也就是最小的体力耗费值。输入数据保证这个值小于 2^{31}231 。
输入输出样例
输入 #1复制
3 1 2 9输出 #1复制
15说明/提示
对于 30\%30% 的数据,保证有 n \le 1000n≤1000:
对于 50\%50% 的数据,保证有 n \le 5000n≤5000;
对于全部的数据,保证有 n \le 10000n≤10000。
思路:
本题需要使用优先队列(priority_queue)
并使用降序队列
使用greater<>后,数据从大到小排列,top()返回的就是最小值而不是最大值
因此可以提取出此时队列中的最小两堆并将其合并
并将力气值加入到sum中
同时将二者的和再push进降序队列中
进行下一次的合并果子堆
结束条件:此时pop出两堆之后就队列为空
则结束程序
AC代码如下
:
#include<bits/stdc++.h>
using namespace std;
int main()
{
int n;int temp;
int sum=0,fin;
cin>>n;
priority_queue <int,vector<int>,greater<int> > q;
for(int i=0;i<n;i++)
{
cin>>temp;
q.push(temp);
}
while (!q.empty())
{ sum=sum+q.top();
fin=0;
fin=fin+q.top();
q.pop();
sum=sum+q.top();
fin=fin+q.top();
q.pop();
if(!q.empty())
q.push(fin);
}
cout<<sum;
}