排序
一、排序的概念
按照其中的某个或某些关键字的大小,递增或递减的排列起来的操作。
稳定性: 假定在待排序的记录序列中,存在多个具有相同的关键字的记录,若经过排序,这些记录的相对次序保持不变,即在原序列中,r[i]=r[j],且r[i]在r[j]之前,而在排序后的序列中,r[i]仍在r[j]之前,则称这种排序算法是稳定的;否则称为不稳定的。
二、常见排序算法的实现
(一)插入排序
直接插入排序
基本思想:把待排序的记录按其关键码值的大小逐个插入到一个已经排好序的有序序列中,直到所有的记录插入完为止,得到一个新的有序序列。 所以开始认为第一个元素是有序的,依次将其后面的元素插入到这个有序序列中,直到整个序列有序为止。
void InsertSort(int* a, int n)
{
assert(a);
//最后一次,是要把n - 1这个数进行排序,则已经
//排好序的尾部为n - 2
for (int i = 0; i < n-1;i++)
{
//end表示已经排好序的尾标
int end = i;
//首先保存要排序的数,一会就会被覆盖了
int tmp = a[end + 1];
//只要前面的数大于end + 1,则前面的这些数都向后挪动一个位置
while (end >= 0 && a[end] > tmp)
{
a[end + 1] = a[end];
end--;
}
a[end + 1] = tmp;
}
}
希尔排序
基本思想:先选定一个整数N,将所有距离为N的记录分在同一组内,并对每一组内的记录进行排序。然后再取小于N的数作为第二增量,重复上述分组和排序的工作。当到达=1时,所有记录在统一组内排好序,也就是进行一次直接插入排序。
void ShellSort(int* a, int n)
{
assert(a);
int gap = n;
//不能写成大于0,因为gap的值始终>=1
while (gap > 1)
{
//只有gap最后为1,才能保证最后有序,所以这里要加1
gap = gap / 3 + 1;
//这里只是把插入排序的1换成gap即可,
//但是这里不是排序完一个分组,再去
//排序另一个分组,而是整体只过一遍
//这样每次对于每组数据只排一部分
//整个循环结束之后,所有组的数据排序完成
for (int i = 0; i < n - gap; ++i)
{
int end = i;
int tmp = a[end + gap];
while (end >= 0 && a[end] > tmp)
{
a[end + gap] = a[end];
end -= gap;
}
a[end + gap] = tmp;
}
}
}
(二)选择排序
直接选择排序
基本思想:每次从待排序列中选出一个最小值(或者最大值),然后放在序列的第一个(或者最后一个)位置,直到全部待排数据排完。
在这里使用两个变量,一趟同时选出最小值和最大值放在序列的第一个和最后一个位置。
void SelectSort(int* a, int n)
{
assert(a);
int begin = 0, end = n - 1;
while (begin < end)
{
int min = begin, max = begin;
for (int i = begin; i <= end; i++)
{
if (a[i] >= a[max])
max = i;
if (a[i] < a[min])
min = i;
}
Swap(&a[begin], &a[min]);
//如果最大的位置在begin位置,说明min是和最大的交换位置
//这个时候max的位置就发生了变换,max变到了min的位置
//所以要更新max的位置
if (begin == max)
max = min;
Swap(&a[end], &a[max]);
begin++;
end--;
}
}
(三)交换排序
冒泡排序
基本思想:从起始位置开始比较,左边比右边大,交换位置,选出最大值放到最后位置。
void Swap(int* p1, int* p2)
{
int tmp = *p1;
*p1 = *p2;
*p2 = tmp;
}
void BubbleSort(int* a, int n)
{
assert(a);
int end = n;
while (end > 0)
{
/*
加一个标记,如果中间没有发生交换
说明前面的值都比后面的小
即本身就是有序的,最好的情况下,
它的时间复杂度就为N
*/
int flag = 0;
for (int i = 1; i < end; i++)
{
if (a[i - 1] > a[i])
{
Swap(&a[i - 1], &a[i]);
flag = 1;
}
}
if (flag == 0)
{
break;
}
end--;
}
}
快速排序
// 三数取中法,三个中取一个中间值
int GetMidIndex(int* a, int begin, int end)
{
int mid = begin + ((end - begin) >> 1);
if (a[begin] < a[mid])
{
if (a[mid] < a[end])
{
return mid;
}
else if (a[begin] > a[end])
{
return begin;
}
else
{
return end;
}
}
else // begin >= mid
{
if (a[mid] > a[end])
{
return mid;
}
else if (a[begin] < a[end])
{
return begin;
}
else
{
return end;
}
}
}
1、hoare版本(左右指针法)
思路:
int PartSort1(int* a, int begin, int end)
{
int midindex = GetMidIndex(a, begin, end);
Swap(&a[begin], &a[midindex]);
int key = a[begin];
int start = begin;
while (begin < end)
{
// end 找小
while (begin < end && a[end] >= key)
--end;
// begin找大
while (begin < end && a[begin] <= key)
++begin;
Swap(&a[begin], &a[end]);
}
//最后的交换一定要保证a[begin] < a[start], 所以要从右边走
Swap(&a[begin], &a[start]);
return begin;
}
2、挖坑法
思路:1、先将第一个数据存放在临时变量key中,形成第一个坑位
2、从右先左遍历找比key小的值,放在第一个坑位,该位成为一个坑
3、从左向右遍历找比key大的值,放在右边坑位,该位成为坑
4,重复遍历
int PartSort2(int* a, int begin, int end)
{
//begin是坑
int key = a[begin];
while (begin < end)
{
while (begin < end && a[end] >= key)
--end;
// end给begin这个坑,end就变成了新的坑。
a[begin] = a[end];
while (begin < end && a[begin] <= key)
++begin;
// end给begin这个坑,begin就变成了新的坑。
a[end] = a[begin];
}
a[begin] = key;
return begin;
}
3、前后指针版本
思路:1、三数取中选出中间值作为key。perv指向序列开头,cur指向prev指针的后一个位置。
2、cur找小,cur找到比key小的值,将prev先++,不等于cur,交换cur和prev的值,再将cur++。若cur的值比key大,则直接cur++,直到cur大于end,结束遍历,交换begin和prev的值。
3、经过第一趟prev左边的值小于prev,右边的值大于prev。
int PartSort3(int* a, int begin, int end)
{
int midindex = GetMidIndex(a, begin, end);
Swap(&a[begin], &a[midindex]);
int key = a[begin];
int prev = begin;
int cur = begin + 1;
while (cur <= end)
{
// cur找小,把小的往前翻,大的往后翻
if (a[cur] < key && ++prev != cur)
Swap(&a[cur], &a[prev]);
++cur;
}
Swap(&a[begin], &a[prev]);
return prev;
}
排序递归算法
void QuickSort(int* a, int left, int right)
{
if (left >= right)
return;
if (right - left + 1 < 10)
{
InsertSort(a+left, right - left + 1);
}
else
{
int key = PartSort3(a, left, right);
//[left, key-1] [key+1, right]
QuickSort(a, left, key - 1);
QuickSort(a, key + 1, right);
}
}
4、快速排序非递归
void QuickSortR(int* a, int left, int right)
{
Stack st;
StackInit(&st, 10);
//先入大区间
if (left < right)
{
StackPush(&st, right);
StackPush(&st, left);
}
//栈不为空,说明还有没处理的区间
while (StackEmpty(&st) != 0)
{
left = StackTop(&st);
StackPop(&st);
right = StackTop(&st);
StackPop(&st);
//快排单趟排序
int key = PartSort1(a, left, right);
// [left key-1]
// 把大于1个数的区间继续入栈
if (left < key - 1)
{
StackPush(&st, key - 1);
StackPush(&st, left);
}
// [key+1, right]
if (key+1 < right)
{
StackPush(&st, right);
StackPush(&st, key + 1);
}
}
}
(四)归并排序
void _MergeSort(int* a, int left, int right, int* tmp)
{
if (left >= right)
return;
int mid = left + ((right - left) >> 1);
// [left, mid]
// [mid+1, right]
_MergeSort(a, left, mid, tmp);
_MergeSort(a, mid+1, right, tmp);
int begin1 = left, end1 = mid;
int begin2 = mid + 1, end2 = right;
int index = left;
while (begin1 <= end1 && begin2 <= end2)
{
if (a[begin1] <= a[begin2])
tmp[index++] = a[begin1++];
else
tmp[index++] = a[begin2++];
}
while (begin1 <= end1)
{
tmp[index++] = a[begin1++];
}
while (begin2 <= end2)
{
tmp[index++] = a[begin2++];
}
memcpy(a+left, tmp+left, sizeof(int)*(right - left+1));
}