CS 基础:线性表

线性表

零个或多个数据元素的有限序列。

线性表的结构

  • 顺序存储
  • 链式存储

顺序存储结构

用一段地址连续的存储单元依次存储线性表的元素,因为地址是连续的,顺序存储结构元素读取很快,但是这个结构的线表进行插入以及删除等操作涉及到的元素的地址都要进行变换,都很不方便,于是衍生出下面那个链式的结构。
一个图:



34567的存储位置都要变更。

链式存储结构(链表)

链表的两种结构

  • 单向链表:每一个节点的只有下一个节点的引用
  • 双向链表:每一个节点有前一个和后一个节点的引用

每一个链表都有头节点以及尾节点:

实现一个双向链表

推荐研读Swift Algorithm Club: Swift Linked List Data Structure

构造一个节点类

public class Node{
    var value : String
    init(value:String){
        self.value = value
    }
}

value 可以是任意的数据类型,根据链表的用途来定。在最后一个章节用泛型来改造这个数据结构。

加入对下一个节点的引用

public class Node{
    var next : Node?
    var value : String
    init(value:String){
        self.value = value
    }
}

加入对上一个节点的引用

public class Node{
    var next : Node?
    weak var previous : Node?
    var value : String
    init(value:String){
        self.value = value
    }
}

建立一个链表类

public class List {
    fileprivate var head : Node?
    private var tail : Node?
    
    public var isEmpty : Bool{
        return head == nil
    }
    
    public var head : Node?{
        return head
    }
    
    public var tail : Node?{
        return tail
    }
}

让链表类可以添加节点元素

public class List {
    fileprivate var head : Node?
    private var tail : Node?
    
    public var isEmpty : Bool{
        return head == nil
    }
    
    public var head : Node?{
        return head
    }
    
    public var tail : Node?{
        return tail
    }
    
    public func append(value:String){
        let newNode = Node(value:value)
        
        if let tailNode = tail{
            newNode.previous = tailNode
            tailNode.next = newNode
        }else{
            head = newNode
        }
        
        tail = newNode
    }
}

List 类的实践

首先添加一个 List 的 extension 协助打印输出

extension List{
    public var description : String{
        var text = "["
        var node = head
        
        while node != nil{
            text += "\(node!.value)"
            node = node!.next
            if node != nil {
                text += " , "
            }
        }
        return text + "]"   
    }
}
let listt = List()
listt.append(value:"熊大")
listt.append(value:"熊二")
listt.append(value:"周末")
listt.append(value:"蛋蛋")

print(listt.description)

打印结果:

[熊大 , 熊二 , 周末 , 蛋蛋]

根据下标读取链表节点

由于链表的性质,从头结点开始读取,直到读到对应的下标停止并返回相应的节点。
给 List 类添加一个读取下标对应节点函数:

public func nodeAt(index:Int) -> Node? {
        if index >= 0 {
            var node = head
            var i = index
            
            while node != nil {
                if i == 0 {
                    return node
                }
                i -= 1
                node = node!.next
            }
        }
        return nil
    }

清空链表

只要将首节点与尾节点置为 nil 整个链表就会被置空。
给 List 添加函数:

public func removeAll() {
        head = nil
        tail = nil
    }

移除其中某个节点

移除节点的三种情况:

  • 移除首节点:如图变更 Head 指针以及 Node1 的 previous 指针
  • 移除尾节点:如图变更 Tail 指针以及 Node2 的 next 指针指为 nil。
  • 移除非首尾节点:如图只需要变更被移除的 Node1 节点的上一个节点 Node0 的 next 指针指向 Node2,Node2 的 previous 指针指向 Node0,那么 Node1 节点就已经脱离了链表。
Screen Shot 2017-02-07 at 10.41.06 AM.png

给 List 添加移除节点函数:

public func remove(node: Node){
        let prev = node.previous
        let next = node.next
        
        if let prev = prev {
            prev.next = next
        }else {
            head = next
        }
        
        next?.previous = prev
        
        if next == nil {
            tail = prev
        }
        
        node.previous = nil
        node.next = nil
    }

在进行一次实践

代码:

let list = List()
list.append(value:"熊大")
list.append(value:"熊二")
list.append(value:"周末")
list.append(value:"蛋蛋")
print(list.description)

let node = list.nodeAt(index:1)
list.remove(node:node!)
print(list.description)

list.removeAll()
print(list.description)

输出:

[熊大 , 熊二 , 周末 , 蛋蛋]
[熊大 , 周末 , 蛋蛋]
[]

使用泛型来改造这个链表类

改造节点 Node 类

public class Node<T>{
    var next : Node<T>?
    weak var previous : Node<T>?
    var value : T
    init(value:T){
        self.value = value
    }
}

改造链表 List 类

public class List<T> {
    fileprivate var head : Node<T>?
    private var tail : Node<T>?
    
    public var isEmpty : Bool{
        return head == nil
    }
    
    public var first : Node<T>?{
        return head
    }
    
    public var last : Node<T>?{
        return tail
    }
    
    public func append(value:T){
        let newNode = Node(value:value)
        
        if let tailNode = tail{
            newNode.previous = tailNode
            tailNode.next = newNode
        }else{
            head = newNode
        }
        
        tail = newNode
    }
    
    public func nodeAt(index:Int) -> Node<T>? {
        if index >= 0 {
            var node = head
            var i = index
            
            while node != nil {
                if i == 0 {
                    return node
                }
                i -= 1
                node = node!.next
            }
        }
        return nil
    }
    
    public func removeAll() {
        head = nil
        tail = nil
    }
    
    public func remove(node: Node<T>){
        let prev = node.previous
        let next = node.next
        
        if let prev = prev {
            prev.next = next
        }else {
            head = next
        }
        
        next?.previous = prev
        
        if next == nil {
            tail = prev
        }
        
        node.previous = nil
        node.next = nil
    }
}

实践

利用泛型特性改造之后,这个链表类则可以储存任意类型的值,比较灵活。

输入:

let list = List<String>()
list.append(value:"熊大")
list.append(value:"熊二")
list.append(value:"周末")
list.append(value:"蛋蛋")

输出:

[熊大 , 熊二 , 周末 , 蛋蛋]

end

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

推荐阅读更多精彩内容

  • 1.线性表的定义 线性表:零个或多个数据元素的有限序列序列:也就是说元素之间是有顺序的,若元素存在多个,则第一个元...
    e40c669177be阅读 2,049评论 6 15
  • 原创文章&经验总结&从校招到A厂一路阳光一路沧桑 详情请戳www.codercc.com 1.Concurrent...
    你听___阅读 6,528评论 0 5
  • 一.线性表 定义:零个或者多个元素的有限序列。也就是说它得满足以下几个条件:  ①该序列的数据元素是有限的。  ②...
    Geeks_Liu阅读 2,694评论 1 12
  • 从数据的逻辑结构来分,数据元素之间存在的关联关系被称为数据的逻辑结构。归纳起来,应用程序中的数据大致哟如下四种基本...
    Jack921阅读 923评论 0 2
  • 黄昏时分,北京五环外的一处建筑工地旁,55岁的老范坐在马路牙子上吃着他的晚饭。三个馒头、一袋混合着豆腐干、花生米和...
    8b1571ffc0d7阅读 287评论 0 1