希尔排序
1959年Shell发明,第一个突破O(n2)的排序算法,是简单插入排序的改进版。它与插入排序的不同之处在于,它会优先比较距离较远的元素。希尔排序又叫缩小增量排序。
算法描述
先将整个待排序的记录序列分割成为若干子序列分别进行直接插入排序,具体算法描述:
- 选择一个增量序列t1,t2,…,tk,其中ti>tj,tk=1;
- 按增量序列个数k,对序列进行k 趟排序;
- 每趟排序,根据对应的增量ti,将待排序列分割成若干长度为m 的子序列,分别对各子表进行直接插入排序。仅增量因子为1
时,整个序列作为一个表来处理,表长度即为整个序列的长度。
希尔排序的核心在于间隔序列的设定。既可以提前设定好间隔序列,也可以动态的定义间隔序列。
动态演示
代码实现
#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;
void shell_sort(vector<int>& arr) {
int n = arr.size();
int step;
//设置间隔序列
for (step = n / 2; step > 0; step /= 2) {
//进行简单插入排序
for (int i = 0; i < step; i++) {
for (int j = i + step; j < n; j += step) {
if (arr[j] < arr[j - step]) {
int temp = arr[j];
int k = j - step;
while (k >= 0 && arr[k] > temp) {
arr[k + step] = arr[k];
k -= step;
}
arr[k + step] = temp;
}
}
}
}
}
void print_arr(vector<int>& arr) {
int n = arr.size();
for (int i = 0; i < n; i++) {
cout << arr[i] << " ";
}
cout << endl;
}
int main() {
vector<int> arr = { 9,5,6,8,7,4,2,3,1 };
cout << "排序前:" << endl;
print_arr(arr);
shell_sort(arr);
cout << "排序后:" << endl;
print_arr(arr);
system("pause");
return 0;
}
结果输出