插入排序还是归并排序

1.题目

插入排序还是归并排序
分数 25

全屏浏览

切换布局
作者 陈越
单位 浙江大学
根据维基百科的定义:

插入排序是迭代算法,逐一获得输入数据,逐步产生有序的输出序列。每步迭代中,算法从输入序列中取出一元素,将之插入有序序列中正确的位置。如此迭代直到全部元素有序。

归并排序进行如下迭代操作:首先将原始序列看成 N 个只包含 1 个元素的有序子序列,然后每次迭代归并两个相邻的有序子序列,直到最后只剩下 1 个有序的序列。

现给定原始序列和由某排序算法产生的中间序列,请你判断该算法究竟是哪种排序算法?

输入格式:
输入在第一行给出正整数 N (≤100);随后一行给出原始序列的 N 个整数;最后一行给出由某排序算法产生的中间序列。这里假设排序的目标序列是升序。数字间以空格分隔。

输出格式:
首先在第 1 行中输出Insertion Sort表示插入排序、或Merge Sort表示归并排序;然后在第 2 行中输出用该排序算法再迭代一轮的结果序列。题目保证每组测试的结果是唯一的。数字间以空格分隔,且行首尾不得有多余空格。

输入样例 1:
10
3 1 2 8 7 5 9 4 6 0
1 2 3 7 8 5 9 4 6 0
输出样例 1:
Insertion Sort
1 2 3 5 7 8 9 4 6 0
输入样例 2:
10
3 1 2 8 7 5 9 4 0 6
1 3 2 8 5 7 4 9 0 6
输出样例 2:
Merge Sort
1 2 3 8 4 5 7 9 0 6

2.错误示例

       建议先了解什么是插入排序和归并排序。

        最开始我想直接对原始数组进行归并排序,每排序一次就和中间序列比较一次,如果一样就输出下一次的归并排序。如果都不一样就输出插入排序的下一次,但是有个一分的测试点没有过,于是我就对原始数组进行插入排序,重复上面操作,但还是没过,大家可以看看错哪了。

//一个测试点没过
#include<iostream>
#include<algorithm>
using namespace std;
const int MAX=1010;
int N;
int arr[MAX],mid[MAX],arr1[MAX];
bool f=true;

//比较两个数组
bool cmp(){
    for(int i=0;i<N;i++){
        if(arr[i]!=mid[i])
            return false;
    }
    return true;
}

//归并排序
void Mer(int h)
{
    int i=0;
    while(i+2*h<=N){
        sort(arr+i,arr+i+2*h);
        i=i+2*h;
    }
    if(i+h<=N){
        sort(arr+i,arr+N);
    }
}
int main()
{
    cin>>N;
    for(int i=0;i<N;i++){cin>>arr[i];arr1[i]=arr[i];}
    for(int i=0;i<N;i++){cin>>mid[i];}
    int h=1;

    //进行插入排序,并检测两个数组是否相同
    for(int i=0;i<N;i++)
    {
        if(!cmp())
            sort(arr,arr+i+1);
        else            //如果匹配,就输出插入排序下次序列
        {
            f=false;
            sort(mid,mid+i+1);
            cout<<"Insertion Sort"<<endl;
            for(int i=0;i<N;i++)
            {
                if(i) cout<<" "<<mid[i];
                else cout<<mid[i];
            }
            return 0;
        }
    }
    for(int i=0;i<N;i++)
    {
        arr[i]=arr1[i];
    }
    if(f)               //不匹配就输出归并排序下一次序列
    {
    while(h<=N){
        if(!cmp())
        {
            Mer(h);
            h*=2;
        }
        else
        {
            cout<<"Merge Sort"<<endl;
            Mer(h);
            for(int i=0;i<N;i++)
            {
                if(i) cout<<" "<<arr[i];
                else cout<<arr[i];
            }
            return 0;
        }
    }
    }
    return 0;
}

 3.重点分析

        除了上述两种方法,还可以判断原始数组和中间序列的后缀序列是否相同,来判断是插入还是归并排序,因为插入排序是从开头依次排序,因此如果中间序列是插入排序的中间序列的话,那么中间序列的后缀就和原始序列相同(从没排序的元素开始到序列末尾)。因此我们可以得到答案。

#include<iostream>
#include<algorithm>
using namespace std;
const int MAX=1010;
int N;
int arr[MAX],mid[MAX];
bool f=true;

//比较函数
bool cmp(){
    for(int i=0;i<N;i++){
        if(arr[i]!=mid[i]) return false;}
    return true;
}

//归并排序
void Mer(int h){
    int i=0;
    while(i+2*h<=N){
        sort(arr+i,arr+i+2*h);
        i=i+2*h;
    }
    if(i+h<N){
        sort(arr+i,arr+N);
    }
}
int main(){
    cin>>N;
    for(int i=0;i<N;i++){cin>>arr[i];}
    for(int i=0;i<N;i++){cin>>mid[i];}
    int h=1,id=0;
    for(int i=0;i<N-1;i++){
        if(mid[i]>mid[i+1]){      //找到中间序列第一个没排序的元素下标
            id=i+1;
            break;
        }
    }
    for(int i=id;i<N;i++){        //判断中间序列后缀和原始序列是否相同
        if(mid[i]!=arr[i])
            f=false;
    }
    if(f){                        //如果相同,是插入排序
        sort(mid,mid+id+1);
        cout<<"Insertion Sort"<<endl;
        for(int i=0;i<N;i++){
            if(i) cout<<" "<<mid[i];
            else cout<<mid[i];
        }
    }  
    else{                       //否则是归并排序
        while(h<N){
        if(!cmp()){             //找到归并的次数
            Mer(h);
            h*=2;
        }
        else{
            cout<<"Merge Sort"<<endl;
            Mer(h);
            for(int i=0;i<N;i++){
                if(i) cout<<" "<<arr[i];
                else cout<<arr[i];
            }
            return 0;
        }
    }
    }
    return 0;
}

  • 10
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值