C语言实现快速排序

我觉得冒泡排序是比较简单的;

所以今天我们实现一个叫做快速排序的;

Problem

你想要将(4,3,5,1,2)排序成(1,2,3,4,5)

你决定使用最简单的快速排序;

Solution

首先,打开你的terminal,我写得C代码通常都是用vi编辑,gcc编译;

vim quickSortSample.c

因为,快速排序需要对数据进行分割然后处理,再分割再处理;

显然需要一个递归的过程;

所以,我们先把递归结束条件写好;

#include <stdio.h>

void quick_sort( int *a, int n)
{
    int i, j, p, tmp;
    if (n < 2)  return;
}

也就是说,当输入只有一个数字就没有必要排序啦!直接返回;

void quick_sort( int *a, int n)
{
    int i, j, p, tmp;
    if (n < 2)  return;                                                                   

    p = a[n / 2];   // Get the middle element as pivot ..                                 

    for ( i = 0, j = n -1;; i++, j--) {                                                   

    //TODO ...

    }
}

注意: p = a[n / 2];的目地是获取数组中得中间位置的数据;

我们都知道,数组下标是整数,因此,如果n=3那么n/2应该是1,此时正好平分数组;

若是,n=4那么n/2应该是2,而此时它们就不完全是平分数组啦;

for循环的作用是为了从前后两个方向分别遍历;

void quick_sort( int *a, int n)
{
    int i, j, p, tmp;
    if (n < 2)  return;                                                                   

    p = a[n / 2];   // Get the middle element as pivot ..                                 

    for ( i = 0, j = n -1;; i++, j--) {
        while (a[i] < p)
            i++;
        while (p < a[j])
            j--;
    }
}

注意:我们在for循环中又进行啦while循环;

它的作用是对元素进行遍历,假设左边的数据没有p中保存的元素大,那么继续寻找,并且把i自增;

当找到比p大的或者相等得则停止第一个while循环;

例如:如果数组(2,5,3,1,4)那么p显然是3,当while循环发现5大于3时停止循环,此时i=1;

同理,第二个while循环中,p依然是3,当while循环第一次发现4大于3则继续循环此时j=4;

当while循环到j=3时,得到的元素是1,1不大于3所以停止循环,此时得到的j=3;

好啦!那么我们知道在for循环中没有明确指定循环停止条件;

它该什么时候停止呢?

void quick_sort( int *a, int n)
{
    int i, j, p, tmp;
    if (n < 2)  return;                                                                   

    p = a[n / 2];   // Get the middle element as pivot ..                                 

    for ( i = 0, j = n -1;; i++, j--) {
        while (a[i] < p)
            i++;
        while (p < a[j])
            j--;
        if ( i >= j)
            break;
    }
}

注意:当i>=j意味着此时数据已经全部遍历完了;

因为i与j分布在数组左右两部分,当它们重合,当然也就表明它们遍历完啦;

刚才,我们的两个while循环分别找到啦一个左边比p大得元素,和一个右边比p小的数字;

显然这两个元素应该交换位置的;

那么怎么实现呢;

void quick_sort( int *a, int n)
{
    int i, j, p, tmp;
    if (n < 2)  return;                                                                   

    p = a[n / 2];   // Get the middle element as pivot ..                                 

    for ( i = 0, j = n -1;; i++, j--) {
        while (a[i] < p)
            i++;
        while (p < a[j])
            j--;
        if ( i >= j)
            break;
        tmp = a[i]; a[i] = a[j]; a[j] = tmp;    //swap both ..
    }
}

我们看到了,在冒泡排序中也用到得交换语句;

此时我们完全遍历了一遍数组;

我们在main函数中测试一下;

#include <stdio.h>

void quick_sort( int *a, int n)
{
    int i, j, p, tmp;
    if (n < 2)  return;

    p = a[n / 2];   // Get the middle element as pivot ..

    for ( i = 0, j = n -1;; i++, j--) {
        while (a[i] < p)
            i++;
        while (p < a[j])
            j--;
        if ( i >= j)
            break;
        tmp = a[i]; a[i] = a[j]; a[j] = tmp;    //swap both ..
    }
}

int main(void)
{
    int a[] = { 2, 5, 3, 1, 4};
    int n = sizeof a /sizeof a[0];
    int i;
    for (i = 0; i < n; i++)
        printf("%d%s", a[i], i == n -1 ? "\n" : " ");
    quick_sort(a, n);
    for (i = 0; i < n; i++)
        printf("%d%s", a[i], i == n -1 ? "\n" : " ");

    return 0;
}

main函数的语句都非常容易懂,就不解释啦;

看看编译运行的结果:

显然,运行的结果是我们预期的;

3左边的都比3小啦!那若果数据多一些呢;

比如{ 2, 5, 7, 3, -1, 1, 4}我们会发现左边有两个元素都比3大,理论上要进行两次交换;

显然当它交换完[1]与[5]元素后,它又是怎么继续的我们关心这个问题;

显然交换完数据后,又进入for循环;

for循环没有结束条件,因此,i++,j--完成后,此时i=2,j=4;

接下来执行两个while发现,第一个while发现a[2]是7 不小于p此时不执行i++;

同样在第二个while中也发现a[4]是-1不大于p此时不执行j--;

然后到if语句不成立,因此继续进行数据交换;

测试如下数据;

#include <stdio.h>

void quick_sort( int *a, int n)
{
    int i, j, p, tmp;
    if (n < 2)  return;

    p = a[n / 2];   // Get the middle element as pivot ..

    for ( i = 0, j = n -1;; i++, j--) {
        while (a[i] < p)
            i++;
        while (p < a[j])
            j--;
        if ( i >= j)
            break;
        tmp = a[i]; a[i] = a[j]; a[j] = tmp;    //swap both ..
    }
}

int main(void)
{
    int a[] = { 2, 5, 7, 3, -1, 1, 4};
    int n = sizeof a /sizeof a[0];
    int i;
    for (i = 0; i < n; i++)
        printf("%d%s", a[i], i == n -1 ? "\n" : " ");
    quick_sort(a, n);
    for (i = 0; i < n; i++)
        printf("%d%s", a[i], i == n -1 ? "\n" : " ");

    return 0;
}

我们会得到如下结果:

显然,我们已经得到啦p左边都是小于p的;

右边则是大于p的;

符合我们的设想;

为了把整个数组正确排序,我们需要分别又对两部分重复一遍刚才的操作;

好,我们递归的调用函数,并且把数组的两部分元素传递到函数中;

void quick_sort( int *a, int n)
{
    int i, j, p, tmp;
    if (n < 2)  return;

    p = a[n / 2];   // Get the middle element as pivot ..

    for ( i = 0, j = n -1;; i++, j--) {
        while (a[i] < p)
            i++;
        while (p < a[j])
            j--;
        if ( i >= j)
            break;
        tmp = a[i]; a[i] = a[j]; a[j] = tmp;    //swap both ..
    }   

    quick_sort( a, i);
    quick_sort( a + i, n - i);
}

注意:

第一个递归调用处理数组的前部分,a对应的是数组的首地址,i是指对应的元素个数;

第二个递归调用处理数组后部分,a + i指的当然是首地址加偏移地址,如果你对偏移地址有问题的话,应该看看计算机寻址方式的资料啦;

接下来看运行结果:

最后,我们把所有代码附上:

#include <stdio.h>

void quick_sort( int *a, int n)
{
    int i, j, p, tmp;
    if (n < 2)  return;

    p = a[n / 2];   // Get the middle element as pivot ..

    for ( i = 0, j = n -1;; i++, j--) {
        while (a[i] < p)
            i++;
        while (p < a[j])
            j--;
        if ( i >= j)
            break;
        tmp = a[i]; a[i] = a[j]; a[j] = tmp;    //swap both ..
    }   

    quick_sort( a, i);
    quick_sort( a + i, n - i);
}

int main(void)
{
    int a[] = { 2, 5, 7, 3, -1, 1, 4};
    int n = sizeof a /sizeof a[0];
    int i;
    for (i = 0; i < n; i++)
        printf("%d%s", a[i], i == n -1 ? "\n" : " ");
    quick_sort(a, n);
    for (i = 0; i < n; i++)
        printf("%d%s", a[i], i == n -1 ? "\n" : " ");

    return 0;
}
时间: 2024-11-05 17:30:30

C语言实现快速排序的相关文章

C语言的快速排序及分析

一.快速排序 一般而言,学习C语言较为简单的排序,则是直接插入排序和冒泡排序.而这两者在数据较大的时候则速度就很慢了. 快速排序的速度大于前者并且较为简单,所以写下学习快速排序的过程,供以后复习. 快速排序的原理: 1.快速排序是分治思想,将数列分解排序. 2.具体过程是:先任取一个值作为基准,然后将小于该基准值的数放在该数的左侧,大于该数的数放在右侧. 3.然后就是重复地将左侧与右侧持续进行该过程,直到将其分成两个数为一组,然后比较交换.这样,就实现了排序. 具体实现: 先定义一个交换函数,方

C语言实现快速排序法(分治法)

title: 快速排序法(quick sort) tags: 分治法(divide and conquer method) grammar_cjkRuby: true --- 算法原理 分治法的基本思想:将原问题分解为若干个更小的与原问题相似的问题,然后递归解决各个子问题,最后再将各个子问题的解组合成原问题的解. 利用分治法可以将解决办法分为 "三步走" 战略: (1) 在数据集中选定一个元素作为"基准"(pivot) (2) 将所有数据集小于基准的元素放在基准左边

C语言排序---快速排序

/* 快速排序由C. A. R.Hoare在1962年提出,是对冒泡排序的一种改进. * 它的基本思想是: * 通过一趟排序将要排序的数据分割成独立的两部分 * ,其中一部分的所有数据都比另外一部分的所有数据都要小 * ,然后再按此方法对这两部分数据分别进*/ #include<stdio.h>#define N 30int QSort(int a[N],int l,int r){ int m,i,j,p; if(l>=r)return 0; /*递归出口,当序列长度小于等于0时,视为排

一起talk C栗子吧(第三十回:C语言实例--快速排序)

各位看官们,大家好,上一回中咱们说的是选择排序的例子,这一回咱们说的例子是:快速排序.闲话休 提,言归正转.让我们一起talk C栗子吧! 快速排序的原理:选定一个关键字,使容器分为两部分,一部分中的任何元素都关键字大,另一部分中的 任何元素都比关键字小.然后分别对这两个部分进行排序. 快速排序的实现步骤: 1.选定一个关键字,我在代码里直接使用容器头部元素为关键字: 2.从容器的尾部到头部遍历容器,直到遇见到比关键字小的元素,然后将它交换到关键点的左边: 3.从容器的头部到尾部遍历容器,直到遇

C语言-排序-快速排序

快速排序是对冒泡排序的一种改进.它的基本思想是:通过一趟排序将要排序的数据分割成独立的两部分,其中一部分的所有数据都比另外一部分的所有数据都要小,然后再按次方法对这两部分数据分别进行快速排序,整个排序过程可以递归进行,以此达到整个数据变成有序序列. 假设要排序的数组是A[1]……A[N],首先任意选取一个数据(通常选用第一个数据)作为关键数据,然后将所有比它的数都放到它前面,所有比它大的数都放到它后面,这个过程称为一趟快速排序.一趟快速排序的算法是: 1)设置两个变量I.J,排序开始的时候 I=

————————C语言中快速排序方法——————————————

在对浮点型排序是一定要用三木运算符(三目运算符内容下去自己看),因为如果也是用整形那样的减法的时候如果是两个十分相近的数字 可能返回一个小数(自己一会去试试),冉冉他cmp返回值是int(试试别的)因此就会将这个小数返回为0,系统认为其相等,失去了本来存在的大小关系.所以为了安全起见对浮点型进行快速排序的时候还是用三目运算符. ————————————1——————————————— 对int类型数组的排序 int a[100];//定义数组 int cmp (const void*a,cons

数据结构之---C语言实现快速排序(多个版本)

快速排序基本上有如下版本 一.国内教材双向扫描版 二.单向扫描版本 三.随机化版本 四.三数取中分割法 五.非递归版 这里给我前两个版本(较为常用) 版本一: 双向扫描版本: 如图: 代码如下: //快速排序(版本一) //带枢轴 //杨鑫 #include <stdio.h> #include <stdlib.h> #define MAXN 100 int a[MAXN + 1], n; void QuickSort(int left, int right) { int i, j

【C语言】快速排序优质算法的动态显示和字符串的快排

快速排序的动态显示比较简单,不多说,直接上代码: //快速排序动态显示 #include<stdio.h> #include <stdlib.h> #include <Windows.h> void swap(int arry[],int num1, int num2) {              int temp = arry[num1];             arry[num1] = arry[num2];             arry[num2] = t

排序算法的C语言实现-快速排序

快速排序是在实践中最快的已知排序算法,它的平均运行时间是O(NlogN),该算法之所以特别的快,主要是由于非常精炼和高度优化的内部循环.它的最坏情形的性能为N^2. 快速排序由下列简单的四步组成: 1.如果S中元素个数是0或1,则返回. 2.取S中任以元素V,称之为枢纽元 3.将S分成俩个不相交的集合,前一个元素都小于V,后一个元素都大于V. 4.返回quicksort(S1)后,加上V,再加上quicksort(S2): 如何选取枢纽元? 有一种安全的方法是随机选取枢纽元,但是产生随机数的代价