c++常见的几种排序

选择排序

找到最小的元素,并记录下它的下标,然后和第i个元素进行交换,每次进行一个循环便找到一个最小值。

比如在一个长度为N的无序数组中,在第一趟遍历N个数据,找出其中最小的数值与第一个元素交换,第二趟遍历剩下的N-1个数据,找出其中最小的数值与第二个元素交换…第N-1趟遍历剩下的2个数据,找出其中最小的数值与第N-1个元素交换,至此选择排序完成。

例如 有一个无序的数组 {50,11,85,91,20}
第一趟遍历:从第二个元素开始找比第一个元素小的元素,为11,下标为2,与第一个元素交换
11,50,85,91,20
第二趟遍历:从第三个元素开始找比第二个元素小的元素,为20,下标为4,与第二个元素交换
11,20,85,91,50
第三趟遍历:从第四个元素开始找比第三个元素 小的元素,为50,下标为4,与第三个元素交换
11,20,50,91,85
第四趟遍历:从第五个元素开始找比第四个元素小的元素,为85,下标为4,与第四个元素交换
11,20,50,85,91

代码如下:

void choiceSort(int array[],int len)
{
	for (int i = 0; i < len; i++)
	{
		int minIndex = i;
		for (int j = minIndex + 1; j < len; j++)
		{
			if (array[j] < array[minIndex ])
			{
				minIndex = j;
			}
		}
		//找到了最小的值再交换
		int temp = array[i];
		array[i] = array[minIndex ];
		array[minIndex ] = temp;
	}
}

冒泡排序

临近的数字两两比较,用两层循环来比较。这样,第一趟(外循环)完后最小或者最大的值就到了最前面
代码如下:

void maoPaoSort(int array[]int len)
{
    for (int i = 0; i < len-1; i++)
    {
	for (int j = 0; j < len-1-i; j++)
        {
	    if (array[j] > array[j+1])
            {
		int temp = array[j+1];
		array[j+1] = array[j];
		array[j] = temp;
	    }
	}
    }
}

插入排序

插入排序是一种简单直观的排序方法,他模仿了人们打牌时候对牌进行插入排序的过程。它的工作原理是从左到右逐步构建递增或者递减序列,对于未排序数据,逐步在已排序的数据中找到合适的位置插入。

需要排列的数组为 4,3,2,10,12,1,5,6

  • 第一步:因为第二个元素3比第一个元素4小,所以将元素3插入到元素4的前面。
    3,4,2,10,12,1,5,6
  • 第二步:将第三个元素2,插入到序列3,4 中。2插入第一个位置,3,4分别往后移动一位
    2,3,4,10,12,1,5,6
  • 剩下的插入步骤与第一步和第二步类似

代码如下:

void insertsort(int arr[],int n)
{
      for (int i=1;i<n;i++)
      {
         int key=arr[i];
         int j=i-1;
         while (j>=0&&arr[j]>key) //与一个的进行比较,小的话就交换
         {
              arr[j+1]=arr[j];//交换
              j--;//下标往前移动
          }
          arr[j+1]=key;//不比前一个小,就不进行交换
      }
}

快速排序(引荐一篇大牛的文章,坐在马桶上看算法:快速排序

假设我们现在对“6 1 2 7 9 3 4 5 10 8”这个10个数进行排序。首先在这个序列中随便找一个数作为基准数。为了方便,就让第一个数6作为基准数吧。接下来,需要将这个序列中所有比基准数大的数放在6的右边,比基准数小的数放在6的左边,类似下面这种排列:

3 1 2 5 4 6 9 7 10 8

在初始状态下,数字6在序列的第1位。我们的目标是将6挪到序列中间的某个位置,假设这个位置是k。现在就需要寻找这个k,并且以第k位为分界点,左边的数都小于等于6,右边的数都大于等于6。想一想,你有办法可以做到这点吗?

排序算法显神威

方法其实很简单:分别从初始序列“6 1 2 7 9 3 4 5 10 8”两端开始“探测”。先从右往左找一个小于6的数,再从左往右找一个大于6的数,然后交换他们。这里可以用两个变量i和j,分别指向序列最左边和最右边。我们为这两个变量起个好听的名字“哨兵i”和“哨兵j”。刚开始的时候让哨兵i指向序列的最左边(即i=1),指向数字6。让哨兵j指向序列的最右边(即=10),指向数字。

在这里插入图片描述

首先哨兵j开始出动。因为此处设置的基准数是最左边的数,所以需要让哨兵j先出动,这一点非常重要(请自己想一想为什么)。哨兵j一步一步地向左挪动(即j–),直到找到一个小于6的数停下来。接下来哨兵i再一步一步向右挪动(即i++),直到找到一个数大于6的数停下来。最后哨兵j停在了数字5面前,哨兵i停在了数字7面前。
在这里插入图片描述

在这里插入图片描述

现在交换哨兵i和哨兵j所指向的元素的值。交换之后的序列如下:

6 1 2 5 9 3 4 7 10 8
在这里插入图片描述

在这里插入图片描述
到此,第一次交换结束。接下来开始哨兵j继续向左挪动(再友情提醒,每次必须是哨兵j先出发)。他发现了4(比基准数6要小,满足要求)之后停了下来。哨兵i也继续向右挪动的,他发现了9(比基准数6要大,满足要求)之后停了下来。此时再次进行交换,交换之后的序列如下:

6 1 2 5 4 3 9 7 10 8

第二次交换结束,“探测”继续。哨兵j继续向左挪动,他发现了3(比基准数6要小,满足要求)之后又停了下来。哨兵i继续向右移动,糟啦!此时哨兵i和哨兵j相遇了,哨兵i和哨兵j都走到3面前。说明此时“探测”结束。我们将基准数6和3进行交换。交换之后的序列如下:

3 1 2 5 4 6 9 7 10 8

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

到此第一轮“探测”真正结束。此时以基准数6为分界点,6左边的数都小于等于6,6右边的数都大于等于6。回顾一下刚才的过程,其实哨兵j的使命就是要找小于基准数的数,而哨兵i的使命就是要找大于基准数的数,直到i和j碰头为止。

OK,解释完毕。现在基准数6已经归位,它正好处在序列的第6位。此时我们已经将原来的序列,以6为分界点拆分成了两个序列,左边的序列是“3 1 2 5 4”,右边的序列是“9 7 10 8”。接下来还需要分别处理这两个序列。因为6左边和右边的序列目前都还是很混乱的。不过不要紧,我们已经掌握了方法,接下来只要模拟刚才的方法分别处理6左边和右边的序列即可。现在先来处理6左边的序列现吧。

左边的序列是“3 1 2 5 4”。请将这个序列以3为基准数进行调整,使得3左边的数都小于等于3,3右边的数都大于等于3。好了开始动笔吧

如果你模拟的没有错,调整完毕之后的序列的顺序应该是:

2 1 3 5 4

OK,现在3已经归位。接下来需要处理3左边的序列“2 1”和右边的序列“5 4”。对序列“2 1”以2为基准数进行调整,处理完毕之后的序列为“1 2”,到此2已经归位。序列“1”只有一个数,也不需要进行任何处理。至此我们对序列“2 1”已全部处理完毕,得到序列是“1 2”。序列“5 4”的处理也仿照此方法,最后得到的序列如下:

1 2 3 4 5 6 9 7 10 8

对于序列“9 7 10 8”也模拟刚才的过程,直到不可拆分出新的子序列为止。最终将会得到这样的序列,如下

1 2 3 4 5 6 7 8 9 10

到此,排序完全结束。细心的同学可能已经发现,快速排序的每一轮处理其实就是将这一轮的基准数归位,直到所有的数都归位为止,排序就结束了。下面上个霸气的图来描述下整个算法的处理过程。
在这里插入图片描述
代码如下:

    #include <stdio.h> 
    int a[101],n;//定义全局变量,这两个变量需要在子函数中使用 
    void quicksort(int left,int right) 
    { 
        int i,j,t,temp; 
        if(left>right) 
           return; 
                                    
        temp=a[left]; //temp中存的就是基准数 
        i=left; 
        j=right; 
        while(i!=j) 
        { 
                       //顺序很重要,要先从右边开始找 
                       while(a[j]>=temp && i<j) 
                                j--; 
                       //再找右边的 
                       while(a[i]<=temp && i<j) 
                                i++; 
                       //交换两个数在数组中的位置 
                       if(i<j) 
                       { 
                                t=a[i]; 
                                a[i]=a[j]; 
                                a[j]=t; 
                       } 
        } 
        //最终将基准数归位 
        a[left]=a[i]; 
        a[i]=temp; 
                                 
        quicksort(left,i-1);//继续处理左边的,这里是一个递归的过程 
        quicksort(i+1,right);//继续处理右边的 ,这里是一个递归的过程 
    } 
    int main() 
    { 
        int i,j,t; 
        //读入数据 
        scanf("%d",&n); 
        for(i=1;i<=n;i++) 
                       scanf("%d",&a[i]); 
        quicksort(1,n); //快速排序调用 
                                 
        //输出排序后的结果 
        for(i=1;i<=n;i++) 
            printf("%d ",a[i]); 
        getchar();getchar(); 
        return 0; 
    } 
  • 13
    点赞
  • 91
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
排序是对一组数据按照某种规则进行排列的算法。在C++中,有多种排序算法可以使用。 其中几个常见排序算法包括选择排序(Selection sort)、冒泡排序(Bubble sort)、希尔排序(Shell sort)和归并排序(Merge sort)。 选择排序是一种简单直观的排序算法,它的工作原理是每次找出第i小的元素,并将这个元素与数组第i个位置上的元素交换。选择排序的时间复杂度为O(n^2),其中n是待排序数组的大小。 冒泡排序是另一种简单的排序算法,它的工作原理是通过相邻元素的比较和交换来将较大的元素逐渐“冒泡”到数组的末尾。冒泡排序的时间复杂度也是O(n^2)。 希尔排序是一种改进的插入排序算法,它通过将整个待排序序列分割成若干个子序列来进行排序,最后再对整个序列进行一次插入排序。希尔排序的时间复杂度为O(n^(3/2)),性能较好。 归并排序是一种分治策略的排序算法,它的工作原理是将待排序序列分成两个子序列,分别对子序列进行递归排序,然后再将已排序的子序列合并成一个有序序列。归并排序的时间复杂度为O(nlogn),具有稳定性。 根据您提供的代码,第一个部分是希尔排序的实现代码,第二个部分是选择排序的实现代码。您可以根据自己的需求选择其中一种排序算法来进行排序。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [C++ 数据结构与算法 (十一)(排序算法)](https://blog.csdn.net/qq_19887221/article/details/124198104)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值