1 .问题描述
我们将乱序的红白蓝三色小球排列成有序的红白蓝三色的同颜色在一起的小球组。这个问题之所以叫荷兰国旗 是因为我们可以将红白蓝三色小球想象成条状物有序排列后正好组成荷兰国旗。
2.问题分析
这个问题我们可以将这个问题视为一个数组排序问题 这个数组分为前部 中部和后部三个部分每一个元素红白蓝分别对应0、 1 、 2必属于其中之一。 由于红、 白、蓝三色小球数量并不一定相同 所以这个三个区域不一定是等分的也就是说如果我们将整个区域放在[0, 1]的区域里 由于三色小球之间数量的比不同此处假设1 :2:2 可能前部为[0,0.2) 中部为[02,0 6) 后部为[06, 1] 。 我们的思路如下将前部和后部各排在数组的前边和后边 中部自然就排好了。具体的
设置两个标志位begin和end分别指向这个数组的开始和末尾 然后用一个标志位current从头开始进行遍历
1 若遍历到的位置为0 则说明它一定属于前部 于是就和begin位置进行交换然后current向前进 begin也向前进表示前边的已经都排好了 。
2若遍历到的位置为1 则说明它一定属于中部 根据总思路 中部的我们都不动然后current向前进。
3若遍历到的位置为2 则说明它一定属于后部 于是就和end位置进行交换 由于交换完毕后current指向的可能是属于前部的 若此时current前进则会导致该位置不能被交换到前部所以此时current不前进。而同1 end向后退1 。
//author:何佳
#include<iostream>usingnamespacestd;voidSwap(int*n1, int*n2) {inttemp;temp=*n1 ;
*n1=*n2;
*n2=temp;
}voidPrint(int*num, intlen)
{for(inti=0; i<len;++i)
{cout<<num[i]<<"";
}cout<<endl;
}
//0, 1,2, 1, 1,2,0,2, 1,0,2, 1,0,2,0, 1,2,0voidWork(int*num, intbegin, intend)
{intcur=begin;whi le(num[cur]==0)
{begin++;cur=begin;
}whi le(num[cur] !=2)
{cur++;
}while(cur!=end)
{if(num[cur]==2)
{
Swap(&num[cur] ,&num[end]) ;end--;
}if(num[cur] !=num[begin]&&num[cur] !=2) {Swap(&num[begin] ,&num[cur]) ;begin++;
}while(num[cur]==num[begin])
{cur++;
}
}if(num[end] !=2)
{
Swap(&num[cur] ,&num[begin]) ;
}
}intmain()
{intnum[]={0, 1,2, 1, 1,2,0,2, 1,0,2,0, 1,2,0, 1,2,0, 1, 1, 1,2, 1,2, 1, 1} ; int len=sizeof(num)/sizeof(int) ;
Print(num, len) ;
Work(num,0, len-1) ;
Print(num, len) ;
}
/*左飞C++数据结构与经典问题求解*/
/*荷兰国旗问题*/
/*众所周知荷兰国旗由红色、 白色和蓝色3中颜色组成现在假设有很多这3中颜色的线被存放在一个数字里 要求每次操作仅能进行一次交换 待对数字进行一遍扫描后 3中颜色自然分开颜色顺序应为红、 白、蓝。 另外 要求在O(n)的复杂度下使移动次数最小。
*/
#include<iostream>usingnamespacestd;constintN=15;intflag[N];intpre[N];intspl it1;intspl it2;intblue_red;intwhite_red;intcounts=0;
//输出结果voidPrint()
{for(inti=0;i<N;++i)
{cout<<flag[ i] ;
}cout<<endl;
}voidSwap(int&x, int&y)
{inttemp=x;x=y;y=temp;counts++;
}voidWork()
{for(inti=0;i<spl it1;++i)
{if(flag[i] !=0)
{if(blue_red>=spl it2)
{
Swap(flag[ i] ,flag[blue_red] ) ;blue_red=pre[blue_red];}else
{
Swap(flag[ i] ,flag[white_red] );white_red=pre[white_red];}
}
}intb=N-1;for(inti=spl it1;i<spl it2;++i)
{if(flag[i] !=1)
{whi le(flag[b]==2)b--;
Swap(flag[ i] ,flag[b]) ;b--;
}
}
}
//初始化voidInit( )
{intred_num=0;intwhite_num=0;intpreI=-1;for(inti=0;i<N;++i)
{flag[i ]=rand()%3;if(flag[i]==0)
{red_num++;pre[ i]=preI;preI=i;
}else if(flag[i]==1)
{white_num++;
}
}
//将国旗分成3中颜色区域
//0~spl it1-1(红),spl ite1~spl it2-1(白) spl it2~N-1(蓝)spl it1=red_num;spl it2=red_num+white_num;blue_red=preI;inti=spl it2-1;whi le(flag[i] !=0) i--;//检查白色中有没有红色white_red=i;
}intmain()
{
Init();cout<<"原始 "<<endl ;
Print( ) ;
Work();cout<<"移动 "<<counts<<"次"<<endl;Print( ) ;return0;
}
达州创梦网络怎么样,达州创梦网络公司位于四川省达州市,属于四川本地企业,资质齐全,IDC/ISP均有,从创梦网络这边租的服务器均可以备案,属于一手资源,高防机柜、大带宽、高防IP业务,一手整C IP段,四川电信,一手四川托管服务商,成都优化线路,机柜租用、服务器云服务器租用,适合建站做游戏,不须要在套CDN,全国访问快,直连省骨干,大网封UDP,无视UDP攻击,机房集群高达1.2TB,单机可提供1...
目前云服务器市场竞争是相当的大的,比如我们在年中活动中看到各大服务商都找准这个噱头的活动发布各种活动,有的甚至就是平时的活动价格,只是换一个说法而已。可见这个行业确实竞争很大,当然我们也可以看到很多主机商几个月就消失,也有看到很多个人商家捣鼓几个品牌然后忽悠一圈跑路的。当然,个人建议在选择服务商的时候尽量选择老牌商家,这样性能更为稳定一些。近期可能会准备重新整理Vultr商家的一些信息和教程。以前...
在刚才更新Vultr 新年福利文章的时候突然想到前几天有网友问到自己有在Vultr 注册账户的时候无法用支付宝付款的问题,当时有帮助他给予解决,这里正好顺带一并介绍整理出来。毕竟对于来说,虽然使用的服务器不多,但是至少是见过世面的,大大小小商家的一些特性特征还是比较清楚的。在这篇文章中,和大家分享如果我们有在Vultr新注册账户或者充值购买云服务器的时候,不支持支付宝付款的原因。毕竟我们是知道的,...