iOS + 常用排序算法

作者: YaoYaoX | 来源:发表于2017-05-26 16:32 被阅读88次

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

    #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;
    }

    相关文章

      网友评论

        本文标题:iOS + 常用排序算法

        本文链接:https://www.haomeiwen.com/subject/ablxfxtx.html