#头歌 数据结构 实验十二 排序的实现

第1关:冒泡排序-交换类

任务描述

本关任务:实现冒泡排序算法,并将乱序数列变成升序。冒泡排序又称简单交换排序。

相关知识

为了完成本关任务,你需要掌握:1.冒泡排序算法。

冒泡排序算法

冒泡排序重复地遍历待排序的数列,每次比较两个相邻元素,如果它们的顺序错误就把它们交换。重复地进行遍历直到没有再需要交换时表示数列已经排序完成。

  • 算法步骤:
  1. 比较相邻的元素:若第一个比第二个大,则交换;
  2. 遍历开始第一对到结尾最后一对,执行步骤1
  3. 重复步骤1~`2`,直到排序完成。
  • 可改进的冒泡排序:第一趟排序之后最后一个元素是最大的,因此下一趟遍历只需执行到倒数第二对。

编程要求

本关的编程任务是补全右侧代码片段sort_arrayBeginEnd中间的代码,具体要求如下:

  • sort_array中,实现冒泡排序算法,完成指定输出。
测试说明

平台将自动编译补全后的代码,并生成若干组测试数据,接着根据程序的输出判断程序是否正确。

以下是平台的测试样例:

测试输入: 10 7 1 4 6 8 9 5 2 3 10 预期输出: 1 4 6 7 8 5 2 3 9 10 1 4 6 7 5 2 3 8 9 10 1 4 6 5 2 3 7 8 9 10 1 2 3 4 5 6 7 8 9 10

测试输入: 15 3 44 38 5 47 15 36 26 27 2 46 4 19 50 48 预期输出: 3 38 5 44 15 36 26 27 2 46 4 19 47 48 50 3 5 38 15 36 26 27 2 44 4 19 46 47 48 50 3 5 15 36 26 27 2 38 4 19 44 46 47 48 50 2 3 4 5 15 19 26 27 36 38 44 46 47 48 50


开始你的任务吧,祝你成功!

最后通关代码

//

//  sort_.cpp

//  Sort

//

//  Created by ljpc on 2018/4/20.

//  Copyright © 2018年 ljpc. All rights reserved.

//

#include "sort_.h"

void print_array(int *arr, int n)

// 打印数组

{

    if(n==0){

        printf("ERROR: Array length is ZERO\n");

        return;

    }

    printf("%d", arr[0]);

    for (int i=1; i<n; i++) {

        printf(" %d", arr[i]);

    }

    printf("\n");

}



 

void sort_array(int *arr, int n)

//  编程实现《冒泡排序算法》:将乱序序列arr转化为升序序列

//  函数参数:乱序整数数组arr 数组长度

//  要求输出:调用print_array(int *arr, int n)输出前三次冒泡操作后的序列,以及最终的升序序列

{

    // 请在这里补充代码,完成本关任务

    /********** Begin *********/

/*

    int i ,j;

    for(i=0;i<n-1;i++)//进行 n-1 次

    {

        for(j=0;j<n-i-1;j++)//每次 n-i-1 轮

            if(arr[j]>arr[j+1])

            {//前>后  大的数后移

                int temp   =arr[j];

                    arr[j]    = arr[j+1];

                    arr[j+1] = temp;

            }

        if(i<3)

            print_array(arr, n); //前三次输出  

    }

       

       print_array(arr, n);          

    */

    //优化:

    int i ,j;

    int flag ;

    for(i=0;i<n-1;i++)//进行 n-1 次

    {

        flag = 0;//交换标识

        for(j=0;j<n-i-1;j++)//每次 n-i-1 轮

            if(arr[j]>arr[j+1])

            {//前>后  大的数后移

                int temp   =arr[j];

                    arr[j]    = arr[j+1];

                    arr[j+1] = temp;

                    flag  = 1;

            }

        //-----------------------------

            if(flag==0)

                break;

            /*

                如果此时flag == 0

                说明当前轮没有发生交换

                即;此时当前数列已经"有序"

            */

        //-----------------------------

            if(i<3)

                print_array(arr, n); //前三次输出  

    }

       print_array(arr, n);  

   

    /********** End **********/

}



第2关:简单选择排序-选择类

任务描述

本关任务:实现选择排序算法,并将乱序数列变成升序。

相关知识

为了完成本关任务,你需要掌握:1.选择排序算法。

选择排序算法

选择排序是一种简单直观的排序算法,首先在未排序序列中找到最小元素,存放到排序序列的起始位置,然后,再从剩余未排序元素中继续寻找最小元素,然后放到已排序序列的末尾。以此类推,直到所有元素均排序完毕。

  • 算法步骤:
  1. 初始状态:无序序列为R[0,n−1],长度n,有序区为空;

  2. 第i=1,..,n−1趟排序从当前无序区R[i−1,n−1]中选出最小的元素R[k],并将它与无序区的第1个记录R[i−1]交换,则R[0,i−1]变为元素个数增加1的新有序区,R[i,n−1]变为元素个数减少1的新无序区;

  3. n−1趟选择交换后结束。

编程要求

本关的编程任务是补全右侧代码片段sort_arrayBeginEnd中间的代码,具体要求如下:

  • sort_array中,实现选择排序算法,完成指定输出。
测试说明

平台将自动编译补全后的代码,并生成若干组测试数据,接着根据程序的输出判断程序是否正确。

以下是平台的测试样例:

测试输入: 10 7 1 4 6 8 9 5 2 3 10 预期输出: 1 7 4 6 8 9 5 2 3 10 1 2 4 6 8 9 5 7 3 10 1 2 3 6 8 9 5 7 4 10 1 2 3 4 5 6 7 8 9 10

测试输入: 15 3 44 38 5 47 15 36 26 27 2 46 4 19 50 48 预期输出: 2 44 38 5 47 15 36 26 27 3 46 4 19 50 48 2 3 38 5 47 15 36 26 27 44 46 4 19 50 48 2 3 4 5 47 15 36 26 27 44 46 38 19 50 48 2 3 4 5 15 19 26 27 36 38 44 46 47 48 50


开始你的任务吧,祝你成功!

最后通关代码

//

//  sort_.cpp

//  Sort

//

//  Created by ljpc on 2018/4/20.

//  Copyright © 2018年 ljpc. All rights reserved.

//

#include "sort_.h"

void print_array(int *arr, int n)

// 打印数组

{

    if(n==0){

        printf("ERROR: Array length is ZERO\n");

        return;

    }

    printf("%d", arr[0]);

    for (int i=1; i<n; i++) {

        printf(" %d", arr[i]);

    }

    printf("\n");

}

void sort_array(int *arr, int n)

//  编程实现《选择排序算法》:将乱序序列arr转化为升序序列

//  函数参数:乱序整数数组(无重复元素) 数组长度

//  要求输出:调用print_array(int *arr, int n)输出前三次选择操作后的序列,以及最终的升序序列

{

    // 请在这里补充代码,完成本关任务

    /********** Begin *********/

 int i ,j,min;

     int flag = 0,temp=0;

    for(i=0;i<n-1;i++)

    {

        min = arr[i];//设定最小值

        for(j=i+1;j<n;j++)

            if(min>arr[j])

                {//如果出现更小的值,保存更小值的下标

                    min = arr[j];

                    flag = 1;

                    temp=j;

                }

       if(flag==1)//"flag==1"说明找到更小的值,即下标发生了变化 -> 交换

       {

          int t = arr[i];

          arr[i]  = arr[temp];

          arr[temp] = t;

          flag = 0 ;

       }

        if(i<3)

            print_array(arr, n); //前三次输出  

    }

       

       print_array(arr, n);    

   

   

    /********** End **********/

}



第3关:直接插入排序-插入类

任务描述

本关任务:实现插入排序算法,并将乱序数列变成升序。

相关知识

为了完成本关任务,你需要掌握:1.插入排序算法。

插入排序算法

插入排序的工作原理是通过构建有序序列,对于未排序数据,在已排序序列中从后向前扫描,找到相应位置并插入。

  • 算法步骤:
  1. 从第一个元素开始,该元素认为已经被排序;

  2. 取下一个元素,在已经排序的元素序列中从后向前扫描;

  3. 如果已排序元素大于新元素,将已排序元素移到下一位置;

  4. 重复步骤3,直到找到已排序的元素小于或者等于新元素的位置;

  5. 将新元素插入到该位置后;

  6. 重复步骤2~`5`。

编程要求

本关的编程任务是补全右侧代码片段sort_arrayBeginEnd中间的代码,具体要求如下:

  • sort_array中,实现插入排序算法,完成指定输出。
测试说明

平台将自动编译补全后的代码,并生成若干组测试数据,接着根据程序的输出判断程序是否正确。

以下是平台的测试样例:

测试输入: 10 7 1 4 6 8 9 5 2 3 10 预期输出: 1 7 4 6 8 9 5 2 3 10 1 4 7 6 8 9 5 2 3 10 1 4 6 7 8 9 5 2 3 10 1 2 3 4 5 6 7 8 9 10

测试输入: 15 3 44 38 5 47 15 36 26 27 2 46 4 19 50 48 预期输出: 3 44 38 5 47 15 36 26 27 2 46 4 19 50 48 3 38 44 5 47 15 36 26 27 2 46 4 19 50 48 3 5 38 44 47 15 36 26 27 2 46 4 19 50 48 2 3 4 5 15 19 26 27 36 38 44 46 47 48 50


开始你的任务吧,祝你成功!

最后通关代码

//

//  sort_.cpp

//  Sort

//

//  Created by ljpc on 2018/4/20.

//  Copyright © 2018年 ljpc. All rights reserved.

//

#include "sort_.h"

void print_array(int *arr, int n)

// 打印数组

{

    if(n==0){

        printf("ERROR: Array length is ZERO\n");

        return;

    }

    printf("%d", arr[0]);

    for (int i=1; i<n; i++) {

        printf(" %d", arr[i]);

    }

    printf("\n");

}

void sort_array(int *arr, int n)

//  编程实现《插入排序算法》:将乱序序列arr转化为升序序列

//  函数参数:乱序整数数组(无重复元素) 数组长度

//  要求输出:调用print_array(int *arr, int n)输出前三次插入操作后的序列,以及最终的升序序列

{

    // 请在这里补充代码,完成本关任务

    /********** Begin *********/

int i ,j ,temp;

    for(i =1;i<n;i++)

    {

      if(arr[i]<arr[i-1])

      {

        temp = arr[i];

        for(j=i-1;temp<arr[j]&&j!=-1;j--)//注意此处的  !=-1          

             arr[j+1] = arr[j];

             

             

        arr[j+1] =temp;

      }

      if(i<4)

        print_array(arr, n);

    }

     print_array(arr, n);  

     

   

    /********** End **********/

}



 

第4关:希尔排序-插入类

任务描述

本关任务:实现希尔排序算法,并将乱序数列变成升序。

相关知识

为了完成本关任务,你需要掌握:1.希尔排序算法。

希尔排序算法

希尔排序由Shell在1959年发明,又叫缩小增量排序,是第一个突破O(n2)的排序算法,属于简单插入排序的改进版,会优先比较距离较远的元素。

  • 算法步骤:
  1. 选择一个增量序列T1​,T2​,… ,Tk​,其中Ti​>Tj​,Tk​=1,i>j;

  2. 每趟排序,根据对应的增量Ti​,将待排序列分割成若干子序列,分别对各子序列进行直接插入排序;

  3. 按增量序列个数k,对序列进行k趟排序。

  • 希尔排序实例: 下图的增量序列为:521,第一趟排序将增量为5的子序列进行插入排序,第二趟排序将增量为2的子序列进行插入排序,第三趟将增量为1的子序列进行插入排序,最终完成排序。

  • 希尔排序的核心在于增量序列的设定:

既可以提前设定好增量序列,也可以动态的定义增量序列。例如序列长度为n,则动态增量为:147...3x+1<n/3

编程要求

本关的编程任务是补全右侧代码片段sort_arrayBeginEnd中间的代码,具体要求如下:

  • sort_array中,使用增量序列[5, 2, 1]实现希尔排序算法,完成指定输出。
测试说明

平台将自动编译补全后的代码,并生成若干组测试数据,接着根据程序的输出判断程序是否正确。

以下是平台的测试样例:

测试输入: 8 6 10 5 2 4 9 1 7 预期输出: 6 1 5 2 4 9 10 7 4 1 5 2 6 7 10 9 1 2 4 5 6 7 9 10 1 2 4 5 6 7 9 10

测试输入: 10 7 1 4 6 8 9 5 2 3 10 预期输出: 7 1 2 3 8 9 5 4 6 10 2 1 5 3 6 4 7 9 8 10 1 2 3 4 5 6 7 8 9 10 1 2 3 4 5 6 7 8 9 10


开始你的任务吧,祝你成功!

最后通关代码

//

//  sort_.cpp

//  Sort

//

//  Created by ljpc on 2018/4/20.

//  Copyright © 2018年 ljpc. All rights reserved.

//

#include "sort_.h"

void print_array(int *arr, int n)

// 打印数组

{

    if(n==0){

        printf("ERROR: Array length is ZERO\n");

        return;

    }

    printf("%d", arr[0]);

    for (int i=1; i<n; i++) {

        printf(" %d", arr[i]);

    }

    printf("\n");

}

void sort_array(int *arr, int n)

//  编程实现《希尔排序算法》:将乱序序列arr转化为升序序列

//  函数参数:乱序整数数组 数组长度

//  要求输出:调用print_array(int *arr, int n)输出三遍增量排序操作后的序列,以及最终的升序序列

{

    // 请在这里补充代码,完成本关任务

    /********** Begin *********/

int i = 0,t=0;

    int key = 0;

    int end = 0;

    int Gap[4] = {5,2,1,-1};//最后的-1为终止条件

    //如果需要跟给增量,直接在这里给

    int gap = Gap[t];//给增量

    while (gap > 0)

    {

        for (i = gap; i < n; i++)

        {

            key = arr[i];

            end = i - gap;

            while (end >= 0 && key<arr[end])

            {

                arr[end + gap] = arr[end];

                end=end-gap;

            }

            arr[end + gap] = key;

        }

       

        print_array(arr, n);

        t++;

        gap=Gap[t];

   

    }

    print_array(arr, n);  

   

    /********** End **********/

}



第5关:归并排序-归并类

任务描述

本关任务:实现归并排序算法,并将乱序数列变成升序。

相关知识

为了完成本关任务,你需要掌握:1.归并排序算法。

归并排序算法

归并排序(MERGE-SORT)是利用归并的思想实现的排序方法,是采用分治法Divide and Conquer的一个非常典型的应用。分Divide:将问题分成一些小的问题然后递归求解;治Conquer:将分的阶段得到的各答案合并在一起。

  • 算法步骤:
  1. 把长度为n的输入序列分成两个长度为n/2的子序列;

  2. 对这两个子序列分别采用归并排序;

  3. 将两个排序好的子序列合并成一个最终的排序序列。

编程要求

本关的编程任务是补全右侧代码片段merge_arraymerge_sortBeginEnd中间的代码,具体要求如下:

  • merge_array中,实现两个有序数组arr1arr2合并。
  • merge_sort中,实现归并排序:自上而下的递归方法。
测试说明

平台将自动编译补全后的代码,并生成若干组测试数据,接着根据程序的输出判断程序是否正确。

以下是平台的测试样例:

测试输入: 10 7 1 4 6 8 9 5 2 3 10 预期输出: 1 2 3 4 5 6 7 8 9 10

测试输入: 15 3 44 38 5 47 15 36 26 27 2 46 4 19 50 48 预期输出: 2 3 4 5 15 19 26 27 36 38 44 46 47 48 50


开始你的任务吧,祝你成功!

最后通关代码:

//

//  sort_.cpp

//  Sort

//

//  Created by ljpc on 2018/4/20.

//  Copyright © 2018年 ljpc. All rights reserved.

//

#include "sort_.h"

void print_array(int *arr, int n)

// 打印数组

{

    if(n==0){

        printf("ERROR: Array length is ZERO\n");

        return;

    }

    printf("%d", arr[0]);

    for (int i=1; i<n; i++) {

        printf(" %d", arr[i]);

    }

    printf("\n");

}

int* merge_array(int *arr1, int n1, int* arr2, int n2)

//  编程实现两个有序数组arr1和arr2合并

//  函数参数:有序数组arr1 数组arr1长度 有序数组arr2 数组arr2长度

//  函数返回值:返回从小到大排序后的合并数组

{

    // 请在这里补充代码,完成本关任务

    /********** Begin *********/

//定义

    int arr3_length =  n1 + n2;

    int *arr3 = (int *)malloc(sizeof(int)*arr3_length);//注解①

     

    int i=0 , j=0,t=0;

    //比较两个顺序表中的值,并将小的值放入LC中,当其中一个表比较完成后

    while(i<n1&&j<n2)

        {

            if(arr1[i]<=arr2[j])

                arr3[t++] = arr1[i++];

            else

                arr3[t++] = arr2[j++];  

        }


 

    //直接将另一个未比较完成的表中的剩余值全部放到LC中

    if(i==n1)

        {

            while(t<arr3_length)

                arr3[t++] = arr2[j++];  

        }

    else

        {

            while(t<arr3_length)

            arr3[t++] = arr1[i++];

        }

    return arr3;

/*

    注解①:

        1.此处必须使用malloc函数动态分配内存

          若使用int arr3[arr3_length]的方式

          分配内存,则当该函数结束,此空间已经

          归并好的数据将被程序释放,在另一个

          函数中接收到的将是一堆垃圾值

   

        2.通常来说动态分配内存很少失败,但

        为使程序更加健壮,可以增加以下代码

        if(arr3==NULL)

            return NULL; //动态分配空间失败

   

*/  

    /********** End **********/

}

int* merge_sort(int *arr, int n)

//  基于merge_array函数编程实现归并排序:自上而下的递归方法

//  函数参数:有序数组arr 数组arr长度

//  函数返回值:返回从小到大排序后的数组

{

    // 请在这里补充代码,完成本关任务

    /********** Begin *********/

int mid ;

    int left = 0 ,right = n;//注解①

    mid = (left+right)/2;

   

    if(right!=1)//注解②

    {

        //从mid将数组分成两部分

        int *Arr_1 = &arr[left],*Arr_2 = &arr[mid];

        //左侧

        Arr_1 = merge_sort(Arr_1,mid);

        //右侧

        Arr_2 = merge_sort(Arr_2,right-mid);

       

        //合并

        arr = merge_array(Arr_1,mid,Arr_2,right-mid);

    }

    return arr;

   

/*

    注解①:

        此处 right==n

   

    注解②:

        1.  当right==1时说明"小数组"的长度为1

            即:不可在分

       

        2.  也可使用

            if(mid!=0)

                因为:left一定等于0,且right==1

                故:mid = (0+1)/2 = 0;

           

        3.  此处若换成

            if(left<right)是错误的

            因为按理说最后一次 left = 0,right = 1,

            此时 left<right始终成立,将一直递归下去,

            不会终止          

*/

       

   

    /********** End **********/

}



 

第6关:快速排序-交换类

任务描述

本关任务:实现快速排序算法,并将乱序数列变成升序。

相关知识

为了完成本关任务,你需要掌握:1.快速排序算法。

快速排序算法

快速排序是最常用的一种排序算法,它的特点是速度快、效率高。快速排序的基本思想:选择一个关键值作为基准值。比基准值小的都在左边序列(一般是无序的),比基准值大的都在右边(一般是无序的)。一般选择序列的第一个元素作为基准值。

  • 算法步骤:
  1. 从数列中挑出一个元素,称为基准pivot

  2. 分区partition操作:比基准值小的元素放在左边,比基准值大的元素放在右边;

  3. 递归recursive:把小于基准值元素的子数列和大于基准值元素的子数列分别递归排序。

编程要求

本关的编程任务是补全右侧代码片段partition_arrayquick_sortBeginEnd中间的代码,具体要求如下:

  • partition_array中,实现数组分区:选定一个基准,左边比基准小,右边比基准大,返回基准所处位置。
  • quick_sort中,实现快速排序:自上而下的递归方法。
测试说明

平台将自动编译补全后的代码,并生成若干组测试数据,接着根据程序的输出判断程序是否正确。

以下是平台的测试样例:

测试输入: 10 7 1 4 6 8 9 5 2 3 10 预期输出: 1 2 3 4 5 6 7 8 9 10

测试输入: 15 3 44 38 5 47 15 36 26 27 2 46 4 19 50 48 预期输出: 2 3 4 5 15 19 26 27 36 38 44 46 47 48 50


开始你的任务吧,祝你成功!

最后通关代码:

//

//  sort_.cpp

//  Sort

//

//  Created by ljpc on 2018/4/20.

//  Copyright © 2018年 ljpc. All rights reserved.

//

#include "sort_.h"

void print_array(int *arr, int n)

// 打印数组

{

    if(n==0){

        printf("ERROR: Array length is ZERO\n");

        return;

    }

    printf("%d", arr[0]);

    for (int i=1; i<n; i++) {

        printf(" %d", arr[i]);

    }

    printf("\n");

}

int partition_array(int *arr ,int l,int r)

// 编程实现arr[l, r]分区:选定一个基准,左边比基准小,右边比基准大

// 返回基准所处位置

{

    // 请在这里补充代码,完成本关任务

    /********** Begin *********/

int temp = arr[l];//临时空间,存放"枢轴"

    while(l<r)//终止条件

    {        

        while((l<r)&&arr[r]>=temp)//"大"数在高位不动

            r--;//注解①

        arr[l] = arr[r];//"小"数在高位移动

        while((l<r)&&arr[l]<=temp)

            l++;

        arr[r] = arr[l];

    }

    arr[l] = temp;

    return l;

    /*

    注解①

    注意: 此处的两个内部的while循环均再次进行了判断

                low<high 判断,因为在比较的过程中,如果未发生

                值的覆盖,则low/high将一直移动,此过程中就可能

                导致 循环不在满足low<high ;

*/  

    /********** End **********/

}

int* quick_sort(int *arr, int l, int r)

//  基于partition_array函数编程实现快速排序:自上而下的递归方法

//  函数参数:有序数组arr 初始l=0,r=n-1

//  函数返回值:返回从小到大排序后的数组

{

    // 请在这里补充代码,完成本关任务

    /********** Begin *********/

int pos = partition_array(arr,l,r);

    if(l<r)

    {

        if(l<pos-1)

            arr = quick_sort(arr,l,pos-1);

        if(pos+1<r)

            arr = quick_sort(arr,pos+1,r);

    }

    return arr;  

    /********** End **********/

}

第7关:堆排序-选择类

任务描述

本关任务:实现堆排序算法,并将乱序数列变成升序。堆排序是对树型选择排序的进一步改进。

相关知识

为了完成本关任务,你需要掌握:1.堆排序算法。

堆排序算法

堆排序Heapsort是指利用堆这种数据结构所设计的一种排序算法。堆积是一个近似完全二叉树的结构,并同时满足堆积的性质:即子结点的键值或索引总是小于(或者大于)它的父节点。

  • 算法步骤:
  1. 将初始待排序关键字序列(R1,R2….Rn)构建成大顶堆,此堆为初始的无序区;

  2. 将堆顶元素R[1]与最后一个元素R[n]交换,此时得到新的无序区(R1,R2,……Rn-1)和新的有序区(Rn),且满足R[1,2…n-1]<=R[n]

  3. 由于交换后新的堆顶R[1]可能违反堆的性质,因此需要对当前无序区(R1,R2,……Rn-1)调整为新堆,然后再次将R[1]与无序区最后一个元素交换,得到新的无序区(R1,R2….Rn-2)和新的有序区(Rn-1,Rn)。不断重复此过程直到有序区的元素个数为n-1,则整个排序过程完成。

编程要求

本关的编程任务是补全右侧代码片段adjustHeapheap_sortBeginEnd中间的代码,具体要求如下:

  • adjustHeap中,实现堆的调整。
  • heap_sort中,构建大顶堆,并调用adjustHeap实现堆的调整。
测试说明

平台将自动编译补全后的代码,并生成若干组测试数据,接着根据程序的输出判断程序是否正确。

以下是平台的测试样例:

测试输入: 10 7 1 4 6 8 9 5 2 3 10 预期输出: 1 2 3 4 5 6 7 8 9 10

测试输入: 15 3 44 38 5 47 15 36 26 27 2 46 4 19 50 48 预期输出: 2 3 4 5 15 19 26 27 36 38 44 46 47 48 50


开始你的任务吧,祝你成功!

最后通关代码:

//

//  sort_.cpp

//  Sort

//

//  Created by ljpc on 2018/4/20.

//  Copyright © 2018年 ljpc. All rights reserved.

//

#include "sort_.h"

void print_array(int *arr, int n)

// 打印数组

{

    if(n==0){

        printf("ERROR: Array length is ZERO\n");

        return;

    }

    printf("%d", arr[0]);

    for (int i=1; i<n; i++) {

        printf(" %d", arr[i]);

    }

    printf("\n");

}

void adjustHeap(int *arr, int param1, int j)

// 编程实现堆的调整

{

    // 请在这里补充代码,完成本关任务

    /********** Begin *********/

int temp;

    if(j<param1)

    {

        int max=j;//根结点

        int s1=2*j+1;//左子节点

        int s2=2*j+2;//右子结点

        //找出最大结点

        if(arr[s1]>arr[max]&&s1<param1)

            max=s1;

        if(arr[s2]>arr[max]&&s2<param1)

            max=s2;

        //交换最大子节点到根结点并做递归

        if(max!=j)

        {

           

            temp     = arr[max];

            arr[max] = arr[j];

            arr[j]   = temp;

           

            adjustHeap(arr,param1,max);    

        }

    }  

    /********** End **********/

}

int* heap_sort(int *arr, int n)

//  基于adjustHeap函数编程实现堆排序

//  函数参数:无序数组arr 数组长度n

//  函数返回值:返回从小到大排序后的数组

{

    // 请在这里补充代码,完成本关任务

    /********** Begin *********/

//创建初始堆

    int i,temp;

    int last=n-1;               //最后一个子结点位置

    int parent=(last-1)/2;      //最后一个子结点的父结点

    for(int i=parent;i>=0;i--)  

    {

        adjustHeap(arr,n,i);        //从最后一个父结点开始做最大堆调整

    }

    for(int i=n-1;i>=0;i--)//依次将最大堆的根结点(最大值)取出

    {

    //将最大堆的根(最大值)换到最后

        temp   = arr[i];

        arr[i] = arr[0];

        arr[0] = temp;

    //除去最大值,对交换后的二叉树做最大堆调整,使二叉树根结点始终为最大值    

        adjustHeap(arr,i,0);        

    }

    return arr;


 

    /********** End **********/

}


第8关:基数排序-分配类

任务描述

本关任务:实现基数排序算法,并将乱序数列变成升序。

相关知识

为了完成本关任务,你需要掌握:1.基数排序算法。

基数排序算法

基数排序是按照低位先排序,然后收集;再按照高位排序,然后再收集;依次类推,直到最高位。有时候有些属性是有优先级顺序的,先按低优先级排序,再按高优先级排序。最后的次序就是高优先级高的在前,高优先级相同的低优先级高的在前。

  • 算法步骤:
  1. 取得数组中的最大数,并取得位数;

  2. arr为原始数组,从最低位开始取每个位组成radix数组;

  3. radix进行计数排序(利用计数排序适用于小范围数的特点);

编程要求

本关的编程任务是补全右侧代码片段sort_arrayBeginEnd中间的代码,具体要求如下:

  • sort_array中,实现基数排序算法,并返回升序的数组。
测试说明

平台将自动编译补全后的代码,并生成若干组测试数据,接着根据程序的输出判断程序是否正确。

以下是平台的测试样例:

测试输入: 10 7 1 4 6 8 9 5 2 3 10 预期输出: 1 2 3 4 5 6 7 8 9 10

测试输入: 15 3 44 38 5 47 15 36 26 27 2 46 4 19 50 48 预期输出: 2 3 4 5 15 19 26 27 36 38 44 46 47 48 50


开始你的任务吧,祝你成功!

最后通关代码:

//

//  sort_.cpp

//  Sort

//

//  Created by ljpc on 2018/4/20.

//  Copyright © 2018年 ljpc. All rights reserved.

//

#include "sort_.h"

void print_array(int *arr, int n)

// 打印数组

{

    if(n==0){

        printf("ERROR: Array length is ZERO\n");

        return;

    }

    printf("%d", arr[0]);

    for (int i=1; i<n; i++) {

        printf(" %d", arr[i]);

    }

    printf("\n");

}


 

int* sort_array(int *arr, int n)

//  编程实现《基数排序算法》

//  函数参数:乱序整数数组 数组长度

//  函数返回值:返回从小到大排序后的数组

{

    // 请在这里补充代码,完成本关任务

    /********** Begin *********/

int MAX =arr[0],MIN=arr[0], t =0 ;

    int i ;

    //寻找数组中最大最小值

    for(i=0;i<n;i++)

        {

            if(arr[i]>=MAX)

                    MAX = arr[i];      

             if(arr[i]<=MIN)

                    MIN = arr[i];

        }

       

    //生成统计数组,并清空数据

    int Arr_2[MAX] ;

    int  j=0;

    while(j<=MAX)

    {

            Arr_2[j] =0;

            j++;

    }

    for( i=0;i<n;i++)

         Arr_2[arr[i]]++;//统计

         

    for( i = MIN;i<=MAX;i++)

        {

        //   int temp = Arr_2[i];//保存arr数组中每个数出现的次数

           while(Arr_2[i]!=0)

            {

                arr[t] = i;//还原数组

                t++;

                Arr_2[i]--;

            }

            //if(temp!=0)

           //   printf("%d %d\n",i,temp);

         

        }

   

    return arr;//输出

   

    /********** End **********/

}



 

  • 22
    点赞
  • 28
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
南邮数据结构实验中,我们实现了各种内排序算法,并进行了性能比较。为了更直观地展示比较结果,我们使用了Excel进行数据记录和可视化分析。 首先,我们按照实验要求,分别实现了冒泡排序、插入排序、选择排序、快速排序、堆排序和归并排序等多种内排序算法。每种算法在相同的输入数据集上进行了测试,并记录了它们的执行时间。 接下来,我们将实验结果整理并输入Excel中。Excel的表格功能使得我们可以更直观地观察数据,并进行各种计算和比较。我们将每个算法的执行时间输入到不同的列中,每个数据集的执行时间占据一行。 在Excel中,我们可以使用各种功能和图表来进行性能比较。例如,我们可以使用线性图表来比较不同算法在不同数据集上的执行时间。通过对图表的观察,我们可以直观地了解各种算法的性能表现,并比较它们的优劣。 此外,我们还可以使用Excel的排序功能对数据进行排序,以便更好地进行比较和分析。我们可以按照执行时间的大小对算法进行排序,并观察它们的排名。通过排序和比较,我们可以更清楚地看到哪些算法在不同数据集上表现较好。 总之,使用Excel进行南邮数据结构实验中各种内排序算法的性能比较,能够使得数据更加直观、易于分析。通过Excel的表格和图表功能,我们可以对不同算法的执行时间进行比较,并找出性能较好的算法。这样的比较结果对我们在实际应用中选择合适的排序算法具有重要的参考价值。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值