iOS 数据结构与算法

iOS 数据结构与算法

一、数据结构

  • 1、集合结构:无序、无重复的元素结构,可看成特殊的数组(没有顺序,并且数据元素不重复)

  • 2、线性结构:
    a、集合中必然存在一个唯一的一个第一元素;
    b、集合中必然存在一个唯一的一个最后元素
    c、除了最后一个元素之外,其他元素均有唯一的后继
    d、除了第一个元素之外,其他元素均有唯一的前驱

  • 3、树形结构:元素存在一对多的树形关系的数据结构,是重要的非线性数据结构


    image.png
  • 4、图形结构:节点的前驱和后继的个数没有限制,类似这样的结构称之为图形数据结构


    image.png

二、数据结构的存储

  • 1、顺序存储结构:连续的内存地址,比如数组

  • 2、链式存储结构
    a、单向链表


    image.png

b、双向链表


image.png

c、循环链表


image.png
  • 3、二叉树/平衡二叉树(五种形式二叉树:a、没有节点,空二叉树;b、只有根节点;c、只有左子树;d、只有右子树;e、有左右子树)


    image.png

二、算法例子

  • 1、不使用中间变量交换两个变量的值
void exchangeWithPlus(int a,int b){
        a = a + b;
        b = a - b;
        a = a - b;
}
void exchangeWithXor(int a,int b){
        a = a ^ b;
        b = a ^ b;
        a = a ^ b;
}
  • 2、求最大公约数
    a、直接遍历法
//swift语言
func normalMaxCommonDivisor(a:Int,b:Int) -> Int{
        let r = a > b ? b : a
        var result = 1
        for i in 2..<r+1{
            if a % i == 0 && b % i == 0{
                result = i
            }
        }
        return result
}

b、辗转相除法:(若a=b*r+q,则gcd(a,b)=gcd(b,q),可自行查找验证)

//swift语言
func tossAndTurnDivisor(a:Int,b:Int) -> Int{
        var ta = a,tb = b
        var resullt = tb
        while ta % tb != 0 {
            resullt = ta % tb
            ta = tb
            tb = resullt
        }
        return resullt
}

c、选择排序:最值出现在起始端(第一趟:在n个数中找到最小/最大的数值与第一个交换;第二趟:在剩下n-1个数中找到最小/最大的数值与第二个数交换;依此类推)

//swift
func selectSort(arr:[Int]) -> [Int]{
        var tArr = arr
        let count = tArr.count
        for i in 0..<count{
            for j in i..<count{
                if tArr[i] > tArr[j]{
                    let temp = tArr[i]
                    tArr[i] = tArr[j]
                    tArr[j] = temp
                }
            }
        }
        return tArr
}

d、冒泡排序:相邻两个元素两两比较,比较完一趟,最值出现在末尾(第 1 趟:依次比较相邻的两个数,不断交换(小数放前,大数放后)逐个推进,最值最后出现在第 n 个元素位置;第 2 趟:依次比较相邻的两个数,不断交换(小数放前,大数放后)逐个推进,最值最后出现在第 n-1 个元素位置;以此类推)

//swift
func bubbleSort(arr:[Int]) -> [Int]{
        var tArr = arr
        let count = tArr.count
        for i in 0..<count{
            for j in 1..<count-i{
                if tArr[j-1] > tArr[j]{
                    let temp = tArr[j-1]
                    tArr[j-1] = tArr[j]
                    tArr[j] = temp
                }
            }
        }
        return tArr
}

e、快速排序:从数列中挑出一个元素,称为 "基准"(pivot);重新排序数列,所有元素比基准值小的摆放在基准前面,所有元素比基准值大的摆在基准的后面(相同的数可以到任一边)。在这个分区退出之后,该基准就处于数列的中间位置。这个称为分区(partition)操作;递归地(recursive)把小于基准值元素的子数列和大于基准值元素的子数列排序;

//swift
//使用例子:quickSort(arr: arr, start: 0, end: arr.count-1)
func quickSort(arr:[Int],start:Int,end:Int) -> [Int]{
        if start >= end{
            return arr
        }
        var tArr = arr
        var base = start
        var offsetI=start,offsetJ=end
        while offsetI<offsetJ {
            while offsetJ >= offsetI && tArr[offsetJ] >= tArr[base] {
                offsetJ = offsetJ - 1
            }
            if offsetJ>=offsetI{
                let temp = tArr[offsetJ]
                tArr[offsetJ] = tArr[base]
                tArr[base] = temp
                base = offsetJ
                offsetJ = offsetJ - 1
            }
            while offsetJ >= offsetI && tArr[offsetI] <= tArr[base] {
                offsetI = offsetI + 1
            }
            if offsetI <= offsetJ{
                let temp = tArr[base]
                tArr[base] = tArr[offsetI]
                tArr[offsetI] = temp
                base = offsetI
                offsetI = offsetI + 1
            }
        }
        let ttarr = quickSort(arr: tArr, start: start, end: base-1)
        let rarr = quickSort(arr: ttarr, start: base+1, end: end)
        return rarr
}

f、折半查找(二分查找):折半查找:优化查找时间(不用遍历全部数据) 折半查找的原理:1> 数组必须是有序的;2> 必须已知 min 和 max(知道范围);3> 动态计算 mid 的值,取出 mid 对应的值进行比较;4> 如果 mid 对应的值大于要查找的值,那么 max 要变小为 mid-1;5> 如果 mid 对应的值小于要查找的值,那么 min 要变大为 mid+1;

//swif
// 已知一个有序数组, 和一个 key, 要求从数组中找到 key 对应的索引位置
//使用例子:binarySearch(arr: arr, start: 0, end: result.count, key: keyValue)
func binarySearch(arr:[Int],start:Int,end:Int,key:Int){
        if start >= end{
            print("找不到该该值的位置")
            return
        }
        let midle = (start + end) / 2
        let midleValue = arr[midle]
        if midleValue < key{
            binarySearch(arr: arr, start: midle+1, end: end, key: key)
        }else if midleValue > key{
            binarySearch(arr: arr, start: start, end: midle-1, key: key)
        }else{
            print("index:\(midle)")
            return
        }
}

g、字符串反转

//swift
func reverseString(originalString:String) -> String{
        var arr = Array(originalString)
        let count = arr.count
        for i in 0..<count/2{
            arr.swapAt(i, count-1-i)
        }
        return String(arr)
}

h、有序数组合并

//swift
func sortArrayCombine(arrA:[Int],arrB:[Int]) -> [Int]{
        if arrA.count <= 0{
            return arrB
        }
        if arrB.count <= 0{
            return arrA
        }
        var resultArr = [Int]()
        var i = 0, j = 0
        let aCount = arrA.count,bCount = arrB.count
        while i < aCount && j < bCount {
            while j < bCount && arrA[i] >= arrB[j] {
                resultArr.append(arrB[j])
                j = j + 1
            }
            while i < aCount && arrA[i] <= arrB[j] {
                resultArr.append(arrA[i])
                i = i + 1
            }
        }
        if i >= aCount{
            while j < bCount {
                resultArr.append(arrB[j])
                j = j + 1
            }
        }
        if j >= bCount{
            while i < aCount {
                resultArr.append(arrA[i])
                i = i + 1
            }
        }
        return resultArr
}

i、hash算法:哈希表 例:给定值是字母 a,对应 ASCII 码值是 97,数组索引下标为 97。 这里的 ASCII 码,就算是一种哈希函数,存储和查找都通过该函数,有效地提高查找效率。在一个字符串中找到第一个只出现一次的字符。如输入"abaccdeff",输出'b'字符(char)是一个长度为 8 的数据类型,因此总共有 256 种可能。每个字母根据其 ASCII 码值作为数组下标对应数组种的一个数 字。数组中存储的是每个字符出现的次数。

//swift
func hashFindFindFisrtOneCharacter(string:String) -> String{
        let strArr = Array(string)
        var sult = [Int]()
        for _ in 0..<256{
            sult.append(0)
        }
        for ch in strArr{
            let tStr = String(ch)
            let index = tStr.unicodeScalars.first!.value
            sult[Int(index)] = sult[Int(index)] + 1
        }
        for i in 0..<sult.count{
            if sult[i] == 1{
                let ch = Character(UnicodeScalar(i)!)
                return String(ch)
            }
        }
        return ""
}

j、查找两个视图的共同父视图:分别记录两个子视图的所有父视图并保存到数组中,然后倒序寻找,直至找到第一个不一样的父视图

func findFaterView(subView:UIView) -> [UIView]{
        var resultArr = [UIView]()
        var temp = subView.superview
        while temp != nil {
            resultArr.append(temp!)
            temp = temp!.superview
        }
        return resultArr
    }
    
    func findTwoViewCommonFathers(viewOne:UIView,viewTwo:UIView) -> [UIView]{
        let viewOneFathers = findFaterView(subView: viewOne)
        let viewTwoFathers = findFaterView(subView: viewTwo)
        var i = 0
        let oneCount = viewOneFathers.count,twoCount = viewTwoFathers.count
        let count = min(oneCount, twoCount)
        var result = [UIView]()
        while i < count {
            let view1 = viewOneFathers[oneCount-i-1]
            let view2 = viewTwoFathers[twoCount-i-1]
            if view1 == view2{
                result.append(view1)
            }else{
                break
            }
            i = i + 1
        }
        return result
    }
  • 转载请标明出处
  • 如有错误理解,还请各路大神批评指出
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,544评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,430评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,764评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,193评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,216评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,182评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,063评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,917评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,329评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,543评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,722评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,425评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,019评论 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,671评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,825评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,729评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,614评论 2 353

推荐阅读更多精彩内容