快速排序变种实现:一次宏定义引发的熬夜事件

Posted zhangyi-studio

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了快速排序变种实现:一次宏定义引发的熬夜事件相关的知识,希望对你有一定的参考价值。

一、背景  

  睡前忽然想写两行代码练练手,想起快速排序的一种变种实现,于是写了快速排序,在理论上完全没问题的情况下测试结果却很诡异,debug半天发现是宏定义导致,作为经验教训记录下来。

二、快速排序变种实现

  原理:

    |_| |_| |_________________________________________|

    L    R                     unSorted

              ---------------------------------------------------------------------------------------------------------------------

   1、取哨兵下标为low,即首元素;初始状态L和R集合都为空,L集合用来存放小于pivot的值,R集合用于存放大于pivot的值。

              pivot = vector<T>& vec[low] ;   mi = low;

                 ↓mi

              |——|

              |_| |_| |_________________________________________|

    L    R                     unSorted

   2、遍历初始集合,如果发现当前下标元素值大于pivot则迭代器加1,不做任何操作,效果相当于移动元素到R中。

                          idx

              |_| |__| |________________________________________|

    L    R                     unSorted

       3、如果发现当前下标元素值小于pivot则需要将当前值放于集合L中,为了提升效率,使当前元素和R集合的首元素交换,然后迭代器加1;作用效果相当于L元素新增,R元素整体后移一位。

                    mi

                     ↓      idx

              |__| |__| |_______________________________________|

    L    R                     unSorted

  4、当遍历完当前序列后unSorted的将为空,L和R将包含所有小于/大于pivot元素的集合。

                                                      mi

                 ↓

              |___________________| |_________________________|

                    L                                           R

  5、最终将首元素,即pivot和mi所指的元素交换,左右效果是mi这个选出来的哨兵节点已经就位,完成了一趟排序,之后将调用递归的方式对前半段和后半段进行排序(和大家所熟知的快排一致)。

三、具体实现,以下代码中swap函数用宏定义实现,为了逻辑紧凑,在swap的参数中用了“++”操作符。 

  1、排序逻辑                   

 1 #ifndef __ALGO_SORT_QUICK_SORT_IMPROVED_H__
 2 #define __ALGO_SORT_QUICK_SORT_IMPROVED_H__
 3 
 4 #include <vector>
 5 #include <iostream>
 6 using namespace std;
 7 
 8 #define swap(a, b)    ({ 9     typeof(a) tmp = (a) ; 10     (a) = (b); 11     (b) = tmp; })
12 
13 template<typename Type>
14 static int partion(std::vector<Type>& elems, int low, int high)
15 {
16     Type pivot = elems[low];
17     int mi = low;
18     for (int i = low + 1; i <= high; i++)
19     {   
20         if (elems[i] < pivot)
21         {
22             //++mi;
23             swap(elems[++mi], elems[i]);
24         }
25     }   
26     swap(elems[mi], elems[low]);
27     return mi; 
28 }
29 
30 template<typename Type>
31 void quick_sort_improved(std::vector<Type>& elems, int low, int high)
32 {
33     if (low < high) 
34     {   
35         int mi = partion(elems, low, high);
36         quick_sort_improved(elems, low, mi - 1); 
37         quick_sort_improved(elems, mi + 1, high);
38     }   
39 }
40 
41 #endif

  2、测试程序

1     vector<double> nums{18.1,16.12,32.21,12.22,13.1,53.21,221.5,354,123,42,22.11,33};
2     quick_sort_improved<double>(nums, 0, nums.size() - 1); 
3     for (int i = 0; i < nums.size(); i++ )
4     {   
5         cout << nums[i] << " ";
6     }   
7     cout << endl;

  3、测试输出

yg@yg-PC:~/workspace/Algo/src/sort/insertSort$ ./a.out 
6.52013e-319 6.52013e-319 6.52013e-319 12.22 12.22 16.12 33 0 0 0 42 42 

   4、分析:由上结果可知,测试结果并未按照预期输出,乱七八糟的。

四、修复版

  加了诸多debug信息,得知问题出在使用宏定义的过程中传入了++操作符,熟悉宏定义的人肯定遇到过类似问题;当前问题是:宏定义中表达式出现几次,++将会被调用几次,这当然不是期望的结果;知道原因后稍加修改。

template<typename Type>
static int partion(std::vector<Type>& elems, int low, int high)
{
    Type pivot = elems[low];
    int mi = low;
    for (int i = low + 1; i <= high; i++)
    {   
        if (elems[i] < pivot)
        {
            ++mi;
            swap(elems[mi], elems[i]);
        }
    }   
    swap(elems[mi], elems[low]);
    return mi; 
}

   测试输出:正确,符合预期

yg@yg-PC:~/workspace/Algo/src/sort/insertSort$ ./a.out 
12.22 13.1 16.12 18.1 22.11 32.21 33 42 53.21 123 221.5 354

 

五、总结

  日常代码中宏定义有时候无法避免,就像上面用到的swap,即便实现上已经避免了很多低级错误;但宏定义依然有很多不尽人意之处,就像上面的使用,编译器甚至都不会给个警告,在运行期也可以正常运行,但结果通却是莫名其妙的错误;通过一番折腾找出原因所在,以后在使用宏定义的地方一定要避免这种操作。

以上是关于快速排序变种实现:一次宏定义引发的熬夜事件的主要内容,如果未能解决你的问题,请参考以下文章

104,排序-快速排序

在项目中只调用一次宏

java 自定义事件的触发及监听

代码片段如何使用CSS来快速定义多彩光标

MySQL:想实现sql语句进行批量删除数据库或表,而引发的熬夜探究

VS中添加自定义代码片段——偷懒小技巧