用React构建fiber树的思想解决“反转链表”问题

最近在阅读React源码,还是比较有收获的,这不,今天刷题的时候,直接就把React渲染组件的思想给落地了,并且好像也知道了如何去写React原理专栏。

看题吧:

就是这样,给你一个单链表,要求你把单链表反向输出。

接下来我们来看一下如何使用React渲染组件的思想来把这道题AC。

javascript 复制代码
function ListNode(val, next){
    this.val = (val === undefined) ? 0 : val;
    this.next = (next === undefined) ? null : next;
}

let vdom1 = new ListNode(1);
let vdom2 = new ListNode(2);
let vdom3 = new ListNode(3);
let vdom4 = new ListNode(4);
let vdom5 = new ListNode(5);

vdom1.next = vdom2;
vdom2.next = vdom3;
vdom3.next = vdom4;
vdom4.next = vdom5;

上面就是创建了一个ListNode的构造函数,使用这个构造函数可以创建出一条单链表,如下:

javascript 复制代码
1 -> 2 -> 3 -> 4 -> 5

如何实现翻转链表呢?我们似乎也只有遍历这一条路可走吧,如下:

javascript 复制代码
let reverseList = function (head){
    // 存储当前正在工作的节点
    let workInProgressFiber = head;
    
    // 这个用来存储翻转后的链表
    let result = new ListFiber(0, null);
    let result1 = result;
    
    while(workInProgressFiber){
        // 对链表上的每个node都进行相应的工作
        performUnitOfWork(workInProgressFiber, result)
    }
    
    // 将结果返回
    return result1.next;
}

上面的代码工作流程如下:

1、存储当前遍历到的节点(workInProgressFiber)。

2、对每个节点都进行工作处理(performUnitOfWork)。

3、当workInProgressFiber为空时,意味着链表已经遍历完成,那此时应该将翻转后的链表返回。

那接下来的关键,就是看看performUnitOfWork函数应该如何实现呗,代码如下:

javascript 复制代码
function performUnitOfWork(curWorkInProgressFiber, result){
    // 获取下一个节点
    let nextWorkInProgressFiber = curWorkInProgressFiber.next;
    
    if (!nextWorkInProgressFiber){
        // 说明已经是链表的尾部了,开始归阶段
        completeWork(curWorkInProgressFiber, result);
        // 归阶段结束后,结束整个workLoopSync
        curWorkInProgressFiber = null;
        return
    }
    
    // 让下一个节点的return属性指向当前节点
    nextWorkInProgressFiber.return = curWorkInProgressFiber;
    
    // 将当前节点指向下一个节点,开启下一个performUnitOfWork
    curWorkInProgressFiber = nextWorkInProgressFiber;
}

上述思想如下:

1、将 下一个节点的return属性 指向 当前节点。

2、当遍历到尾部时,咱们最原始的 单链表 就变为了 双向链表。

3、双向链表不是最终答案,根据当前节点的return属性可以找到上级(开启归阶段),我们可以这个关系构建出真正的反转链表。

所以,我们来着重观察下completeWork是如何工作的:

javascript 复制代码
function completeWork(curFiber, result){
    while(curFiber){
        curFiber.next = null;
        result.next = curFiber;
        curFiber = curFiber.return;
        result = result.next;
    }
}

completeWork工作流程如下:

1、自底向上的通过return属性拿到上级节点。

2、将当前节点加入到result表里,因为是自底向上,所以当completeWork工作完成后,我们就可以得到一个翻转的链表。

最后,我们就解决了一个"翻转链表"的问题,leetcode上可以看到执行用时:

用时上也还说的过去,当然React里还有一种更高效的思路可以解决这个问题,那就是收集update。我们都知道组件内部的同类型更新,是通过环形链表来维护的,你可以把一个节点当作一次更新,说到这里,剩下的就交给各位大佬了,欢迎各位把答案发到讨论区里。

以上就是这次的分享内容,其实这个问题还是很简单的,用React思想来解决这种问题并不是最优解,但它可以锻炼你对React源码的理解,所以我觉着还是很有营养的,那么我们下期再见啦,拜拜~~

相关推荐
贵沫末17 分钟前
React——基础
前端·react.js·前端框架
aklry28 分钟前
uniapp三步完成一维码的生成
前端·vue.js
Rubin9336 分钟前
判断元素在可视区域?用于滚动加载,数据埋点等
前端
爱学习的茄子36 分钟前
AI驱动的单词学习应用:从图片识别到语音合成的完整实现
前端·深度学习·react.js
用户38022585982437 分钟前
使用three.js实现3D地球
前端·three.js
weixin_4461224637 分钟前
LinkedList剖析
算法
程序无bug39 分钟前
Spring 面向切面编程AOP 详细讲解
java·前端
zhanshuo39 分钟前
鸿蒙UI开发全解:JS与Java双引擎实战指南
前端·javascript·harmonyos
撰卢1 小时前
如何提高网站加载速度速度
前端·javascript·css·html
10年前端老司机1 小时前
在React项目中如何封装一个可扩展,复用性强的组件
前端·javascript·react.js