贪新算法以及具体实例
2017-04-23 13:50
204 查看
贪心算法的基本要素:1.贪心选择性质。所谓贪心选择性质是指所求问题的整体最优解可以通过一系列局部最优的选择,即贪心选择来达到。这是贪心算法可行的第一个基本要素,也是贪心算法与动态规划算法的主要区别。动态规划算法通常以自底向上的方式解各子问题,而贪心算法则通常以自顶向下的方式进行,以迭代的方式作出相继的贪心选择,每作一次贪心选择就将所求问题简化为规模更小的子问题。对于一个具体问题,要确定它是否具有贪心选择性质,必须证明每一步所作的贪心选择最终导致问题的整体最优解。2.当一个问题的最优解包含其子问题的最优解时,称此问题具有最优子结构性质。问题的最优子结构性质是该问题可用动态规划算法或贪心算法求解的关键特征。贪心算法的基本思路:从问题的某一个初始解出发逐步逼近给定的目标,以尽可能快的地求得更好的解。当达到算法中的某一步不能再继续前进时,算法停止。该算法存在问题:1.
不能保证求得的最后解是最佳的;2.不能用来求最大或最小解问题;3.只能求满足某些约束条件的可行解的范围。实现该算法的过程:从问题的某一初始解出发;while能朝给定总目标前进一步do 求出可行解的一个解元素;由所有解元素组合成问题的一个可行解;用背包问题来介绍贪心算法:背包问题:有一个背包,背包容量是M=150。有7个物品,物品可以分割成任意大小。要求尽可能让装入背包中的物品总价值最大,但不能超过总容量。物品ABCDEFG重量35306050401025价值1040
3050354030分析如下目标函数:∑pi最大约束条件是装入的物品总重量不超过背包容量:∑wi<=M(M=150)。(1)根据贪心的策略,每
4000
次挑选价值最大的物品装入背包,得到的结果是否最优?(2)每次挑选所占重量最小的物品装入是否能得到最优解?(3)每次选取单位重量价值最大的物品,成为解本题的策略。值得注意的是,贪心算法并不是完全不可以使用,贪心策略一旦经过证明成立后,它就是一种高效的算法。贪心算法还是很常见的算法之一,这是由于它简单易行,构造贪心策略不是很困难。可惜的是,它需要证明后才能真正运用到题目的算法中。一般来说,贪心算法的证明围绕着:整个问题的最优解一定由在贪心策略中存在的子问题的最优解得来的。对于背包问题中的3种贪心策略,都是无法成立(无法被证明)的,解释如下:贪心策略:选取价值最大者。反例:W=30物品:A
BC重量:281212价值:302020根据策略,首先选取物品A,接下来就无法再选取了,可是,选取B、C则更好。(2)贪心策略:选取重量最小。它的反例与第一种策略的反例差不多。(3)贪心策略:选取单位重量价值最大的物品。反例:W=30物品:ABC重量:282010价值:282010根据策略,三种物品单位重量价值一样,程序无法依据现有策略作出判断,如果选择A,则答案错误。但是果在条件中加一句当遇见单位价值相同的时候,优先装重量小的,这样的问题就可以解决.所以需要说明的是,贪心算法可以与随机化算法一起使用,具体的例子就不再多举了。(因为这一类算法普及性不高,而且技术含量是非常高的,需要通过一些反例确定随机的对象是什么,随机程度如何,但也是不能保证完全正确,只能是极大的几率正确)。
网上对于这个装包问题的描述就就只有这些,但是在这里我还是要写一下,假设条件是什么?假设条件是上述几种反例的情况不存在的时候该如何求解:
贪心算法的基本要素:
1.贪心选择性质。所谓贪心选择性质是指所求问题的整体最优解可以通过一系列局部最优的选择,即贪心选择来达到。这是贪心算法可行的第一个基本要素,也是贪心算法与动态规划算法的主要区别。
动态规划算法通常以自底向上的方式解各子问题,而贪心算法则通常以自顶向下的方式进行,以迭代的方式作出相继的贪心选择,每作一次贪心选择就将所求问题简化为规模更小的子问题。
对于一个具体问题,要确定它是否具有贪心选择性质,必须证明每一步所作的贪心选择最终导致问题的整体最优解。
2.当一个问题的最优解包含其子问题的最优解时,称此问题具有最优子结构性质。问题的最优子结构性质是该问题可用动态规划算法或贪心算法求解的关键特征。
贪心算法的基本思路:
从问题的某一个初始解出发逐步逼近给定的目标,以尽可能快的地求得更好的解。当达到算法中的某一步不能再继续前进时,算法停止。
该算法存在问题:
1.不能保证求得的最后解是最佳的;
2.不能用来求最大或最小解问题;
3.只能求满足某些约束条件的可行解的范围。
实现该算法的过程:
从问题的某一初始解出发;
while能朝给定总目标前进一步do
求出可行解的一个解元素;
由所有解元素组合成问题的一个可行解;
用背包问题来介绍贪心算法:
背包问题:有一个背包,背包容量是M=150。有7个物品,物品可以分割成任意大小。要求尽可能让装入背包中的物品总价值最大,但不能超过总容量。
物品ABCDEFG
重量35306050401025
价值10403050354030
分析如下
目标函数:∑pi最大
约束条件是装入的物品总重量不超过背包容量:∑wi<=M(M=150)。
(1)根据贪心的策略,每次挑选价值最大的物品装入背包,得到的结果是否最优?
(2)每次挑选所占重量最小的物品装入是否能得到最优解?
(3)每次选取单位重量价值最大的物品,成为解本题的策略。
值得注意的是,贪心算法并不是完全不可以使用,贪心策略一旦经过证明成立后,它就是一种高效的算法。
贪心算法还是很常见的算法之一,这是由于它简单易行,构造贪心策略不是很困难。
可惜的是,它需要证明后才能真正运用到题目的算法中。
一般来说,贪心算法的证明围绕着:整个问题的最优解一定由在贪心策略中存在的子问题的最优解得来的。
对于背包问题中的3种贪心策略,都是无法成立(无法被证明)的,解释如下:
贪心策略:选取价值最大者。反例:
W=30
物品:ABC
重量:281212
价值:302020
根据策略,首先选取物品A,接下来就无法再选取了,可是,选取B、C则更好。
(2)贪心策略:选取重量最小。它的反例与第一种策略的反例差不多。
(3)贪心策略:选取单位重量价值最大的物品。反例:
W=30
物品:ABC
重量:282010
价值:282010
根据策略,三种物品单位重量价值一样,程序无法依据现有策略作出判断,如果选择A,则答案错误。但是果在条件中加一句当遇见单位价值相同的时候,优先装重量小的,这样的问题就可以解决.
所以需要说明的是,贪心算法可以与随机化算法一起使用,具体的例子就不再多举了。(因为这一类算法普及性不高,而且技术含量是非常高的,需要通过一些反例确定随机的对象是什么,随机程度如何,但也是不能保证完全正确,只能是极大的几率正确)。
网上对于这个装包问题的描述就就只有这些,但是在这里我还是要写一下,假设条件是什么?假设条件是上述几种反例的情况不存在的时候该如何求解:
[cpp]view
plaincopy
#include<iostream>
usingnamespacestd;
structNode
{
floatweight;
floatvalue;
boolmark;
charchar_mark;
floatpre_weight_value;
};
intmain(intargc,char*argv[])
{
floatWeight[7]={35,30,60,50,40,15,20};
floatValue[7]={10,40,30,50,35,40,30};
Nodearray[7];
for(inti=0;i<7;i++)
{
array[i].value=Value[i];
array[i].weight=Weight[i];
array[i].char_mark=65+i;
array[i].mark=false;
array[i].pre_weight_value=Value[i]/Weight[i];
}
for(i=0;i<7;i++)
cout<<array[i].pre_weight_value<<"";
cout<<endl;
floatweight_all=0.0;
floatvalue_all=0.0;
floatmax=0.0;
charcharArray[7];
intflag,n=0;
while(weight_all<=150)
{
for(intindex=0;index<7;++index)
{
if(array[index].pre_weight_value>max&&array[index].mark==false)
{
max=array[index].pre_weight_value;
flag=index;
}
}
charArray[n++]=array[flag].char_mark;
array[flag].mark=true;
weight_all+=array[flag].weight;
value_all+=array[flag].value;
max=0.0;
}
for(i=0;i<n-1;i++)
cout<<charArray[i]<<"";
cout<<endl;
cout<<"weight_all:"<<weight_all-array[n-1].weight<<endl;
cout<<"value_all:"<<value_all<<endl;
system("pause");
return0;
}
下面我要说的是,这个算法里面就是采用的贪心第三方案,一般这个方案是成功率最大的,其他两个方案我在这里没有考虑,在这里得到的结果是利用了115容量装了价值195的东西,但是这明显不是最优结果,分明还可以装一个A进去!刚好满足150重量,由于在算法中我单纯的利用第三种贪心方法求解,当剩余的包裹中最优的再加进来的时候已经超过了,所以这个时候可以选择剩余包裹中次优的(如这里选择A),再不行就次次优的,尽量把包裹装满,这样得到的结果就很接近了(不保证一定为最优),但是我们一般不这样来求解,下一文章会介绍动态规划算法来解决这个问题,动态规划很好的弥补了贪心算法的不足!详见下一章!!
还需要说明的是,如果包裹是可以拆分的,那这个问题就得到了整体最优解,前面不变,就是当最后一次装进去已经超过容量的时候可以选择只装她的一部分!很多编程题一般是这种情况!
经自己总结的贪心算法几大经典问题:
1:活动时间安排的问题
设有N个活动时间集合,每个活动都要使用同一个资源,比如说会议场,而且同一时间内只能有一个活动使用,每个活动都有一个使用活动的开始si和结束时间fi,即他的使用区间为(si,fi),现在要求你分配活动占用时间表,即哪些活动占用该会议室,哪些不占用,使得他们不冲突,要求是尽可能多的使参加的活动最大化,即所占时间区间最大化!
上图为每个活动的开始和结束时间,我们的任务就是设计程序输出哪些活动可以占用会议室!
[cpp]view
plaincopy
#include<iostream>
usingnamespacestd;
voidGreedyChoose(intlen,int*s,int*f,bool*flag);
intmain(intargc,char*argv[])
{
ints[11]={1,3,0,5,3,5,6,8,8,2,12};
intf[11]={4,5,6,7,8,9,10,11,12,13,14};
boolmark[11]={0};
GreedyChoose(11,s,f,mark);
for(inti=0;i<11;i++)
if(mark[i])
cout<<i<<"";
system("pause");
return0;
}
voidGreedyChoose(intlen,int*s,int*f,bool*flag)
{
flag[0]=true;
intj=0;
for(inti=1;i<len;++i)
if(s[i]>=f[j])
{
flag[i]=true;
j=i;
}
}
得出结果是03710,也就是对应的时间段
值得说明一下,虽然贪心算法不是一定可以得到最好的解,但是对于这种活动时间的问题,他却得到的总是最优解,这点可以用数学归纳法证明,在这里,体现出来的贪心策略是:每一个活动时间的挑选总是选择最优的,就是刚好匹配的,这样得出的结果也就是最优的了!由于这个算法很简单,在这里就没有注释了!
类似这种题还有个区间覆盖问题,就是说很多个区间,其中有些是相互覆盖着的,要求去除多余的区间,使剩下的区间占用长度最大,实际就是这个题,只是问法变换了而已!接下来让我们看线性覆盖的问题,跟上面的相反!
2.贪心实例之线段覆盖(linescover)
题目大意:
在一维空间中告诉你N条线段的起始坐标与终止坐标,要求求出这些线段一共覆盖了多大的长度。
为了方便说明,我们采用上述表格中的数据代表10条线段的起始点和终点,注意,这里是用起始点为顺序进行排列,和上面的不一样,知道了这些我们就可以着手开始设计这个程序:
[cpp]view
plaincopy
#include<iostream>
usingnamespacestd;
intmain(intargc,char*argv[])
{
ints[10]={2,3,4,5,6,7,8,9,10,11};
intf[10]={3,5,7,6,9,8,12,10,13,15};
intTotalLength=(3-2);
for(inti=1,intj=0;i<10;++i)
{
if(s[i]>=f[j])
{
TotalLength+=(f[i]-s[i]);
j=i;
}
else
{
if(f[i]<=f[j])
continue;
else
{
TotalLength+=f[i]-f[j];
j=i;
}
}
}
cout<<TotalLength<<endl;
system("pause");
return0;
}
运行结果为13,显然这是我们需要的结果,这里注明一下,上面图表中数据有点问题,实际以程序中给出的为主!
3,:数字组合问题!
设有N个正整数,现在需要你设计一个程序,使他们连接在一起成为最大的数字,例3个整数12,456,342很明显是45634212为最大,4个整数342,45,7,98显然为98745342最大
程序要求:输入整数N接下来一行输入N个数字,最后一行输出最大的那个数字!
题目解析:拿到这题目,看起要来也简单,看起来也难,简单在什么地方,简单在好像就是寻找哪个开头最大,然后连在一起就是了,难在如果N大了,假如几千几万,好像就不是那么回事了,要解答这个题目需要选对合适的贪心策略,并不是把数字由大排到小那么简单,网上的解法是将数字转化为字符串,比如a+b和b+a,用strcmp函数比较一下就知道谁大,也就知道了谁该排在谁前面,不过我觉得这个完全没必要,在这里我采用一种比较巧妙的方法来解答,不知道大家还记得冒泡排序法不,那是排序最早接触的一种方法,我们先看看它的源代码:
[cpp]view
plaincopy
#include<iostream>
usingnamespacestd;
intmain(intargc,char*argv[])
{
intarray[10];
for(inti=0;i<10;i++)
cin>>array[i];
inttemp;
for(i=0;i<=9;++i)
for(intj=0;j<10-1-i;j++)
if(array[j]>array[j+1])
{
temp=array[j];
array[j]=array[j+1];
array[j+1]=temp;
}
for(i=0;i<10;i++)
cout<<array[i]<<"";
cout<<endl;
system("pause");
return0;
[cpp]view
plaincopy
}
[cpp]view
plaincopy
相信这种冒泡已经很熟悉了,注意看程序中最核心的比较规则是什么,是这一句if(array[j]>array[j+1])他是以数字大小作为比较准则来返回true或者是false,那么我们完全可以改变一下这个排序准则,比如23,123,这两个数字,在我们这个题中它可以组成两个数字23123和12323,分明是前者大些,所以我们可以说23排在123前面,也就是23的优先级比123大,123的优先级比23小,所以不妨写个函数,传递参数a和b,如果ab比ba大,则返回true,反之返回false,函数原型如下:
[cpp]view
plaincopy
[cpp]view
plaincopy
<preclass="cpp"name="code">boolcompare(intNum1,intNum2)
{
intcount1,count2;
intMidNum1=Num1,MidNum2=Num2;
while(MidNum1)
{
++count1;
MidNum1/=10;
}
while(MidNum2)
{
++count2;
MidNum2/=10;
}
inta=Num1*pow(10,count2)+Num2;
intb=Num2*pow(10,count1)+Num1;
return(a>b)?true:false;
}</pre>
<pre></pre>
<p>好了,我们的比较准则函数也已经完成了,只需要把这个比较准则加到关键的地方,这个题就算完成了,最终代码如下:</p>
<preclass="cpp"name="code">#include<iostream>
#include<cmath>
usingnamespacestd;
boolcompare(intNum1,intNum2);
intmain(intargc,char*argv[])
{
intN;
cout<<"pleaseenterthenumbern:"<<endl;
cin>>N;
int*array=newint
;
for(inti=0;i<N;i++)
cin>>array[i];
inttemp;
for(i=0;i<=N-1;++i)
{
for(intj=0;j<N-i-1;j++)
if(compare(array[j],array[j+1]))
{
temp=array[j];
array[j]=array[j+1];
array[j+1]=temp;
}
}
cout<<"themaxnumberis:";
for(i=N-1;i>=0;--i)
cout<<array[i];
cout<<endl;
delete[]array;
system("pause");
return0;
}
boolcompare(intNum1,intNum2)
{
intcount1=0,count2=0;
intMidNum1=Num1,MidNum2=Num2;
while(MidNum1)
{
++count1;
MidNum1/=10;
}
while(MidNum2)
{
++count2;
MidNum2/=10;
}
inta=Num1*pow(10,count2)+Num2;
intb=Num2*pow(10,count1)+Num1;
return(a>b)?true:false;
}</pre>
<p><br>
</p>
<p>运行测试:</p>
<p><imgalt=""src="http://img.my.csdn.net/uploads/201303/31/1364663151_2496.jpg"></p>
<p>可以看见这样很巧妙的改编冒泡排序就解决了这个问题,当然也可以用其他的排序算法,或者干脆用一个仿函数作为set容器的排序准则,insert进去就可以了,但是这个程序有一点小问题,假如输入的数字中有两个濒临越界的数据,合并在一起就越界了,那样就只能用字符串的形式进行比较!</p>
<p></p>
<p><spanstyle="color:#ff0000">4,:找零钱的问题</span></p>
<p>在贪心算法里面最常见的莫过于找零钱的问题了,题目大意如下,对于人民币的面值有1元5元10元20元50元100元,下面要求设计一个程序,输入找零的钱,输出找钱方案中最少张数的方案,比如123元,最少是1张100的,1张20的,3张1元的,一共5张!</p>
<p>解析:这样的题目运用的贪心策略是每次选择最大的钱,如果最后超过了,再选择次大的面值,然后次次大的面值,一直到最后与找的钱相等,这种情况大家再熟悉不过了,下面就直接看源代码:</p>
<p></p>
<preclass="cpp"name="code">#include<iostream>
#include<cmath>
usingnamespacestd;
intmain(intargc,char*argv[])
{
intMoneyClass[6]={100,50,20,10,5,1};//记录钱的面值
intMoneyIndex[6]={0};//记录每种面值的数量
intMoneyAll,MoneyCount=0,count=0;
cout<<"pleaseentertheallmoneyyouwanttoexchange:"<<endl;
cin>>MoneyAll;
for(inti=0;i<6;)//只有这个循环才是主体
{
if(MoneyCount+MoneyClass[i]>MoneyAll)
{
i++;
continue;
}
MoneyCount+=MoneyClass[i];
++MoneyIndex[i];
++count;
if(MoneyCount==MoneyAll)
break;
}
for(i=0;i<6;++i)//控制输出的循环
{
if(MoneyIndex[i]!=0)
{
switch(i)
{
case0:
cout<<"the100have:"<<MoneyIndex[i]<<endl;
break;
case1:
cout<<"the50have:"<<MoneyIndex[i]<<endl;
break;
case2:
cout<<"the20have:"<<MoneyIndex[i]<<endl;
break;
case3:
cout<<"the10have:"<<MoneyIndex[i]<<endl;
break;
case4:
cout<<"the5have:"<<MoneyIndex[i]<<endl;
break;
case5:
cout<<"the1have:"<<MoneyIndex[i]<<endl;
break;
}
}
}
cout<<"thetotalmoneyhave:"<<count<<endl;
system("pause");
return0;
}</pre><preclass="cpp"name="code"></pre><preclass="cpp"name="code"></pre>
<p><spanstyle="font-size:24px;color:#ff0000">由于精力有限,贪心算法的很多题还没写,以后有时间会补上,其实最主要就是记住贪心策略,每次选择的都是对于当前而言最优的,贪心思想不难,利用好就需要多练习,望一起进步!(下一算法:动态规划算法)</span></p>
<pre></pre>
<pre></pre>
<pre></pre>
<linkrel="stylesheet"href="http://static.blog.csdn.net/public/res-min/markdown_views.css?v=1.0">
下面我要说的是,这个算法里面就是采用的贪心第三方案,一般这个方案是成功率最大的,其他两个方案我在这里没有考虑,在这里得到的结果是利用了115容量装了价值195的东西,但是这明显不是最优结果,分明还可以装一个A进去!刚好满足150重量,由于在算法中我单纯的利用第三种贪心方法求解,当剩余的包裹中最优的再加进来的时候已经超过了,所以这个时候可以选择剩余包裹中次优的(如这里选择A),再不行就次次优的,尽量把包裹装满,这样得到的结果就很接近了(不保证一定为最优),但是我们一般不这样来求解,下一文章会介绍动态规划算法来解决这个问题,动态规划很好的弥补了贪心算法的不足!详见下一章!!
还需要说明的是,如果包裹是可以拆分的,那这个问题就得到了整体最优解,前面不变,就是当最后一次装进去已经超过容量的时候可以选择只装她的一部分!很多编程题一般是这种情况!经自己总结的贪心算法几大经典问题:1:活动时间安排的问题设有N个活动时间集合,每个活动都要使用同一个资源,比如说会议场,而且同一时间内只能有一个活动使用,每个活动都有一个使用活动的开始si和结束时间fi,即他的使用区间为(si,fi),现在要求你分配活动占用时间表,即哪些活动占用该会议室,哪些不占用,使得他们不冲突,要求是尽可能多的使参加的活动最大化,即所占时间区间最大化!上图为每个活动的开始和结束时间,我们的任务就是设计程序输出哪些活动可以占用会议室![cpp]
viewplaincopy#includeusingnamespacestd;voidGreedyChoose(intlen,int*s,int*f,bool*flag);intmain(intargc,char*argv[]){ints[11]={1,3,0,5,3,5,6,8,8,2,12};intf[11]={4,5,6,7,8,9,10,11,12,13,14};boolmark[11]={0};GreedyChoose(11,s,f,mark);
for(inti=0;i<11;i++)if(mark[i])cout<<i<<"";system("pause");return0;}voidGreedyChoose(intlen,int*s,int*f,bool*flag){flag[0]=true;intj=0;for(inti=1;i=f[j]){flag[i]=true;j=i;}}得出结果是03710,也就是对应的时间段值得说明一下,虽然贪心算法不是一定可以得到最好的解
,但是对于这种活动时间的问题,他却得到的总是最优解,这点可以用数学归纳法证明,在这里,体现出来的贪心策略是:每一个活动时间的挑选总是选择最优的,就是刚好匹配的,这样得出的结果也就是最优的了!由于这个算法很简单,在这里就没有注释了!类似这种题还有个区间覆盖问题,就是说很多个区间,其中有些是相互覆盖着的,要求去除多余的区间,使剩下的区间占用长度最大,实际就是这个题,只是问法变换了而已!接下来让我们看线性覆盖的问题,跟上面的相反!2.贪心实例之线段覆盖(linescover)题目大意:在一维空间中告诉你N条线段的起始坐标与终止坐标,要求求出这些线段一共覆盖了多大的长度。为了方便说明,我们采用上述表格中的数据代表10条线段的起始点和终点,注意,这里是用起始点为顺序进行排列,和上面的不一样,知道了这些我们就可以着手开始设计这个程序:
[cpp]viewplaincopy#includeusingnamespacestd;intmain(intargc,char*argv[]){ints[10]={2,3,4,5,6,7,8,9,10,11};intf[10]={3,5,7,6,9,8,12,10,13,15};intTotalLength=(3-2);for(inti=1,intj=0;i<10i=""if=""s=""i="">=f[j]){TotalLength
+=(f[i]-s[i]);j=i;}else{if(f[i]<=f[j])continue;else{TotalLength+=f[i]-f[j];j=i;}}}cout<<TotalLength<<endl;system("pause");return0;}运行结果为13,显然这是我们需要的结果,这里注明一下,上面图表中数据有点问题,实际以程序中给出的为主!3,:数字组合问题!设有N个正整数,现在需要你设计一个程序,使他们连接在一起成为最大的数字,例3个整数
12,456,342很明显是45634212为最大,4个整数342,45,7,98显然为98745342最大程序要求:输入整数N接下来一行输入N个数字,最后一行输出最大的那个数字!题目解析:拿到这题目,看起要来也简单,看起来也难,简单在什么地方,简单在好像就是寻找哪个开头最大,然后连在一起就是了,难在如果N大了,假如几千几万,好像就不是那么回事了,要解答这个题目需要选对合适的贪心策略,并不是把数字由大排到小那么简单,网上的解法是将数字转化为字符串,比如a+b和b+a,用strcmp函数比较一下就知道谁大,也就知道了谁该排在谁前面,不过我觉得这个完全没必要,在这里我采用一种比较巧妙的方法来解答,不知道大家还记得冒泡排序法不,那是排序最早接触的一种方法,我们先看看它的源代码:[cpp]
viewplaincopy#includeusingnamespacestd;intmain(intargc,char*argv[]){intarray[10];for(inti=0;i<10i=""cin="">>array[i];inttemp;for(i=0;i<=9;++i)for(intj=0;j<10-1-ij=""if=""array=""j="">array[j+1]){temp=array[j];array[j]=
array[j+1];array[j+1]=temp;}for(i=0;i<10;i++)cout<<array[i]<<"";cout<array[j+1])他是以数字大小作为比较准则来返回true或者是false,那么我们完全可以改变一下这个排序准则,比如23,123,这两个数字,在我们这个题中它可以组成两个数字23123和12323,分明是前者大些,所以我们可以说23排在123前面,也就是23的优先级比123大,123的优先级比23小,所以不妨写个函数,传递参数a和b,如果ab比ba大,则返回true,反之返回false,函数原型如下:
[cpp]viewplaincopy[cpp]viewplaincopy
好了,我们的比较准则函数也已经完成了,只需要把这个比较准则加到关键的地方,这个题就算完成了,最终代码如下:
运行测试:
可以看见这样很巧妙的改编冒泡排序就解决了这个问题,当然也可以用其他的排序算法,或者干脆用一个仿函数作为set容器的排序准则,insert进去就可以了,但是这个程序有一点小问题,假如输入的数字中有两个濒临越界的数据,合并在一起就越界了,那样就只能用字符串的形式进行比较!
4,:找零钱的问题
在贪心算法里面最常见的莫过于找零钱的问题了,题目大意如下,对于人民币的面值有1元5元10元20元50元100元,下面要求设计一个程序,输入找零的钱,输出找钱方案中最少张数的方案,比如123元,最少是1张100的,1张20的,3张1元的,一共5张!
解析:这样的题目运用的贪心策略是每次选择最大的钱,如果最后超过了,再选择次大的面值,然后次次大的面值,一直到最后与找的钱相等,这种情况大家再熟悉不过了,下面就直接看源代码:
由于精力有限,贪心算法的很多题还没写,以后有时间会补上,其实最主要就是记住贪心策略,每次选择的都是对于当前而言最优的,贪心思想不难,利用好就需要多练习,望一起进步!(下一算法:动态规划算法)
好了,我们的比较准则函数也已经完成了,只需要把这个比较准则加到关键的地方,这个题就算完成了,最终代码如下:
运行测试:
可以看见这样很巧妙的改编冒泡排序就解决了这个问题,当然也可以用其他的排序算法,或者干脆用一个仿函数作为set容器的排序准则,insert进去就可以了,但是这个程序有一点小问题,假如输入的数字中有两个濒临越界的数据,合并在一起就越界了,那样就只能用字符串的形式进行比较!
4,:找零钱的问题
在贪心算法里面最常见的莫过于找零钱的问题了,题目大意如下,对于人民币的面值有1元5元10元20元50元100元,下面要求设计一个程序,输入找零的钱,输出找钱方案中最少张数的方案,比如123元,最少是1张100的,1张20的,3张1元的,一共5张!
解析:这样的题目运用的贪心策略是每次选择最大的钱,如果最后超过了,再选择次大的面值,然后次次大的面值,一直到最后与找的钱相等,这种情况大家再熟悉不过了,下面就直接看源代码:
由于精力有限,贪心算法的很多题还没写,以后有时间会补上,其实最主要就是记住贪心策略,每次选择的都是对于当前而言最优的,贪心思想不难,利用好就需要多练习,望一起进步!(下一算法:动态规划算法)
不能保证求得的最后解是最佳的;2.不能用来求最大或最小解问题;3.只能求满足某些约束条件的可行解的范围。实现该算法的过程:从问题的某一初始解出发;while能朝给定总目标前进一步do 求出可行解的一个解元素;由所有解元素组合成问题的一个可行解;用背包问题来介绍贪心算法:背包问题:有一个背包,背包容量是M=150。有7个物品,物品可以分割成任意大小。要求尽可能让装入背包中的物品总价值最大,但不能超过总容量。物品ABCDEFG重量35306050401025价值1040
3050354030分析如下目标函数:∑pi最大约束条件是装入的物品总重量不超过背包容量:∑wi<=M(M=150)。(1)根据贪心的策略,每
4000
次挑选价值最大的物品装入背包,得到的结果是否最优?(2)每次挑选所占重量最小的物品装入是否能得到最优解?(3)每次选取单位重量价值最大的物品,成为解本题的策略。值得注意的是,贪心算法并不是完全不可以使用,贪心策略一旦经过证明成立后,它就是一种高效的算法。贪心算法还是很常见的算法之一,这是由于它简单易行,构造贪心策略不是很困难。可惜的是,它需要证明后才能真正运用到题目的算法中。一般来说,贪心算法的证明围绕着:整个问题的最优解一定由在贪心策略中存在的子问题的最优解得来的。对于背包问题中的3种贪心策略,都是无法成立(无法被证明)的,解释如下:贪心策略:选取价值最大者。反例:W=30物品:A
BC重量:281212价值:302020根据策略,首先选取物品A,接下来就无法再选取了,可是,选取B、C则更好。(2)贪心策略:选取重量最小。它的反例与第一种策略的反例差不多。(3)贪心策略:选取单位重量价值最大的物品。反例:W=30物品:ABC重量:282010价值:282010根据策略,三种物品单位重量价值一样,程序无法依据现有策略作出判断,如果选择A,则答案错误。但是果在条件中加一句当遇见单位价值相同的时候,优先装重量小的,这样的问题就可以解决.所以需要说明的是,贪心算法可以与随机化算法一起使用,具体的例子就不再多举了。(因为这一类算法普及性不高,而且技术含量是非常高的,需要通过一些反例确定随机的对象是什么,随机程度如何,但也是不能保证完全正确,只能是极大的几率正确)。
网上对于这个装包问题的描述就就只有这些,但是在这里我还是要写一下,假设条件是什么?假设条件是上述几种反例的情况不存在的时候该如何求解:
贪心算法的基本要素:
1.贪心选择性质。所谓贪心选择性质是指所求问题的整体最优解可以通过一系列局部最优的选择,即贪心选择来达到。这是贪心算法可行的第一个基本要素,也是贪心算法与动态规划算法的主要区别。
动态规划算法通常以自底向上的方式解各子问题,而贪心算法则通常以自顶向下的方式进行,以迭代的方式作出相继的贪心选择,每作一次贪心选择就将所求问题简化为规模更小的子问题。
对于一个具体问题,要确定它是否具有贪心选择性质,必须证明每一步所作的贪心选择最终导致问题的整体最优解。
2.当一个问题的最优解包含其子问题的最优解时,称此问题具有最优子结构性质。问题的最优子结构性质是该问题可用动态规划算法或贪心算法求解的关键特征。
贪心算法的基本思路:
从问题的某一个初始解出发逐步逼近给定的目标,以尽可能快的地求得更好的解。当达到算法中的某一步不能再继续前进时,算法停止。
该算法存在问题:
1.不能保证求得的最后解是最佳的;
2.不能用来求最大或最小解问题;
3.只能求满足某些约束条件的可行解的范围。
实现该算法的过程:
从问题的某一初始解出发;
while能朝给定总目标前进一步do
求出可行解的一个解元素;
由所有解元素组合成问题的一个可行解;
用背包问题来介绍贪心算法:
背包问题:有一个背包,背包容量是M=150。有7个物品,物品可以分割成任意大小。要求尽可能让装入背包中的物品总价值最大,但不能超过总容量。
物品ABCDEFG
重量35306050401025
价值10403050354030
分析如下
目标函数:∑pi最大
约束条件是装入的物品总重量不超过背包容量:∑wi<=M(M=150)。
(1)根据贪心的策略,每次挑选价值最大的物品装入背包,得到的结果是否最优?
(2)每次挑选所占重量最小的物品装入是否能得到最优解?
(3)每次选取单位重量价值最大的物品,成为解本题的策略。
值得注意的是,贪心算法并不是完全不可以使用,贪心策略一旦经过证明成立后,它就是一种高效的算法。
贪心算法还是很常见的算法之一,这是由于它简单易行,构造贪心策略不是很困难。
可惜的是,它需要证明后才能真正运用到题目的算法中。
一般来说,贪心算法的证明围绕着:整个问题的最优解一定由在贪心策略中存在的子问题的最优解得来的。
对于背包问题中的3种贪心策略,都是无法成立(无法被证明)的,解释如下:
贪心策略:选取价值最大者。反例:
W=30
物品:ABC
重量:281212
价值:302020
根据策略,首先选取物品A,接下来就无法再选取了,可是,选取B、C则更好。
(2)贪心策略:选取重量最小。它的反例与第一种策略的反例差不多。
(3)贪心策略:选取单位重量价值最大的物品。反例:
W=30
物品:ABC
重量:282010
价值:282010
根据策略,三种物品单位重量价值一样,程序无法依据现有策略作出判断,如果选择A,则答案错误。但是果在条件中加一句当遇见单位价值相同的时候,优先装重量小的,这样的问题就可以解决.
所以需要说明的是,贪心算法可以与随机化算法一起使用,具体的例子就不再多举了。(因为这一类算法普及性不高,而且技术含量是非常高的,需要通过一些反例确定随机的对象是什么,随机程度如何,但也是不能保证完全正确,只能是极大的几率正确)。
网上对于这个装包问题的描述就就只有这些,但是在这里我还是要写一下,假设条件是什么?假设条件是上述几种反例的情况不存在的时候该如何求解:
[cpp]view
plain
#include<iostream>
usingnamespacestd;
structNode
{
floatweight;
floatvalue;
boolmark;
charchar_mark;
floatpre_weight_value;
};
intmain(intargc,char*argv[])
{
floatWeight[7]={35,30,60,50,40,15,20};
floatValue[7]={10,40,30,50,35,40,30};
Nodearray[7];
for(inti=0;i<7;i++)
{
array[i].value=Value[i];
array[i].weight=Weight[i];
array[i].char_mark=65+i;
array[i].mark=false;
array[i].pre_weight_value=Value[i]/Weight[i];
}
for(i=0;i<7;i++)
cout<<array[i].pre_weight_value<<"";
cout<<endl;
floatweight_all=0.0;
floatvalue_all=0.0;
floatmax=0.0;
charcharArray[7];
intflag,n=0;
while(weight_all<=150)
{
for(intindex=0;index<7;++index)
{
if(array[index].pre_weight_value>max&&array[index].mark==false)
{
max=array[index].pre_weight_value;
flag=index;
}
}
charArray[n++]=array[flag].char_mark;
array[flag].mark=true;
weight_all+=array[flag].weight;
value_all+=array[flag].value;
max=0.0;
}
for(i=0;i<n-1;i++)
cout<<charArray[i]<<"";
cout<<endl;
cout<<"weight_all:"<<weight_all-array[n-1].weight<<endl;
cout<<"value_all:"<<value_all<<endl;
system("pause");
return0;
}
下面我要说的是,这个算法里面就是采用的贪心第三方案,一般这个方案是成功率最大的,其他两个方案我在这里没有考虑,在这里得到的结果是利用了115容量装了价值195的东西,但是这明显不是最优结果,分明还可以装一个A进去!刚好满足150重量,由于在算法中我单纯的利用第三种贪心方法求解,当剩余的包裹中最优的再加进来的时候已经超过了,所以这个时候可以选择剩余包裹中次优的(如这里选择A),再不行就次次优的,尽量把包裹装满,这样得到的结果就很接近了(不保证一定为最优),但是我们一般不这样来求解,下一文章会介绍动态规划算法来解决这个问题,动态规划很好的弥补了贪心算法的不足!详见下一章!!
还需要说明的是,如果包裹是可以拆分的,那这个问题就得到了整体最优解,前面不变,就是当最后一次装进去已经超过容量的时候可以选择只装她的一部分!很多编程题一般是这种情况!
经自己总结的贪心算法几大经典问题:
1:活动时间安排的问题
设有N个活动时间集合,每个活动都要使用同一个资源,比如说会议场,而且同一时间内只能有一个活动使用,每个活动都有一个使用活动的开始si和结束时间fi,即他的使用区间为(si,fi),现在要求你分配活动占用时间表,即哪些活动占用该会议室,哪些不占用,使得他们不冲突,要求是尽可能多的使参加的活动最大化,即所占时间区间最大化!
上图为每个活动的开始和结束时间,我们的任务就是设计程序输出哪些活动可以占用会议室!
[cpp]view
plain
#include<iostream>
usingnamespacestd;
voidGreedyChoose(intlen,int*s,int*f,bool*flag);
intmain(intargc,char*argv[])
{
ints[11]={1,3,0,5,3,5,6,8,8,2,12};
intf[11]={4,5,6,7,8,9,10,11,12,13,14};
boolmark[11]={0};
GreedyChoose(11,s,f,mark);
for(inti=0;i<11;i++)
if(mark[i])
cout<<i<<"";
system("pause");
return0;
}
voidGreedyChoose(intlen,int*s,int*f,bool*flag)
{
flag[0]=true;
intj=0;
for(inti=1;i<len;++i)
if(s[i]>=f[j])
{
flag[i]=true;
j=i;
}
}
得出结果是03710,也就是对应的时间段
值得说明一下,虽然贪心算法不是一定可以得到最好的解,但是对于这种活动时间的问题,他却得到的总是最优解,这点可以用数学归纳法证明,在这里,体现出来的贪心策略是:每一个活动时间的挑选总是选择最优的,就是刚好匹配的,这样得出的结果也就是最优的了!由于这个算法很简单,在这里就没有注释了!
类似这种题还有个区间覆盖问题,就是说很多个区间,其中有些是相互覆盖着的,要求去除多余的区间,使剩下的区间占用长度最大,实际就是这个题,只是问法变换了而已!接下来让我们看线性覆盖的问题,跟上面的相反!
2.贪心实例之线段覆盖(linescover)
题目大意:
在一维空间中告诉你N条线段的起始坐标与终止坐标,要求求出这些线段一共覆盖了多大的长度。
为了方便说明,我们采用上述表格中的数据代表10条线段的起始点和终点,注意,这里是用起始点为顺序进行排列,和上面的不一样,知道了这些我们就可以着手开始设计这个程序:
[cpp]view
plain
#include<iostream>
usingnamespacestd;
intmain(intargc,char*argv[])
{
ints[10]={2,3,4,5,6,7,8,9,10,11};
intf[10]={3,5,7,6,9,8,12,10,13,15};
intTotalLength=(3-2);
for(inti=1,intj=0;i<10;++i)
{
if(s[i]>=f[j])
{
TotalLength+=(f[i]-s[i]);
j=i;
}
else
{
if(f[i]<=f[j])
continue;
else
{
TotalLength+=f[i]-f[j];
j=i;
}
}
}
cout<<TotalLength<<endl;
system("pause");
return0;
}
运行结果为13,显然这是我们需要的结果,这里注明一下,上面图表中数据有点问题,实际以程序中给出的为主!
3,:数字组合问题!
设有N个正整数,现在需要你设计一个程序,使他们连接在一起成为最大的数字,例3个整数12,456,342很明显是45634212为最大,4个整数342,45,7,98显然为98745342最大
程序要求:输入整数N接下来一行输入N个数字,最后一行输出最大的那个数字!
题目解析:拿到这题目,看起要来也简单,看起来也难,简单在什么地方,简单在好像就是寻找哪个开头最大,然后连在一起就是了,难在如果N大了,假如几千几万,好像就不是那么回事了,要解答这个题目需要选对合适的贪心策略,并不是把数字由大排到小那么简单,网上的解法是将数字转化为字符串,比如a+b和b+a,用strcmp函数比较一下就知道谁大,也就知道了谁该排在谁前面,不过我觉得这个完全没必要,在这里我采用一种比较巧妙的方法来解答,不知道大家还记得冒泡排序法不,那是排序最早接触的一种方法,我们先看看它的源代码:
[cpp]view
plain
#include<iostream>
usingnamespacestd;
intmain(intargc,char*argv[])
{
intarray[10];
for(inti=0;i<10;i++)
cin>>array[i];
inttemp;
for(i=0;i<=9;++i)
for(intj=0;j<10-1-i;j++)
if(array[j]>array[j+1])
{
temp=array[j];
array[j]=array[j+1];
array[j+1]=temp;
}
for(i=0;i<10;i++)
cout<<array[i]<<"";
cout<<endl;
system("pause");
return0;
[cpp]view
plain
}
[cpp]view
plain
相信这种冒泡已经很熟悉了,注意看程序中最核心的比较规则是什么,是这一句if(array[j]>array[j+1])他是以数字大小作为比较准则来返回true或者是false,那么我们完全可以改变一下这个排序准则,比如23,123,这两个数字,在我们这个题中它可以组成两个数字23123和12323,分明是前者大些,所以我们可以说23排在123前面,也就是23的优先级比123大,123的优先级比23小,所以不妨写个函数,传递参数a和b,如果ab比ba大,则返回true,反之返回false,函数原型如下:
[cpp]view
plain
[cpp]view
plain
<preclass="cpp"name="code">boolcompare(intNum1,intNum2)
{
intcount1,count2;
intMidNum1=Num1,MidNum2=Num2;
while(MidNum1)
{
++count1;
MidNum1/=10;
}
while(MidNum2)
{
++count2;
MidNum2/=10;
}
inta=Num1*pow(10,count2)+Num2;
intb=Num2*pow(10,count1)+Num1;
return(a>b)?true:false;
}</pre>
<pre></pre>
<p>好了,我们的比较准则函数也已经完成了,只需要把这个比较准则加到关键的地方,这个题就算完成了,最终代码如下:</p>
<preclass="cpp"name="code">#include<iostream>
#include<cmath>
usingnamespacestd;
boolcompare(intNum1,intNum2);
intmain(intargc,char*argv[])
{
intN;
cout<<"pleaseenterthenumbern:"<<endl;
cin>>N;
int*array=newint
;
for(inti=0;i<N;i++)
cin>>array[i];
inttemp;
for(i=0;i<=N-1;++i)
{
for(intj=0;j<N-i-1;j++)
if(compare(array[j],array[j+1]))
{
temp=array[j];
array[j]=array[j+1];
array[j+1]=temp;
}
}
cout<<"themaxnumberis:";
for(i=N-1;i>=0;--i)
cout<<array[i];
cout<<endl;
delete[]array;
system("pause");
return0;
}
boolcompare(intNum1,intNum2)
{
intcount1=0,count2=0;
intMidNum1=Num1,MidNum2=Num2;
while(MidNum1)
{
++count1;
MidNum1/=10;
}
while(MidNum2)
{
++count2;
MidNum2/=10;
}
inta=Num1*pow(10,count2)+Num2;
intb=Num2*pow(10,count1)+Num1;
return(a>b)?true:false;
}</pre>
<p><br>
</p>
<p>运行测试:</p>
<p><imgalt=""src="http://img.my.csdn.net/uploads/201303/31/1364663151_2496.jpg"></p>
<p>可以看见这样很巧妙的改编冒泡排序就解决了这个问题,当然也可以用其他的排序算法,或者干脆用一个仿函数作为set容器的排序准则,insert进去就可以了,但是这个程序有一点小问题,假如输入的数字中有两个濒临越界的数据,合并在一起就越界了,那样就只能用字符串的形式进行比较!</p>
<p></p>
<p><spanstyle="color:#ff0000">4,:找零钱的问题</span></p>
<p>在贪心算法里面最常见的莫过于找零钱的问题了,题目大意如下,对于人民币的面值有1元5元10元20元50元100元,下面要求设计一个程序,输入找零的钱,输出找钱方案中最少张数的方案,比如123元,最少是1张100的,1张20的,3张1元的,一共5张!</p>
<p>解析:这样的题目运用的贪心策略是每次选择最大的钱,如果最后超过了,再选择次大的面值,然后次次大的面值,一直到最后与找的钱相等,这种情况大家再熟悉不过了,下面就直接看源代码:</p>
<p></p>
<preclass="cpp"name="code">#include<iostream>
#include<cmath>
usingnamespacestd;
intmain(intargc,char*argv[])
{
intMoneyClass[6]={100,50,20,10,5,1};//记录钱的面值
intMoneyIndex[6]={0};//记录每种面值的数量
intMoneyAll,MoneyCount=0,count=0;
cout<<"pleaseentertheallmoneyyouwanttoexchange:"<<endl;
cin>>MoneyAll;
for(inti=0;i<6;)//只有这个循环才是主体
{
if(MoneyCount+MoneyClass[i]>MoneyAll)
{
i++;
continue;
}
MoneyCount+=MoneyClass[i];
++MoneyIndex[i];
++count;
if(MoneyCount==MoneyAll)
break;
}
for(i=0;i<6;++i)//控制输出的循环
{
if(MoneyIndex[i]!=0)
{
switch(i)
{
case0:
cout<<"the100have:"<<MoneyIndex[i]<<endl;
break;
case1:
cout<<"the50have:"<<MoneyIndex[i]<<endl;
break;
case2:
cout<<"the20have:"<<MoneyIndex[i]<<endl;
break;
case3:
cout<<"the10have:"<<MoneyIndex[i]<<endl;
break;
case4:
cout<<"the5have:"<<MoneyIndex[i]<<endl;
break;
case5:
cout<<"the1have:"<<MoneyIndex[i]<<endl;
break;
}
}
}
cout<<"thetotalmoneyhave:"<<count<<endl;
system("pause");
return0;
}</pre><preclass="cpp"name="code"></pre><preclass="cpp"name="code"></pre>
<p><spanstyle="font-size:24px;color:#ff0000">由于精力有限,贪心算法的很多题还没写,以后有时间会补上,其实最主要就是记住贪心策略,每次选择的都是对于当前而言最优的,贪心思想不难,利用好就需要多练习,望一起进步!(下一算法:动态规划算法)</span></p>
<pre></pre>
<pre></pre>
<pre></pre>
<linkrel="stylesheet"href="http://static.blog.csdn.net/public/res-min/markdown_views.css?v=1.0">
下面我要说的是,这个算法里面就是采用的贪心第三方案,一般这个方案是成功率最大的,其他两个方案我在这里没有考虑,在这里得到的结果是利用了115容量装了价值195的东西,但是这明显不是最优结果,分明还可以装一个A进去!刚好满足150重量,由于在算法中我单纯的利用第三种贪心方法求解,当剩余的包裹中最优的再加进来的时候已经超过了,所以这个时候可以选择剩余包裹中次优的(如这里选择A),再不行就次次优的,尽量把包裹装满,这样得到的结果就很接近了(不保证一定为最优),但是我们一般不这样来求解,下一文章会介绍动态规划算法来解决这个问题,动态规划很好的弥补了贪心算法的不足!详见下一章!!
还需要说明的是,如果包裹是可以拆分的,那这个问题就得到了整体最优解,前面不变,就是当最后一次装进去已经超过容量的时候可以选择只装她的一部分!很多编程题一般是这种情况!经自己总结的贪心算法几大经典问题:1:活动时间安排的问题设有N个活动时间集合,每个活动都要使用同一个资源,比如说会议场,而且同一时间内只能有一个活动使用,每个活动都有一个使用活动的开始si和结束时间fi,即他的使用区间为(si,fi),现在要求你分配活动占用时间表,即哪些活动占用该会议室,哪些不占用,使得他们不冲突,要求是尽可能多的使参加的活动最大化,即所占时间区间最大化!上图为每个活动的开始和结束时间,我们的任务就是设计程序输出哪些活动可以占用会议室![cpp]
viewplaincopy#includeusingnamespacestd;voidGreedyChoose(intlen,int*s,int*f,bool*flag);intmain(intargc,char*argv[]){ints[11]={1,3,0,5,3,5,6,8,8,2,12};intf[11]={4,5,6,7,8,9,10,11,12,13,14};boolmark[11]={0};GreedyChoose(11,s,f,mark);
for(inti=0;i<11;i++)if(mark[i])cout<<i<<"";system("pause");return0;}voidGreedyChoose(intlen,int*s,int*f,bool*flag){flag[0]=true;intj=0;for(inti=1;i=f[j]){flag[i]=true;j=i;}}得出结果是03710,也就是对应的时间段值得说明一下,虽然贪心算法不是一定可以得到最好的解
,但是对于这种活动时间的问题,他却得到的总是最优解,这点可以用数学归纳法证明,在这里,体现出来的贪心策略是:每一个活动时间的挑选总是选择最优的,就是刚好匹配的,这样得出的结果也就是最优的了!由于这个算法很简单,在这里就没有注释了!类似这种题还有个区间覆盖问题,就是说很多个区间,其中有些是相互覆盖着的,要求去除多余的区间,使剩下的区间占用长度最大,实际就是这个题,只是问法变换了而已!接下来让我们看线性覆盖的问题,跟上面的相反!2.贪心实例之线段覆盖(linescover)题目大意:在一维空间中告诉你N条线段的起始坐标与终止坐标,要求求出这些线段一共覆盖了多大的长度。为了方便说明,我们采用上述表格中的数据代表10条线段的起始点和终点,注意,这里是用起始点为顺序进行排列,和上面的不一样,知道了这些我们就可以着手开始设计这个程序:
[cpp]viewplaincopy#includeusingnamespacestd;intmain(intargc,char*argv[]){ints[10]={2,3,4,5,6,7,8,9,10,11};intf[10]={3,5,7,6,9,8,12,10,13,15};intTotalLength=(3-2);for(inti=1,intj=0;i<10i=""if=""s=""i="">=f[j]){TotalLength
+=(f[i]-s[i]);j=i;}else{if(f[i]<=f[j])continue;else{TotalLength+=f[i]-f[j];j=i;}}}cout<<TotalLength<<endl;system("pause");return0;}运行结果为13,显然这是我们需要的结果,这里注明一下,上面图表中数据有点问题,实际以程序中给出的为主!3,:数字组合问题!设有N个正整数,现在需要你设计一个程序,使他们连接在一起成为最大的数字,例3个整数
12,456,342很明显是45634212为最大,4个整数342,45,7,98显然为98745342最大程序要求:输入整数N接下来一行输入N个数字,最后一行输出最大的那个数字!题目解析:拿到这题目,看起要来也简单,看起来也难,简单在什么地方,简单在好像就是寻找哪个开头最大,然后连在一起就是了,难在如果N大了,假如几千几万,好像就不是那么回事了,要解答这个题目需要选对合适的贪心策略,并不是把数字由大排到小那么简单,网上的解法是将数字转化为字符串,比如a+b和b+a,用strcmp函数比较一下就知道谁大,也就知道了谁该排在谁前面,不过我觉得这个完全没必要,在这里我采用一种比较巧妙的方法来解答,不知道大家还记得冒泡排序法不,那是排序最早接触的一种方法,我们先看看它的源代码:[cpp]
viewplaincopy#includeusingnamespacestd;intmain(intargc,char*argv[]){intarray[10];for(inti=0;i<10i=""cin="">>array[i];inttemp;for(i=0;i<=9;++i)for(intj=0;j<10-1-ij=""if=""array=""j="">array[j+1]){temp=array[j];array[j]=
array[j+1];array[j+1]=temp;}for(i=0;i<10;i++)cout<<array[i]<<"";cout<array[j+1])他是以数字大小作为比较准则来返回true或者是false,那么我们完全可以改变一下这个排序准则,比如23,123,这两个数字,在我们这个题中它可以组成两个数字23123和12323,分明是前者大些,所以我们可以说23排在123前面,也就是23的优先级比123大,123的优先级比23小,所以不妨写个函数,传递参数a和b,如果ab比ba大,则返回true,反之返回false,函数原型如下:
[cpp]viewplaincopy[cpp]viewplaincopy
boolcompare(intNum1,intNum2) { intcount1,count2; intMidNum1=Num1,MidNum2=Num2; while(MidNum1) { ++count1; MidNum1/=10; } while(MidNum2) { ++count2; MidNum2/=10; } inta=Num1*pow(10,count2)+Num2; intb=Num2*pow(10,count1)+Num1; return(a>b)?true:false; }
好了,我们的比较准则函数也已经完成了,只需要把这个比较准则加到关键的地方,这个题就算完成了,最终代码如下:
#include
#include
usingnamespacestd;
boolcompare(intNum1,intNum2);
intmain(intargc,char*argv[])
{
intN;
cout<<"pleaseenterthenumbern:"<>N;
int*array=newint
;
for(inti=0;i>array[i];
inttemp;
for(i=0;i<=N-1;++i)
{
for(intj=0;j<N-i-1;j++)
if(compare(array[j],array[j+1]))
{
temp=array[j];
array[j]=array[j+1];
array[j+1]=temp;
}
}
cout<<"themaxnumberis:";
for(i=N-1;i>=0;--i)
cout<<array[i];
cout<b)?true:false;
}
运行测试:
可以看见这样很巧妙的改编冒泡排序就解决了这个问题,当然也可以用其他的排序算法,或者干脆用一个仿函数作为set容器的排序准则,insert进去就可以了,但是这个程序有一点小问题,假如输入的数字中有两个濒临越界的数据,合并在一起就越界了,那样就只能用字符串的形式进行比较!
4,:找零钱的问题
在贪心算法里面最常见的莫过于找零钱的问题了,题目大意如下,对于人民币的面值有1元5元10元20元50元100元,下面要求设计一个程序,输入找零的钱,输出找钱方案中最少张数的方案,比如123元,最少是1张100的,1张20的,3张1元的,一共5张!
解析:这样的题目运用的贪心策略是每次选择最大的钱,如果最后超过了,再选择次大的面值,然后次次大的面值,一直到最后与找的钱相等,这种情况大家再熟悉不过了,下面就直接看源代码:
#include
#include
usingnamespacestd;
intmain(intargc,char*argv[])
{
intMoneyClass[6]={100,50,20,10,5,1};//记录钱的面值
intMoneyIndex[6]={0};//记录每种面值的数量
intMoneyAll,MoneyCount=0,count=0;
cout<<"pleaseentertheallmoneyyouwanttoexchange:"<>MoneyAll;
for(inti=0;i<6if=""moneycount=""moneyclass=""i="">MoneyAll)
{
i++;
continue;
}
MoneyCount+=MoneyClass[i];
++MoneyIndex[i];
++count;
if(MoneyCount==MoneyAll)
break;
}
for(i=0;i<6;++i)//控制输出的循环
{
if(MoneyIndex[i]!=0)
{
switch(i)
{
case0:
cout<<"the100have:"<<MoneyIndex[i]<<endl;
break;
case1:
cout<<"the50have:"<<MoneyIndex[i]<<endl;
break;
case2:
cout<<"the20have:"<<MoneyIndex[i]<<endl;
break;
case3:
cout<<"the10have:"<<MoneyIndex[i]<<endl;
break;
case4:
cout<<"the5have:"<<MoneyIndex[i]<<endl;
break;
case5:
cout<<"the1have:"<<MoneyIndex[i]<<endl;
break;
}
}
}
cout<<"thetotalmoneyhave:"<<count<<endl;
system("pause");
return0;
}
由于精力有限,贪心算法的很多题还没写,以后有时间会补上,其实最主要就是记住贪心策略,每次选择的都是对于当前而言最优的,贪心思想不难,利用好就需要多练习,望一起进步!(下一算法:动态规划算法)
boolcompare(intNum1,intNum2) { intcount1,count2; intMidNum1=Num1,MidNum2=Num2; while(MidNum1) { ++count1; MidNum1/=10; } while(MidNum2) { ++count2; MidNum2/=10; } inta=Num1*pow(10,count2)+Num2; intb=Num2*pow(10,count1)+Num1; return(a>b)?true:false; }
好了,我们的比较准则函数也已经完成了,只需要把这个比较准则加到关键的地方,这个题就算完成了,最终代码如下:
#include
#include
usingnamespacestd;
boolcompare(intNum1,intNum2);
intmain(intargc,char*argv[])
{
intN;
cout<<"pleaseenterthenumbern:"<>N;
int*array=newint
;
for(inti=0;i>array[i];
inttemp;
for(i=0;i<=N-1;++i)
{
for(intj=0;j<N-i-1;j++)
if(compare(array[j],array[j+1]))
{
temp=array[j];
array[j]=array[j+1];
array[j+1]=temp;
}
}
cout<<"themaxnumberis:";
for(i=N-1;i>=0;--i)
cout<<array[i];
cout<b)?true:false;
}
运行测试:
可以看见这样很巧妙的改编冒泡排序就解决了这个问题,当然也可以用其他的排序算法,或者干脆用一个仿函数作为set容器的排序准则,insert进去就可以了,但是这个程序有一点小问题,假如输入的数字中有两个濒临越界的数据,合并在一起就越界了,那样就只能用字符串的形式进行比较!
4,:找零钱的问题
在贪心算法里面最常见的莫过于找零钱的问题了,题目大意如下,对于人民币的面值有1元5元10元20元50元100元,下面要求设计一个程序,输入找零的钱,输出找钱方案中最少张数的方案,比如123元,最少是1张100的,1张20的,3张1元的,一共5张!
解析:这样的题目运用的贪心策略是每次选择最大的钱,如果最后超过了,再选择次大的面值,然后次次大的面值,一直到最后与找的钱相等,这种情况大家再熟悉不过了,下面就直接看源代码:
#include
#include
usingnamespacestd;
intmain(intargc,char*argv[])
{
intMoneyClass[6]={100,50,20,10,5,1};//记录钱的面值
intMoneyIndex[6]={0};//记录每种面值的数量
intMoneyAll,MoneyCount=0,count=0;
cout<<"pleaseentertheallmoneyyouwanttoexchange:"<>MoneyAll;
for(inti=0;i<6if=""moneycount=""moneyclass=""i="">MoneyAll)
{
i++;
continue;
}
MoneyCount+=MoneyClass[i];
++MoneyIndex[i];
++count;
if(MoneyCount==MoneyAll)
break;
}
for(i=0;i<6;++i)//控制输出的循环
{
if(MoneyIndex[i]!=0)
{
switch(i)
{
case0:
cout<<"the100have:"<<MoneyIndex[i]<<endl;
break;
case1:
cout<<"the50have:"<<MoneyIndex[i]<<endl;
break;
case2:
cout<<"the20have:"<<MoneyIndex[i]<<endl;
break;
case3:
cout<<"the10have:"<<MoneyIndex[i]<<endl;
break;
case4:
cout<<"the5have:"<<MoneyIndex[i]<<endl;
break;
case5:
cout<<"the1have:"<<MoneyIndex[i]<<endl;
break;
}
}
}
cout<<"thetotalmoneyhave:"<<count<<endl;
system("pause");
return0;
}
由于精力有限,贪心算法的很多题还没写,以后有时间会补上,其实最主要就是记住贪心策略,每次选择的都是对于当前而言最优的,贪心思想不难,利用好就需要多练习,望一起进步!(下一算法:动态规划算法)
相关文章推荐
- 二分法的算法实现(c++)以及具体实例
- VC6.0如何创建以及调用动态链接库具体实例
- 二叉树的各种遍历算法以及实例
- GBDT算法原理以及实例理解
- 也学AES加密算法主要步骤以及具体实例
- docker安装lnmp具体实例以及常用的docker命令
- Linux(centos 6.5) 调用java脚本以及定时运行的脚本实例及配置文件具体解释
- c++实现加密和解密算法以及JNI技术的应用实例
- Git---通过一个具体实例详解Git常用命令以及提交pr(pull request)
- 关于c的4种算法以及实例演示
- DEM山体阴影原理以及算法具体解释
- Java 泛型类 以及 泛型类获取子类的具体参数类型 以及 获取子类型具体参数的实例
- 全面分析iterator(迭代器)模式,以及具体java实例
- 【机器学习】k-近邻算法以及算法实例
- DEM山体阴影原理以及算法具体解释
- 【机器学习】k-近邻算法以及算法实例
- Linear regression with multiple variables(多特征的线型回归)算法实例_梯度下降解法(Gradient DesentMulti)以及正规方程解法(Normal Equation)
- VC6.0如何创建以及调用动态链接库具体实例
- spark ml 算法原理剖析以及具体的源码实现分析
- VC6.0如何创建以及调用动态链接库具体实例