如何使用固定枢轴将迭代快速排序更改为使用随机数据块的迭代快速排序?

时间:2013-03-20 05:50:25

标签: c sorting quicksort

我发现这个代码用于带有固定枢轴的快速排序。它总是将给定表的右侧元素作为一个轴。我希望它将随机元素作为一个支点。我认为x是一个支点,所以我认为将它更改为给定列表中的随机元素是一个好主意,但事实证明它不起作用。

void swap ( int* a, int* b )
{
    int t = *a;
    *a = *b;
    *b = t;
}


int partition (int arr[], int l, int h)
{
    int x = arr[h];
    int i = (l - 1);
    for (int j = l; j <= h- 1; j++)
    {
        if (arr[j] <= x)
        {
            i++;
            swap (&arr[i], &arr[j]);
        }
    }
    swap (&arr[i + 1], &arr[h]);
    return (i + 1);
}

void quickSortIterative (int arr[], int l, int h)
{
    int stack[ h - l + 1 ]; 
    int top = -1;

    stack[ ++top ] = l;
    stack[ ++top ] = h;

    while ( top >= 0 )
    {
        h = stack[ top-- ];
        l = stack[ top-- ];

        int p = partition( arr, l, h );

        if ( p-1 > l )
        {
            stack[ ++top ] = l;
            stack[ ++top ] = p - 1;
        } 
        if ( p+1 < h )
        {
            stack[ ++top ] = p + 1;
            stack[ ++top ] = h;
        }
    }
}

我尝试换行

int x = arr[h];

swap(&arr[i+1], &arr[j]);

int r = l+rand()%(h-l);
int x = arr[r];

然后

swap(&arr[i+1], &arr[r]);

但它没有正确排序。显然我在这里做错了什么。请帮忙。

2 个答案:

答案 0 :(得分:2)

您的分区功能假定数据透视位于分区的末尾。 考虑首先将随机数据移动到分区的末尾。

即。添加

swap(&arr[r], &arr[h]);
选择你的支点后

答案 1 :(得分:2)

问题是'partition'函数现在移动了pivot,因此它不会保留在r索引处。并且该函数也会错过最后一个元素(在索引h处)。 最简单的解决方案是在选择它之后将枢轴放在最右边的位置,并保持其他一切相同:在swap(&arr[r], &arr[h]);的第一个循环之前放置partition(),并将最后一个交换恢复到swap (&arr[i + 1], &arr[h]);