排序

作者: 路小白同学 | 来源:发表于2018-03-16 10:24 被阅读0次

    1.冒泡排序
    冒泡算法是一种基础的排序算法,这种算法会重复的比较数组中相邻的两个元素。如果一个元素比另一个元素大(小),那么就交换这两个元素的位置。重复这一比较直至最后一个元素。这一比较会重复n-1趟,每一趟比较n-j次,j是已经排序好的元素个数。每一趟比较都能找出未排序元素中最大或者最小的那个数字。这就如同水泡从水底逐个飘到水面一样。冒泡排序是一种时间复杂度较高,效率较低的排序方法。其空间复杂度是O(n)。

    1, 最差时间复杂度 O(n^2)
      2, 平均时间复杂度 O(n^2)

    如图:


    image

    OC代码如下:

    #pragma mark - 冒泡升序排序
    - (void)bubbleAscendingOrderSortWithArray:(NSMutableArray *)array
    {
        for (int i = 0; i < array.count; i++)
        {
            for (int j = 0; j < array.count - 1 - i;j++)
            {
                if ([array[j+1]intValue] < [array[j] intValue])
                {
                    NSString *temp = array[j];
                    array[j] = array[j+1];
                    array[j+1] = temp;
                }
            }
        }
    }
    
    #pragma mark - 冒泡降序排序
    - (void)bubbleDescendingSortWithArray:(NSMutableArray *)array
    {
        for (int i = 0; i<array.count; i++)
        {
            for (int j = 0; j<array.count-1-i; j++)
            {
                if ([array[j] integerValue]<[array[j+1] integerValue])
                {
                    NSString *temp = array[j];
                    array[j] = array[j+1];
                    array[j+1] = temp;
                }
            }
        }
    }
    
    

    2.选择排序
    实现思路:

    1. 设数组内存放了n个待排数字,数组下标从1开始,到n结束。
       2. i=1
       3. 从数组的第i个元素开始到第n个元素,寻找最小的元素。(具体过程为:先设arr[i]为最小,逐一比较,若遇到比之小的则交换)
       4. 将上一步找到的最小元素和第i位元素交换。
       5. 如果i=n-1算法结束,否则回到第3步

    复杂度:
      平均时间复杂度:O(n^2)
      平均空间复杂度:O(1)


    image

    代码如下:

    #pragma mark - 选择升序排序
    - (NSMutableArray *)selectionAscendingOrderSortWithArray:(NSMutableArray *)array
    {
        for (int i = 0; i<array.count; i++)
        {
            for (int j = i+1; j<array.count; j++)
            {
                if ([array[i] integerValue]<[array[j] integerValue])
                {
                    NSString *temp = array[j];
                    array[j] = array[i];
                    array[i] = temp;
                }
            }
        }
        return array;
    }
    
    #pragma mark - 选择降序排序
    - (NSMutableArray *)selectionDescendingOrderSortWithArray:(NSMutableArray *)array
    {
        for (int i = 0; i<array.count; i++)
        {
            for (int j = i+1; j<array.count; j++)
            {
                if ([array[i] integerValue]>[array[j] integerValue])
                {
                    NSString *temp = array[j];
                    array[j] = array[i];
                    array[i] = temp;
                }
            }
        }
        return array;
    }
    

    3.快速排序:
    快速排序是由东尼·霍尔所发展的一种排序算法。在平均状况下,排序n个元素要O(nlogn)次比较。在最坏状况下则需要O(n^2)次比较,但这种状况并不常见。事实上,快速排序通常明显比其他O(nlogn)算法更快,因为它的内部循环可以在大部分的架构上很有效率地被实现出来。
    快速排序使用分治策略(Divide and Conquer)来把一个序列分为两个子序列。步骤为:
    从序列中挑出一个元素,作为"基准"(pivot).
    把所有比基准值小的元素放在基准前面,所有比基准值大的元素放在基准的后面(相同的数可以到任一边),这个称为分区(partition)操作。
    对每个分区递归地进行步骤1~2,递归的结束条件是序列的大小是0或1,这时整体已经被排好序了。
    如图:


    image

    代码如下:

    #pragma mark - 快速升序排序
    -(void)quickAscendingOrderSort:(NSMutableArray *)arr leftIndex:(NSInteger)left rightIndex:(NSInteger)right
    {
        if (left < right)
        {
            NSInteger temp = [self getMiddleIndex:arr leftIndex:left rightIndex:right];
            [self quickAscendingOrderSort:arr leftIndex:left rightIndex:temp-1];
            [self quickAscendingOrderSort:arr leftIndex:temp + 1 rightIndex:right];
        }
        NSLog(@"快速升序排序结果:%@", arr);
    }
    - (NSInteger)getMiddleIndex:(NSMutableArray *)arr leftIndex:(NSInteger)left rightIndex:(NSInteger)right
    {
        NSInteger tempValue = [arr[left] integerValue];
        while (left<right)
        {
            while (left<right&&[arr[right] integerValue]>=tempValue)
            {
                right --;
            }
            if(left<right)
            {
                arr[left] = arr[right];
            }
            while (left<right&&[arr[left] integerValue]<=tempValue)
            {
                left ++;
            }
            if (left<right)
            {
                arr[right] = arr[left];
            }
        }
        arr[left] = [NSString stringWithFormat:@"%ld",tempValue];
        return left;
    }
    

    4.插入排序
    实现思路:

    1. 从第一个元素开始,认为该元素已经是排好序的。
      2. 取下一个元素,在已经排好序的元素序列中从后向前扫描。
      3. 如果已经排好序的序列中元素大于新元素,则将该元素往右移动一个位置。
      4. 重复步骤3,直到已排好序的元素小于或等于新元素。
      5. 在当前位置插入新元素。
      6. 重复步骤2。

    复杂度:
      平均时间复杂度:O(n^2)
      平均空间复杂度:O(1)

    如图:


    image

    代码如下:

    #pragma mark - 插入升序排序
    - (void)insertionAscendingOrderSort:(NSMutableArray *)ascendingArr
    {
        for (NSInteger i = 1; i < ascendingArr.count; i ++) {
            NSInteger temp = [ascendingArr[i] integerValue];
            for (NSInteger j = i - 1; j >= 0 && temp < [ascendingArr[j] integerValue]; j --) {
                ascendingArr[j + 1] = ascendingArr[j];
                ascendingArr[j] = [NSNumber numberWithInteger:temp];
            }
        }
        NSLog(@"插入升序排序结果:%@",ascendingArr);
    }
    

    5.归并排序

    思路:
    1.申请空间,使其大小为两个已经排序序列之和,该空间用来存放合并后的序列
    2.设定两个指针,最初位置分别为两个已经排序序列的起始位置
    3.比较两个指针所指向的元素,选择相对小的元素放入到合并空间,并移动指针到下一位置
    4.重复步骤3直到某一指针到达序列尾
    5.将另一序列剩下的所有元素直接复制到合并序列尾

    如图:


    image

    代码如下:

    #pragma mark - 归并升序排序
    - (void)megerSortAscendingOrderSort:(NSMutableArray *)ascendingArr
    {
        NSMutableArray *tempArray = [NSMutableArray arrayWithCapacity:1];
        for (NSNumber *num in ascendingArr) {
            NSMutableArray *subArray = [NSMutableArray array];
            [subArray addObject:num];
            [tempArray addObject:subArray];
        }
        while (tempArray.count != 1) {
            NSInteger i = 0;
            while (i < tempArray.count - 1) {
                tempArray[i] = [self mergeArrayFirstList:tempArray[i] secondList:tempArray[i + 1]];
                [tempArray removeObjectAtIndex:i + 1];
                
                i++;
            }
        }
        NSLog(@"归并升序排序结果:%@", ascendingArr);
    }
    
    - (NSArray *)mergeArrayFirstList:(NSArray *)array1 secondList:(NSArray *)array2 {
        NSMutableArray *resultArray = [NSMutableArray array];
        NSInteger firstIndex = 0, secondIndex = 0;
        while (firstIndex < array1.count && secondIndex < array2.count) {
            if ([array1[firstIndex] floatValue] < [array2[secondIndex] floatValue]) {
                [resultArray addObject:array1[firstIndex]];
                firstIndex++;
            } else {
                [resultArray addObject:array2[secondIndex]];
                secondIndex++;
            }
        }
        while (firstIndex < array1.count) {
            [resultArray addObject:array1[firstIndex]];
            firstIndex++;
        }
        while (secondIndex < array2.count) {
            [resultArray addObject:array2[secondIndex]];
            secondIndex++;
        }
        return resultArray.copy;
    }
    

    6.OC自带的数组排序
    一般排序算法在开发总很少用到,因为OC已经有封装好接口。

     NSArray *sortArray = [array sortedArrayUsingComparator:^NSComparisonResult(id  _Nonnull obj1, id  _Nonnull obj2) {
          
          NSNumber *num1 = [NSNumber numberWithInteger:[obj1 integerValue]];
          NSNumber *num2 = [NSNumber numberWithInteger:[obj2 integerValue]];
          //升序
          return [num1 compare:num2];
          //降序
    //      return [num2 compare:num1] ;
      }];
    

    Demo地址

    相关文章

      网友评论

          本文标题:排序

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