前端算法与数据结构——链表/树-二叉树

链表

链表和数组相似,都是有序的列表,都是线性结构(有且仅有一个前驱,有且仅有一个后续)。

不同点在于:链表中,数据单位的名称叫做“结点”,而结点和结点的分布,在内存中可以是离散的。

这个“离散”是相对于数组的“连续”来说的。

数组在内存中最为关键的一个特征,我在之前介绍数组的文章里有介绍,它一般是对应一段位于自己上界和下界之间的、一段连续的内存空间。元素和元素之间,紧紧相连。

而链表中的结点,则允许散落在内存空间的各个角落里。一个内容为1->2->3->4->5的链表,在内存中的形态可以是散乱的:

链表

由于数组中的元素是连续的,每个元素的内存地址可以根据其索引距离数组头部的距离来计算出来。

因此对数组来说,每一个元素都可以通过数组的索引下标直接定位。对链表来所,元素与元素之间似乎毫无内存上的瓜葛。

这种各据山头,我们又该如何遍历呢?

在链表中,每一个结点都包含了两部分内容:数据域指针域

JS中的链表,是以嵌套的对象的形式来实现的:

{ // 数据域

    val: 1,

    // 指针域,指向下一个结点

    next: { val:2, next: ... }

}

数据域:存储的是当前结点所存储的数据值

指针域:代表下一个结点(后续结点)

各个结点之间的联系

把这个关系简化一下:

链表

要想访问链表中的任何一个元素,我们都得从起点结点开始,逐个访问next,一直访问到目标结点为止。

为了确保起点结点是可抵达的,我们有时还会设定一个head指针(dummy结点)来专门指向链表的开始位置:

链表的基本形态

链表结点的创建:

function ListNode(val) {

    this.val = val;

    this.next = null;

}

在使用构造函数创建结点是,传入val(数据域),指定next(下一个链表结点)即可:

const node = new ListNode(1)

node.next = new ListNode(2)

// 这就创建了一个数据域为1,next结点数据域为2的链表1->2

链表的操作:


添加

在尾部添加,改变一个next指针就行。

在两个结点间插入一个结点(这也是链表基础中的一个关键考点):

要想完成这个操作,我们需要变更的是前驱结点目标结点的next指针指向。如图:

插入过程

// 如果目标结点本来不存在,那么记得手动创建

const node3 = new ListNode(3)

// 把node3的 next 指针指向 node2(即 node1.next)

node3.next = node1.next

// 把node1的 next 指针指向 node3

node1.next = node3

删除

删除的标准:在链表的遍历过程中,无法再遍历到某个结点的存在。

按照这个标准,要想遍历不到 node3,我们直接让它的前驱结点 node1 的 next 指针跳过它、指向 node3 的后继即可:

删除操作

node1.next = node3.next

在涉及链表删除操作的题目中,重点不是定位目标结点,而是定位目标结点的前驱结点

做题时,完全可以只使用一个指针(引用),这个指针用来定位目标结点的前驱结点。比如说咱们这个题里,其实只要能拿到 node1 就行了:

// 利用 node1 可以定位到 node3

const target = node1.next

node1.next = target.next

高效的增删操作

在链表中,添加和删除操作的复杂度是固定的——不管链表里面的结点个数 n 有多大,只要我们明确了要插入/删除的目标位置,那么我们需要做的都仅仅是改变目标结点及其前驱/后继结点的指针指向。 因此我们说链表增删操作的复杂度是常数级别的复杂度,用大 O 表示法表示为 O(1)。

麻烦的访问操作

但是链表也有一个弊端:当我们试图读取某一个特定的链表结点时,必须遍历整个链表来查找它。

随着链表长度的增加,我们搜索的范围也会变大、遍历其中任意元素的时间成本自然随之提高。这个变化的趋势呈线性规律,用大 O 表示法表示为 O(n)。

但在数组中,我们直接访问索引、可以做到一步到位,这个操作的复杂度会被降级为常数级别(O(1)):

对数组还不清楚的可以看我的另外一篇对数组讲解的文章。

链表的插入/删除效率较高,而访问效率较低;数组的访问效率较高,而插入效率较低。

理解树结构


在理解计算机世界的树结构之前,大家不妨回忆一下现实世界中的树有什么特点:一棵树往往只有一个树根,向上生长后,却可以伸展出无数的树枝、树枝上会长出树叶。由树根从泥土中吸收水、无机盐等营养物质,源源不断地输送到树枝与树叶的那一端。一棵树往往呈现这样的基本形态:

数据结构中的树,首先是对现实世界中树的一层简化:

树根——>根节点

树枝——>边

树枝的两端——>结点

树叶——>叶子结点

数据结构中的树

树的关键特性和重点概念:

1.树的层次计算规则:根节点所在的那一层为第一层,其子结点所在的就是第二层,以此类推。

2.结点和树“高度”计算规则:叶子结点高度记为1,每往上一层高度就加1,逐层向上累加至目标结点时,所得到的值就是目标结点的高度。树中结点的最大高度,称为“树的高度”。

3.“”的概念:一个结点开叉出去多少个子树,被称为结点的“度”。比如我们上图中,根节点的“度”就是3。

4."叶子结点":叶子结点就是度为0的结点。

这里我们着重了解二叉树。

二叉树


满足二叉树的条件:

1.可以没有根节点,作为一棵空树存在

2.如果它不是空树,那么必须由根节点,左子树和右子树组成,且左右子树都是二叉树。

TIps:二叉树不能被简单定义为每个结点的度都是2的树

普通的树并不会区分子树和子树,但在二叉树中,左右子树的位置是严格约定,不能交换的。

二叉树JS编码实现


在JS中,二叉树使用对象来定义。

1.数据域

2.左侧子结点的引用

3.右侧子结点的引用

在定义二叉树构造函数时,我们需要把左侧子结点和右侧子结点都预置为空:

// 二叉树结点的构造函数

function TreeNode(val) {

    this.val = val;

    this.left = this.right = null;

}

构建结点

以这个结点为根结点,我们可以通过给 left/right 赋值拓展其子树信息,延展出一棵二叉树。因此从更加细化的角度来看,一棵二叉树的形态实际是这样的:

二叉树

关于二叉树的遍历,由于内容较多,决定在下一篇文章里再做详细讲解。

以上均为阅读掘金小册的《前端算法与数据结构面试:底层逻辑解读与大厂真题训练》所获。富有的朋友们值得购买一读。

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