分治,字面上的解释是"分而治之",就是把一个复杂的问题分成两个或更多的相同或相似的子问题,再把子问题分成更小的子问题……直到最后子问题可以简单的直接求解,原问题的解即子问题的解的合并。在计算机科学中,分治法就是运用分治思想的一种很重要的算法。分治法是很多高效算法的基础,如排序算法(快速排序,归并排序),傅立叶变换(快速傅立叶变换) 等等。
void mergex(int l,int mid,int r){
int z=l,y=mid+1;
p=l;
while(z<=mid && y<=r){
if(data[z]<=data[y]){
tmp[p++]=data[z++];
}else{
tmp[p++]=data[y++];
}
}
while(z<=mid){tmp[p++]=data[z++];}
while(y<=r){tmp[p++]=data[y++];}
for(int i=l;i<=r;i++)
data[i]=tmp[i];
return;
}
void merge(int l,int r){
if(l<r){
int mid=(l+r)>>1;
merge(l,mid); //第一个
merge(mid+1,r); //第二个
mergex(l,mid,r);
}
return;
}
这里为了帮助大家理解,我写成两个部分。
形象的说:
merge 部分是分,mergex 部分是治
这里为了表达方便,规定上面的merge为第一个,下面的为第二个(注释里也有标注)
假设我们排序下列数据:
2 1 3 5 4 6 9
那么就进入函数merge(1,7)
那么有:
于是再次进入第一个merge(1,4)
所以有:
还是可以继续,进入第一个merge(1,2)
接下来进入merge(1,1)
但是规定l
然后进入merge(2,2) 同理,也是return
接下来进入mergex(1,1,2) 了
进入mergex函数后就会多很多变量,就像酱紫
先看这一条:
while(z<=mid && y<=r){
if(data[z]<=data[y]){
tmp[p++]=data[z++];
}else{
tmp[p++]=data[y++];
}
}
发现第一个判断data[z]<=data[y] 无法通过,于是执行else 的tmp[p++]=data[y++];
所以:
接着发现while循环条件没有满足了,于是出了循环。
但是我们发现左边边那一块还都没填在tmp数组里面。
所以我们使用两个while循环分别处理剩余没填入的东西。
由于我们已经满足把小的填到左,大的填到右的原则,而且左边或者右边都已经有序,因此直接填入。
while(z<=mid){tmp[p++]=data[z++];}
while(y<=r){tmp[p++]=data[y++];}
for(int i=l;i<=r;i++)
data[i]=tmp[i];
最后,我们把临时存在tmp里的数据填入原来的data数组。
所以有:
接下来从第一个第一个merge(1,2) 出来,进入第二个merge(3,4)
然后由第二个merge(3,4) 进入第一个merge(3,3)
但是规定l
同理,进入merge(3,3) 也是一样的。
这里不多做解释。
然后接下来就是mergex(1,2,4) 了
就像酱紫:
首先判断data[z]<=data[y] ,然后发现正确(data[1]<=data[3] )。
于是执行tmp[p++]=data[z++];
结果就是酱紫:
然后再比较一轮,此时data[z]<=data[y] 仍然成立(data[2]<=data[3])
所以有:
接着发现while循环条件没有满足了,于是出了循环。
接下来笔者无需继续模拟了,请拿起你的笔自己模拟一下,如果上述过程没问题的话,我相信最终你也可以完成排序。
下面提供完整代码:
#include
#include
#include
#include
using namespace std;
#define maxm 1000001
long long data[maxm],tmp[maxm],ans,n,p;
void mergex(int l,int mid,int r){
int z=l,y=mid+1;
p=l;
while(z<=mid && y<=r){
if(data[z]<=data[y]){
tmp[p++]=data[z++];
}else{
tmp[p++]=data[y++];
}
}
while(z<=mid){tmp[p++]=data[z++];}
while(y<=r){tmp[p++]=data[y++];}
for(int i=l;i<=r;i++)
data[i]=tmp[i];
return;
}
void merge(int l,int r){
if(l<r){
int mid=(l+r)>>1;
merge(l,mid);
merge(mid+1,r);
mergex(l,mid,r);
}
return;
}
int main(){
freopen("min.txt","r",stdin);
std::ios::sync_with_stdio(false);
cin>>n;
for(int i=1;i<=n;i++)
cin>>data[i];
merge(1,n);
for(int i=1;i<=n;i++){cout<<tmp[i]<<" ";}
return 0;
}
实际上,为了让代码看起来更简单,熟练的同学可以把上下两个函数merge 和mergex 合并。
就像我下面提供的写法,推荐使用这种:
#include
#include
#include
using namespace std;
const int maxm=100005;
int n,d[maxm],tmp[maxm];
void mergesort(int l,int r){
if(l>=r) return;
int mid=(l+r)>>1;
mergesort(l,mid);
mergesort(mid+1,r);
int z=l,y=mid+1,p=l;
while(z<=mid&&y<=r){
if(d[z]<=d[y]) tmp[p++]=d[z++];
else tmp[p++]=d[y++];
}
while(z<=mid) tmp[p++]=d[z++];
while(y<=r) tmp[p++]=d[y++];
for(int i=l;i<=r;i++)
d[i]=tmp[i];
return;
}
int main(){
std::ios::sync_with_stdio(false);
cin>>n;
for(int i=1;i<=n;i++)
cin>>d[i];
mergesort(1,n);
for(int i=1;i<=n;i++)
cout<<d[i]<<" ";
return 0;
}