问题描述:
有n个选手(n为2的K次方)进行比赛,两个选手中胜者参加下一场,负者出局,请求出最后的冠军。(比赛的胜负由cmp()函数决定,这里是比较两个字符的大小)。
分析:
本体很快可以想到两种方法,分治法和减治法。
分治法:
将选手平均分为两组,递归求出胜者;
减治法:
将选手分为n/2组,两两进行比较,胜者留在比较的数组范围中,直到只有一位选手为止;
分治法代码:
#include <iostream>
using namespace std;
int cmp(char a,char b);
int G(char r[],int low,int high);
int main()
{
char a[100];
int i,n;
cin>>n;
for(i=0;i<n;i++)
{
cin>>a[i];
}
cout<<a[G(a,0,n-1)];
return 0;
}
int G(char r[],int low,int high)
{
if(high - low == 1)
{
if(cmp(r[low],r[high]))
return low;
else
return high;
}
int m=(low+high)/2;
int wl,wr;
wl = G(r,low,m);
wr = G(r,m+1,high);
if(cmp(r[wl],r[wr]))
return wl;
else
return wr;
}
int cmp(char a,char b)
{
return a>b?1:0;
}
减治法代码:
#include <iostream>
using namespace std;
int cmp(char a,char b);
char G(char r[],int n);
int main()
{
char a[100];
int i,n;
cin>>n;
for(i=0;i<n;i++)
cin>>a[i];
cout<<G(a,n)<<endl;
return 0;
}
char G(char r[],int n)
{
int i = n;
while(i>1)
{
i = i/2;
for(int j=0;j<i;j++)
{
if(cmp(r[j+i],r[j]))
r[j]=r[j+i];
}
}
return r[0];
}
int cmp(char a,char b)
{
return a>b?1:0;
}