C++ 初学者指南 第五篇(13)

项目5-1 快速排序
    在第四章中,我们学习了一种简单的排序方法叫做冒泡排序法。当时我们也谈到过有更好的排序方法。本章中我们就将学习一种最好的排序方法:快速排序法。快速排序法是由C.A.R.Hoare发明的。它是目前最好的通用的排序算法。在第四章中我们之所以没有使用快速排序法是因为其最好的实现需要使用到递归。本章中我们编写的快速排序算法将是针对一个字符数组进行排序,但是其排序的逻辑是可以应用于任何类型的对象的。
    快速排序的思想是基于分段的。过程如下:选择一个称之为比较数的值,然后把数组分成两部分。所有大于或者等于比较数的值放置在数组的一端;小于比较数的值放置在数组的另一端。然后对两端再分别重复上面的过程,直到整个数组变成有序的。例如,数组fedacb,比较数为d,那么第一次进行快速排序后的结果将为bcadef。然后对其两个部分bca和def分别重复上面的过程。从中可以看出这实际上是一个递归的过程。实践中最简洁的快速排序法的实现也就是使用递归。
    比较数的选择可以有两种方式:随机的选取或者是对数组中一小部分数字取其平均值。最好的排序是选择数组值中的中间值作为比较数。但是,对于大多数的数据集合来说,找出这个中间值可不容易。最差的排序是选择数是数组中的最大值或者最小值这种极端的情况。
    即使是在上面的极端情况下,快速排序法仍然能够正确的进行排序。本章中我们编写的快速排序算法是采用中间位置的元素作为比较数。
    还有一点,C++的库函数中已经含有了一个进行快速排序的函数qsort()。我们可以把它和我们下面自己编写的快速排序函数进行比较。
步骤:
 1. 创建一个名称为QSDemo.cpp的文件
 2. 快速排序将通过一对函数来实现。其中函数quicksort()为用户提供了一个方便的接口,其中调用到了实际进行排序的函数qs()。首先我们来实现quicksort()函数,如下:

    //函数实际上调用qs()函数来完成排序功能
    void quicksort(char *items, int len)
    {
        qs(items, 0, len -1);
    }


    其中,items指向需要排序的数组,len指明了该数组的大小。在下面我们会看到函数qs()需要一个初始的分段,这里在函数quicksort()中提供了这个初始的分段。这样做的好处是我们只需要调用函数quicksort()来进行排序,它只需要一个指向数组的指针参数和一个表示数组大小的参数,共计两个参数。由这个函数把数组的开始和结束索引作为参数传递给qs()函数完成排序。
3.  添加完成实际排序功能的函数qs()的实现,如下:
    

 //递归函数完成对字符数组的快速排序
     void qs(char *items, int left, int right)
     {
         int i,j;
         char x, y;
 
         i = left;
         j = right;
         x = items[(left + right) / 2];
     
         do
         {
             while(items[i] < x && ( i < right) ) i++;
             while(items[j] > x && ( j > left ) ) j--;
 
             if ( i <=j )
             {
                 y = items[i];
                 items[i] = items[j];
                 items[j] = y;
                 i++;
                 j--;
             }
 
         }while(i <= j );
 
         if ( left < j ) qs(items,left, j);
         if ( i < right ) qs(items, i,right);
     }


    调用这个函数的时候需要传入分段的边界索引。变量left表示分段的左边界。变量right表示分段的右边界。第一次调用该函数的时候,分段就是整个数组。每次递归调用的时候分段就会变得越来越小。
 4. 进行快速排序的时候,我们只需要调用函数quicksort()即可,传入数组的名称和数组的长度。当该函数返回后,数组中的元素就是有序的了。请记住,我们编写的这个快速排序函数是针对字符数组的。但是,排序的逻辑是适用于任何类型的数组的。
 5. 完整的程序如下:
   

 /* 采用快速排序法完成对字符数组的排序*/
    #include <iostream>
    #include <cstring>

    using namespace std;

    void qs(char *items, int left, int right);
    void quicksort(char *items, int len);

    int main()
    {
        char str[] ="jfmckldoelazlkper";

        cout << "Original order: " << str << "\n";

        quicksort(str, strlen(str) );

        cout << "Sorted order: " << str << "\n";

        return 0;

    }

    //递归函数完成对字符数组的快速排序
    void qs(char *items, int left, int right)
    {
        int i,j;
        char x, y;

        i = left;
        j = right;
        x = items[(left + right) / 2];

        do
        {
            while(items[i] < x && ( i < right) ) i++;
            while(items[j] > x && ( j > left ) ) j--;

            if ( i <=j )
            {
                y = items[i];
                items[i] = items[j];
                items[j] = y;
                i++;
                j--;
            }

        }while(i <= j );

        if ( left < j ) qs(items,left, j);
        if ( i < right ) qs(items, i,right);
    }

    //函数实际上调用qs()函数来完成排序功能
    void quicksort(char *items, int len)
    {
        qs(items, 0, len -1);
    }


    程序的输出如下:
    Original order: jfmckldoelazlkper
    Sorted order: acdeefjkklllmoprz
问专家:
问:我听说有“默认是int类型”的规则,它是什么意思,适用于C++吗?
答:在最早的C语言中以及早期的C++版本中,如果没有指明类型,那么类型就是整型。例如,在老的代码中,下面的函数是有效的,其返回值为整型:
f() //缺省地函数返回整型
{
     int x;
         // ...
     return x;
}
这里没有指明函数的返回值,则函数f()的返回值采用缺省的整型。然而,现在的C++是不支持这种“默认是int类型的”规则。尽管大多数的编译器都是支持对该规则的编译,以便实现向前兼容,但是我们还是应该明确指定自己所编写的每个函数的返回类型。由于老代码中依然会存在这种“默认是int类型”的写法,所以当我们在对老代码进行转换的时候需要牢记这点。

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
第 1 章:C++ 基础知识 此模块将向您介绍 C++,包括其历史、设计理念以及几个最重要的功能。此模块简要概述几个 C++ 功能,包括 C++ 程序的一般形式、一些基本控制语句和运算符。它不会介绍太多细节,而会重点介绍对所有 C++ 程序都通用的一般概念。 第 2 章:数据类型和运算符简介 编程语言的核心在于其数据类型和运算符。不出您所料,C++ 支持大量数据类型和运算符,使其适合的编程范围非常广泛。此模块对 C++ 基本数据类型及其最常用运算符进行探讨。我们还将进一步了解变量,并研究表达式。 第 3 章:程序控制语句 此模块讨论用于控制程序执行流的语句。有三种类别的程序控制语句:选择语句,包括 if 和 switch 语句;迭代语句,包括 for、while 和 do-while 循环;以及跳转语句,包括 break、continue、return 和 goto 语句。 第 4 章:数组、字符串和指针 此模块讨论数组、字符串和指针。数组是变量的集合,这些变量具有相同的类型,由一个公用名引用。数组为创建相关变量的列表提供了一种便利方法。C++ 语言不定义内置字符串数据类型。相反,字符串作为字符数组实现。指针是包含内存地址的对象。通常,指针用于访问另一个对象的值。 第 5 章:函数简介 此模块开始深度探讨函数。函数是 C++ 的构建基块,深入理解函数是成为成功 C++ 编程人员的基础。下面,您将了解如何创建函数。您还将了解传递参数、返回值、局部变量和全局变量、函数原型和递归。 第 6 章:进一步了解函数 此模块继续探讨函数。它讨论了 C++ 的三个最重要的函数相关主题:引用、函数重载和默认参数。 第 7 章:更多数据类型和运算符 此模块返回到数据类型和运算符的主题。除了您到目前为止已在使用的数据类型,C++ 还支持其他几种数据类型。其中一些数据类型由已知类型加上修饰符组成。其他数据类型包括 enumeration 和 typedef。C++ 还提供多个附加运算符,极大地扩展了 C++ 可以应用到的编程任务范围。 第 8 章:类和对象 类是 C++ 的基本封装单位。类用于创建对象。若要编写面向对象的程序,需要使用类。类和对象对于 C++ 非常重要,因此本书其余内容大部分都或多或少与它们相关。 第 9 章:进一步了解类 此模块继续探讨模块 8 中谈到的类。它涉及很多与类相关的主题,包括重载构造函数、传递对象到函数以及返回对象。它还介绍一种特殊类型的构造函数(称为复制构造函数),这种函数在需要对象副本时使用。接下来介绍友元函数,然后是结构和联合,以及 this 关键字。此模块最后介绍运算符重载,这是 C++ 中最吸引人的功能之一。 第 10 章:继承、虚函数和多态性 此模块讨论 C++ 中与面向对象编程直接相关的三个功能:继承、虚函数和多态性。继承是允许一个类继承另一个类特性的功能。虚函数是在继承的基础上构建的。虚函数支持多态性(面向对象编程的“一个接口,多种方法”原理)。 第 11 章:C++ I/O 系统 C++ I/O 系统非常大,无法在此讨论每个类、函数或功能,不过此模块将介绍最重要和最常用的部分。具体而言,它说明如何输入或输出所设计类的对象。它还介绍如何设置输出格式以及如何使用 I/O 操纵器。此模块最后讨论文件 I/O。 第 12 章:异常、模板和其他高级主题 最后一个模块将介绍几个重要的、高级 C++ 主题,包括异常处理、模板、动态分配和命名空间。另外还介绍运行时类型 ID 和转换运算符。完成此模块后,您将掌握这种语言的核心元素,能够开始编写实际程序。 掌握检查的答案 附录 A:预处理器 预处理器是编译器的一部分,在将源代码实际转换为对象代码之前,预处理器对程序执行各种文本操作。可以为预处理器提供文本操作命令。这些命令称为预处理器指令,它们实际上不是 C++ 的组成部分,但扩展了 C++ 编程环境的范围。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值