算练习吧
参照的原文
常用排序算法总结(一)
八大排序算法
#include
void printArrayWithRange(int array[], int left, int right){
for (int i = left; i <= right ; i++) {
printf("%d ",array[i]);
}
printf("\n");
}
void printArray(int array[], int n){
printArrayWithRange(array, 0, n-1);
}
void exchange(int *a, int *b){
if (a==b) {
return;
}
// // 异或交换
// *a ^= *b; // atemp = a^b
// *b ^= *a; // btemp = a^b^b = a
// *a ^= *b; // atemp = a^b^a = b
// // 加减法交换
// *a = *a + *b;
// *b = *a - *b;
// *a = *a - *b;
// 中间变量
int temp = *a;
*a = *b;
*b = temp;
}
#pragma mark - 冒泡
/**
冒泡:相邻对比、交换,最后冒出最大(最小)值;
稳定性:稳定;
时间复杂度:O(n^2);
*/
void bubbleSort(int array[], int n){
for (int i = n ; i > 0 ; i--) { // 待排序个数
for (int j = 0; j < i - 1; j++) { // 排序指针
if (array[j] > array[j+1]) {
exchange(&array[j], &array[j+1]);
}
}
printArray(array, n);
}
}
// 冒泡改进:鸡尾酒算法:同时冒泡最大和最小
void cocktailSort(int array[], int n){
int left = 0;
int right = n-1;
while (left array[index+1]) {
exchange(&array[index], &array[index+1]);
}
}
right --;
// 再冒小
for (; index > left; index--) {
if (array[index] < array[index-1]) {
exchange(&array[index], &array[index-1]);
}
}
left++;
printArray(array, n);
}
}
#pragma mark - 选择排序
/**
选择:每次找到最大(小),与未排序的第一个数交换;
稳定性:不稳定;
时间复杂度:O(n^2);
*/
void selectionSort(int array[], int n){
int index = 0;
// 从第一个数开始排,找到比起大的数并交换,指针前移,对后面的数继续排序
for (int i = 0 ; i < n ; i++) {
// 比对值
index = i;
// 找到最大数
for (int j = i+1 ; j < n ; j++) {
if (array[j] > array[index]) {
index = j;
}
}
// 找到则交换
if (index != i) {
exchange(array+i, array+index);
}
printArray(array, n);
}
}
#pragma mark - 插入排序
/**
直接插入:在未排序的部分拿出一个数,比前面数小(大)则前移,否则处理下个数;
稳定性:稳定;
时间复杂度:最好O(n);最坏、平均O(n^2);
*/
void insertionSort(int array[], int n){
for (int i = 1 ; i < n ; i++) {
// 挑选出未排序的一个数,准备排序
int data = array[i];
// 空出来的位置
int j = i;
// 将比data大的数后移
while (j > 0 && data < array[j-1]) {
// 将j的位置空出来
array[j] = array[j-1];
j--;
}
// 插入到正确位置
array[j] = data;
printArray(array, n);
}
}
// 二分插入排序:稳定:在未排序的部分拿出一个数,与排好序的部分采用二分查找分进行对比,找到其位置插入
// 稳定性:稳定;
// 时间复杂度:最好O(nlog2n);最坏、平均O(n^2);
void binaryInsertionSort(int array[], int n){
int data, left, right, middle;
for (int i = 1 ; i < n ; i++) {
// 挑选出未排序的一个数,准备排序
data = array[i];
// [0, i-1]为排好序的部分
left = 0;
right = i-1;
// 二分查找:left为最终位置
while (left <= right) {
middle = (right+left)/2;
if (data > array[middle]) {
right = middle-1;
} else {
left = middle+1;
}
}
// 将left之后的数据往后挪一位
for (int j = i-1 ; j >= left ; j--) {
array[j+1] = array[j];
}
// 插入到正确位置
array[left] = data;
printArray(array, n);
}
}
// 希尔排序:递减增量分成小组,进行插入排序,插入排序对有序数组的效率比较高
// 稳定性:不稳定
// 时间复杂度:最好O(n),平均O(n^1.3),最坏O(n^2)
void shellSort(int array[], int n){
// 子序列增量
int incre = n, still = 1;
while(still){
incre /= 2;
printf("\n\n第一层\n\n");
// 拆分子序列
for (int i = 0 ; i < n ; i++) {
printf("第二层");
// 单个子序列i,i+incre, i+2*incre, i+3*incre...
for (int j = i + incre ; j < n ; j += incre) {
// 子序列采用插入排序
for(int z = j ; z > i ; z -= incre){
if(array[z] < array[z-incre]){
int temp = array[z-incre];
array[z-incre] = array[z];
array[z] = temp;
printArray(array, n);
} else {
printArray(array, n);
break;
}
}
}
}
if (incre == 1) {
still = 0;
}
// printArray(array, n);
}
}
#pragma mark - 归并排序
/*
归并:将数组分成多个小组,将其排好序,再将小组合并
稳定性:稳定;
时间复杂度:O(nlog2n);
*/
// 合并数组中排好序的两列[left, middle]和 [middle+1, right]
void merge(int array[],uint left, uint middle, uint right){
// 异常判断
if ( middle < left || middle >right || right <= left) {
return;
}
// left、right只相差1
if(left+1 == right){
if (array[left]>array[right]) {
exchange(array+left, array+right);
}
return;
}
// 临时存储
int count = right - left + 1, index = 0;
int *arrayTemp = malloc(sizeof(int)*count);
for (int i = left; i<=right; i++,index++) {
arrayTemp[index] = array[i];
}
// 合并
int tempMid = middle - left, tempLeft = 0, tempRight = right - left;
int i=tempLeft , j=tempMid+1;
index = left;
while (i<=tempMid && j<=tempRight) {
if (arrayTemp[i] < arrayTemp[j]) {
array[index] = arrayTemp[i];
i++;
} else {
array[index] = arrayTemp[j];
j++;
}
index ++;
}
//处理尾数
while (i<=tempMid) {
array[index] = arrayTemp[i];
i++;
index++;
}
while (j<=tempRight) {
array[index] = arrayTemp[j];
j++;
index++;
}
//free(arrayTemp);
}
// 归并:递归
void mergeSortRecursion(int array[], int left, int right){
// 当left==right时,即为一个数,不执行拆分操作
if(left n/2 的时候就可以归成一组
while (incre < n) {
// 更改间隔后重新归并
left = 0;
while (left < n) {
// 相邻两组归并
// 界定righ
right = left+2*incre-1;
if (right>=n) {
right=n-1;
}
// 界定middle
middle = left+incre-1;
// 合并
merge(array, left, middle, right);
// 下一组
left = right+1;
}
printf("排完一次\n");
printArray(array, n);
// 组的个数增大,继续
incre *= 2;
}
}
#pragma mark - 快速排序
/**
快速排序:
1. 以尾数当基准数,分队,大的在一边(大边)、小的在另一边(小边),最后可以确定基准数的位置
2. 小边、大边继续按步骤1处理,
稳定性:不稳定;
时间复杂度:最好、平均 O(nlog2n),最坏 O(n^2);
空间复杂度:O(nlog2n)
*/
// 快排分组
int partition(int array[], int left, int right){
int pivot = array[right]; // 基准数
int separator = left-1; // separator左边的数小于基准数,右边大于基准数
for (int i = left; i= 0; i--) {
heapAdjust(array, i, length);
}
}
/// 堆排序
void heapSort(int array[], int length){
heapBuild(array, length);
for (int i = 0; i < length; i++) {
// 末尾数与堆顶交换,并重新调整堆;
exchange(array, array+length-i-1);
heapAdjust(array, 0, length-i-1);
}
}
int main(int argc, const char * argv[]) {
int array[] = {2, 6, 0, 9, 7, 5, 8, 1, 4, 3, 43, 10, 4, 49, 89, 56,2, 45,75, 23,54,7,45,97,23,65,32,54,23};
int count = sizeof(array)/sizeof(int);
heapSort(array, count);
printArray(array, count);
return 0;
}