swift 归并排序

归并排序,简单来说就是先将数组不断细分成最小的单位,然后每个单位分别排序,排序完毕后合并,重复以上过程最后就可以得到排序结果。该算法采用经典的分治策略(分治法将问题分成一些小的问题然后递归求解,而治的阶段则将分的阶段得到的各答案"修补"在一起,即分而治之)。

(上图摘自网络,如有侵权删除).png

其代码如下:

//排序入口
func mergeSort(arr:inout [Int]) {
        if arr.count <= 1 {
            return
        }
        process(arr: &arr, left: 0, right: arr.count - 1)
    }
    //该方法主要用作二分
    func process(arr:inout [Int],left:Int,right:Int) {
        if left == right {
            return
        }
        let middle = left + (right - left) >> 1
        process(arr: &arr, left: left, right: middle)
        process(arr: &arr, left: middle + 1, right: right)
        merge(arr: &arr, left: left, middle: middle, right: right)
        
        
    }
    //将二分后的数组进行合并
    func merge(arr:inout [Int],left:Int,middle:Int,right:Int){
        var help:[Int] = [Int]()
        
        var leftLocation:Int = left
        var rightLocation:Int = middle + 1
        
        while (leftLocation <= middle && rightLocation <= right) {
            if arr[leftLocation] <= arr[rightLocation] {
                help.append(arr[leftLocation])
                leftLocation += 1
            }else{
                help.append(arr[rightLocation])
                rightLocation += 1
            }
        }
        
        while leftLocation <= middle {
            help.append(arr[leftLocation])
            leftLocation += 1
            
        }
        while rightLocation <= right {
            help.append(arr[rightLocation])
            rightLocation += 1
        }
        var index = left
        for item in help {
            arr[index] = item
            index += 1
        }
        
    }

其核心为merge方法:

  • 1、准备两个变量分别指向二分后数组的左端,上方代码分别为leftLocationrightLocation
  • 2、比较arr[leftLocation]arr[rightLocation]位置上的数谁小,谁小取谁,并位置+1,直到其中一个走到终点;
  • 3、然后把未走到终点的数组添加到help数组中,然后修改回原数组。

具体merge如下图:


image.png

归并排序的非递归实现

//===  通过步长来解决--非递归版的归并排序
    func mergeSortFeidiGui(arr:inout [Int]) {
        if arr.count <= 1 {
            return
        }
        let maxCount = arr.count
        //步长
        var step:Int = 1
        
        while step < maxCount  {
            var left:Int = 0
            while left < maxCount {
                let middle = left + step - 1
                if middle > maxCount {
                    break
                }
                let right:Int = (middle + step) > (maxCount - 1) ? (maxCount - 1) : (middle + step)
                
                merge(arr: &arr, left: left, middle: middle, right: right)
                left = right + 1
            }
            if step > (maxCount / 2) {
                break
            }
            step = step << 1
        }
        
    }

其核心思想-借助于步长的概念:

  • 1、先让步长为1,然后使数组0-1、2-3、4-5、6-7、…………n-1到n,范围上的数组有序
  • 2、然后步长翻倍,使数组0-4、4-8、8-12、…………n-3到N范围上的数有序
  • 3、然后再步长翻倍,使数组有序
  • 4、不断循环直到数组有序

小和问题

在一个数组中,每一个数左边比当前数小的数累加起来,叫做这个数组的小和。求一个数组 的小和。
例子:
[1,3,4,2,5]
1左边比1小的数,没有;
3左边比3小的数,1;
4左边比4小的数,1、3;
2左边比2小的数,1;
5左边比5小的数,1、3、4、2;
所以小和为1+1+3+1+1+3+4+2=16

func getArrMinCount(arr:inout [Int]) -> Int{
        if arr.count < 2 {
            return 0
        }
        return getProcess(arr: &arr,left: 0,right: arr.count - 1)
    }
    //该方法主要用作二分
    func getProcess(arr:inout [Int],left:Int,right:Int) -> Int {
        if left == right {
            return 0
        }
        let middle = left + (right - left) >> 1
        let leftCount = getProcess(arr: &arr, left: left, right: middle)
        let rightCount = getProcess(arr: &arr, left: middle + 1, right: right)
        let mergeCount = getMerge(arr: &arr, left: left, middle: middle, right: right)
        return leftCount + rightCount + mergeCount
    }
    
    func getMerge(arr:inout [Int],left:Int,middle:Int,right:Int) -> Int{
        var res:Int = 0
        var help:[Int] = [Int]()
        var leftLocation:Int = left
        var rightLocation:Int = middle + 1
        while leftLocation <= middle && rightLocation <= right {
            if arr[leftLocation] < arr[rightLocation] {
                help.append(arr[leftLocation])
                let numberCount = right - rightLocation + 1
                res += arr[leftLocation] * numberCount
                print("===>\(arr[leftLocation]) --- > \(numberCount)")
                
                leftLocation += 1
                
            }else if arr[leftLocation] == arr[rightLocation]{
                help.append(arr[rightLocation])
                rightLocation += 1
            }else{
                help.append(arr[rightLocation])
                rightLocation += 1
            }
        }
        while leftLocation <= middle {
            help.append(arr[leftLocation])
            leftLocation += 1
        }
        while rightLocation <= right {
            help.append(arr[rightLocation])
            rightLocation += 1
        }
        var index:Int = left
        for item in help {
            arr[index] = item
            index += 1
        }
        
        return res
    }

在num数组中任意位置右边的数*2后依然比该位置数小,求数组中这样数的个数


    func getArrNumberDouble(arr:inout [Int]) -> Int{
        if arr.count < 2 {
            return 0
        }
        return getNumberDoubleProcess(arr: &arr,left: 0,right: arr.count - 1)
    }
    //该方法主要用作二分
    func getNumberDoubleProcess(arr:inout [Int],left:Int,right:Int) -> Int {
        if left == right {
            return 0
        }
        let middle = left + (right - left) >> 1
        let leftCount = getNumberDoubleProcess(arr: &arr, left: left, right: middle)
        let rightCount = getNumberDoubleProcess(arr: &arr, left: middle + 1, right: right)
        let mergeCount = getNumberDoubleMerge(arr: &arr, left: left, middle: middle, right: right)
        return leftCount + rightCount + mergeCount
    }
    func getNumberDoubleMerge(arr:inout [Int],left:Int,middle:Int,right:Int) -> Int{
        var help:[Int] = [Int]()
        var res:Int = 0
        
        var l:Int = left
        var r:Int = middle + 1
        
        //[26,9,4,1]
        print(" OOOOOOOOOO--- L >\(left) --- M >\(middle) --- R >\(right)")
        while l <= middle {
            while r <= right && arr[l] > arr[r] * 2{
                r += 1
            }
            res = res + r - middle - 1
            print(" --- > \(res) --- L >\(left) --- M >\(middle) --- R >\(right)")

            l += 1
        }
        
        var leftLocation:Int = left
        var rightLocation:Int = middle + 1
        while leftLocation <= middle && rightLocation <= right {
            if arr[leftLocation] < arr[rightLocation]{
                help.append(arr[leftLocation])
                leftLocation += 1
            }else{
                help.append(arr[rightLocation])
                rightLocation += 1
            }
        }
        while leftLocation <= middle {
            help.append(arr[leftLocation])
            leftLocation += 1
        }
        while rightLocation <= right {
            help.append(arr[rightLocation])
            rightLocation += 1
        }
        var index = left
        for item in help {
            arr[index] = item
            index += 1
        }
        return res
    }

思路如下:
1、在进行merge之前,先对数组中符合要求的数据做过滤并保存
2、边merge边进行符合要求数组的记录保存
符合要求的数据做过滤及保存逻辑如下

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

推荐阅读更多精彩内容

  • 归并排序基于分治,利用归并来实现排序,其基本思想是:如果一个数组有n个数据,则可以把这个数组看作n个有序的子序列,...
    FlyElephant阅读 268评论 0 1
  • 1. 解题思路 归并排序的核心思想是先让序列的左半部分有序、再让序列的右半部分有序,最后从两个子序列(左右两半)从...
    霍运浩阅读 424评论 0 0
  • 问题描述 在一个数组中,每一个数左边比当前数小的数累加起来,叫做这个数组的小和。 举个栗子 {1,3,4,2,5}...
    dlihasa阅读 565评论 0 1
  • 核心思想:将一个数组进行排序,可以先(递归)将他分成两半分别排序,然后把结果合并起来.若将两个有序表合并成一个有序...
    浩林Leon阅读 2,255评论 0 1
  • 本文是对 Swift Algorithm Club 翻译的一篇文章。Swift Algorithm Club是 r...
    Andy_Ron阅读 1,188评论 0 3