iOS + 常用排序算法

算练习吧
参照的原文
常用排序算法总结(一)
八大排序算法

#include <stdio.h>

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<right) {
        // 先冒大
        int index = left;
        for (; index < right; index++){
            if (array[index] > 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<right) {
        
        // 1. 递归对半拆分数组成子序列
        int middle = (left+right)/2;
        // 1.1 左[left, middle]
        mergeSortRecursion(array, left, middle);
        // 1.2 右[middle+1, right]
        mergeSortRecursion(array, middle+1, right);
        
        printf("\n待排\n");
        printArrayWithRange(array, left, right);
        
        // 2. 合并有序的子序列
        merge(array, left, middle, right);
        
        printf("排完一次\n");
        printArrayWithRange(array, left, right);
        printf("最新结果\n");
        printArray(array, right+1);
    }
}

// 归并:非递归:将大问题分割成小问题分别解决
void mergeSortInteration(int array[], int n){
    
    // 按2的幂次方间隔,将数组等分成小组(最后一个小组个数不规则),相隔两个小组进行归并;
    // 不断更改间隔拆分小组,直到能归成一个小组为止
    // [left,middle,right] [left,middle,right] [left,middle,right] .....
    // 由1个为一组开始拆分
    
    int incre = 1 , left = 0, middle=0, right = 0;
    // incre > 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<right; i++) {
        if (array[i] < pivot ){
            separator++;
            exchange(array+i, array+separator);
        }
    }
    
    separator++;
    exchange(array+separator, array+right);
    
    //  separator即基准数的位置
    return separator;
}

// 快排:基准,大的一边、小的另一边,分而治之
// 不稳定:平均:O(nlogn); 最差:O(n^2); 最好:o(nlogn)
void quickSort(int array[], int left, int right){
    
    if (left < right) {
        // 站队,大的一边,小的一边
        int separator = partition(array,left,right);
        // 对小边继续排
        quickSort(array, left, separator-1);
        // 对大边继续排
        quickSort(array, separator+1, right);
    }
}

#pragma mark - 堆排序
/*
 堆排序:选择排序
 1. 将整个数组认为是一个完全二叉树,i的左右节点分别是2*i+1、2*i+2
 2. 建堆:从二叉树最后一个节点的父节点开始调整堆,依次往前
 3. 堆建好后,
    1. 将root元素与堆未元素交换
    2. 获取了一个最大(小)数,由于root元素更换,破坏了堆,需继续调整堆
    3. 堆调整后,重复1,直到剩下两个元素的堆,调整大小
 4. 堆调整:前提是原本是推,由于root元素更换,破坏了堆,需对其进行调整
    1. 与左右子节点比较,是否符合堆的条件,不符合,将更大(小)的元素与堆交换,并记录下被调整堆字节点newRoot
    2. 如果步骤1破坏堆,按步骤1的方式继续调整以newRoot为根的堆
    3. 如果步骤1没有破坏堆,则堆调整好了
 
 稳定性:不稳定;
 时间复杂度:O(nlog2n);
 */

/// 堆调整
void heapAdjust(int array[], int root, int length){
    // 最大堆
    int leftChildIndex = root*2+1;
    int rightChildIndex = root*2+2;
    int newRoot = root;
    
    // 比左节点比较,如果左节点大,则左节点称为root的候选值,
    if (leftChildIndex < length && array[root] < array[leftChildIndex] ){
          newRoot =  leftChildIndex;
    }
    // 将newRoot的值与右节点比较,若右节点大,更换root候选值
    if (rightChildIndex < length && array[newRoot] < array[rightChildIndex] ){
        newRoot =  rightChildIndex;
    }
    
    // root变更,以newRoot为根的堆被破坏,则以newRoot重新调整堆
    if (newRoot != root) {
        exchange(array+newRoot, array+root);
        heapAdjust(array, newRoot, length);
    }
}

/// 建堆
void heapBuild(int array[], int length){
    // 从最后一个元素的父元素开始堆调整
    for (int i = (length-1) / 2; 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;
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,904评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,581评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,527评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,463评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,546评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,572评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,582评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,330评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,776评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,087评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,257评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,923评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,571评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,192评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,436评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,145评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容