React 模拟面试题 2023 版
之前总结了 Vue 相关的面试题,看的人没想到挺多的,但是也为我带来了不小的争议。虽然我有明确写出答案《仅供参考》但是还是有同学照抄呀,这么说呢就是你完全没有思考直接跟面试官对答,就算你没说错但也经不起细问的呀。
废话不多说,俺还是比较关心大伙的春秋招面试准备得咋样了,面试的 React 复习得怎么样了呢?
如果你感觉在 React 这方面还比较薄弱的话,不如来做一做这套模拟面试吧,看看大伙能不能打个满分,祝你顺利~
🌸🌸🌸🌸🌸答案仅供参考🌸🌸🌸🌸🌸
欢迎访问552
道大厂真题了,涵盖各类前端的真题。也可以访问我的 面试专栏 ,我的文章会第一时间更新在这。
免杠声明
1、咱们是 模拟面试 并且是 React 专题 ,不是你真实面试的场景,咱们是把 React 面试题集中到了这场模拟面试中了。
2、因为是模拟面试,所以会有一些 情景导入 以及模拟面试官(也就是我)的 心理活动。目的是能让你有点被面试的感觉,以及让你了解一些面试官在想些什么。当然如果觉得啰嗦的话,直接点目录就可以到你想看的题了。
3、 答案会详略得当,不会面面俱到。我更偏向于告诉你,这个题第一层你可以怎么展开,第二层在你展开的这几点各自下面的重要点又是什么。至于你要不要扩充各个层级的点,以及要不要更深入增加层级,就是需要自己去思考和研究的了。
---- 进入正题化身为面试官 ----
📞📞 电话拨通中,喂~,听得到吗,听得到是吧 😊,那我们面试开始了,你先做个自我介绍吧 。。。
在你自我介绍的时候呢,我就看看你做过的项目,技术栈什么的。
看你的项目经历写的是 React 项目,有写过其他项目嘛?你觉得 React 好在哪里呢?或者为什么选择 React 呢?
1、简述一下 React?
首先要明确的是这是一个开发性的问题,他没有标准答案,目的其实更偏向于打开你的话匣子,所以这里可以尽量分维度的多说一些。同时这里要避免几个坑:第一是陷入一个点一直说(除非面试官显得很有兴趣),第二是说不到点子,第三是对比框架踩一捧一。
参考答案
(我这里写的比较全哦,比较概况,可以根据自己掌握情况做适当拓展)
1、一句话解释技术本质:React 是一个用于构建用户界面的 JavaScript 库。
2、核心概念:核心概念有三点,分别是声明式、组件化与 通用性。声明式-直观与组合。组件化-视图的拆分与模块复用,高内聚低耦合。通用性-一次学习,随处编写。
3、对比/优缺点:首先对比其他库比如 vue、jQuery,React 没有引入模板,而是通过 jsx 的语法编写组件化的 ui,代码更为简洁。其次 React 在减少操作 dom 等性能优化上采用的了虚拟 dom 的技术,现在这个技术被广泛的学习和认可。在者,React 提供的各种 hook,能让开发者编写更高效稳定的代码。最后,当然 React 也有缺点, 由于 React 并不是一个一揽子框架,所以导致在技术选型与学习使用上有比较高的成本。
😶 至于为啥问这个问题,其实也是有点私心哈哈,毕竟这个问题你可以自己说出一些你认为 react 核心的东西,往往可能是 hook、虚拟 dom、组件化、jsx 等等,这种你说出来的我更乐意问,不然真的就像考试一样,拿着面试题一个个问了。这里可能我会顺着你说的 react 区别其他组件的点,顺着说,比如下面的这几个例子。
诶,你刚刚说到了 JSX,那你说说什么是 JSX 呗?为什么 React 要使用 JSX?
2、什么是 JSX,为什么要使用 JSX?
参考答案
是什么?:JSX 是一个 JavaScript 的语法扩展,结构类似 XML。JSX 主要用于声明 React 元素,但 React 中并不强制使用 JSX。即使使用了 JSX,也会在构建过程中,通过 Babel 插件编译为 React.createElement。所以 JSX 更像是 React.createElement 的一种语法糖。
为什么?:
1、JSX 是一个 JavaScript 的语法扩展,没有引入新概念,上手简单
2、JavaScript 的语法扩展+结构类似 XML,代码变得更为简洁,而且代码结构层次更为清晰。
3、结构类似 XML,React 需要将组件转化为虚拟 DOM 树,所以在编写代码时,实际上是在手写一棵结构树。而 XML 在树结构的描述上天生具有可读性强的优势。
听你有说到 hooks,我们公司现在项目基本都是用 hooks,那你自己有写过 hooks 嘛?说一下 useRef 的实际使用场景吧?
3、详细说说某某 hook?
可能第一个问题你有说到 hooks 以及他的好处什么的,那么这下面这道题就是看看你对是不是只是停留在会说。这类问题主要考察你对 hooks 的熟练程度,至少需要答到,它什么时候用,以及需要注意的点。
这里我就不一个个的列出 hook 了,就以 useRef 为例吧:
参考答案
useRef 的应用场景主要是:函数组件需有访问 dom 元素的场景,以及保持可变变量的场景。 需要知道的是:ref.current 发生变化并不会造成 re-render; useRef 和 useState 不同,如果一个状态或者数据会影响 DOM 的渲染结果,一定要避免使用 useRef 来保持引用4、React 组件通信
参考答案
父组件 => 子组件:- Props
- Instance Methods
子组件 => 父组件:
- Callback Functions
- Event Bubbling
兄弟组件之间:
- Parent Component
不太相关的组件之间:
- Context
- Portals
- Observer Pattern
- Redux 等
------ 源码考察分界线 ------
第三题也能聊到源码哈,下面的几道题是关于 React 核心的 架构 和虚拟 DOM 的。
5、setState 是同步更新还是异步更新?
参考答案
在源码中,通过 isBatchingUpdates 来判断 setState 是先存进 state 队列还是直接更新,如果值为 true 则执行异步操作,为 false 则直接更新。一般认为,做异步设计是为了性能优化、减少渲染次数。一般会被追问(也可以自己说):在什么情况下 isBatchingUpdates 会为 true 呢?
在 React 可以控制的地方,就为 true,比如在 React 生命周期事件和合成事件中,都会走合并操作,延迟更新的策略。
在 React 无法控制的地方,比如原生事件,具体就是在 addEventListener 、setTimeout、setInterval 等事件中,就只能同步更新。
有看过 React 的源码嘛?可以简单的向我描述一下 Fiber 架构吗
6、简述 fiber 架构
参考答案
React16 开始的fiber架构可以分为三层,相较于React15,新增了Scheduler(调度器),Reconciler从递归处理虚拟DOM变为可中断的循环过程,:- Scheduler(调度器)—— 调度任务的优先级,高优任务优先进入 Reconciler
- Reconciler(协调器)—— 负责找出变化的组件
- Renderer(渲染器)—— 负责将变化的组件渲染到页面上
官方也推荐了
是什么?
Fiber 是 React 16 中采用的新协调(reconciliation)引擎,主要目标是支持虚拟 DOM 的渐进式渲染。
解决了什么问题?
Fiber 将原有的 Stack Reconciler 替换为 Fiber Reconciler,提高了复杂应用的可响应性和性能。
怎么解决的
- 对大型复杂任务的分片。
- 对任务划分优先级,优先调度高优先级的任务。
- 调度过程中,可以对任务进行挂起、恢复、终止等操作。
TODO
简述可能错在错误理解,后续会更新详细文章解答。
7、虚拟 DOM 的工作原理
参考答案
这个问题可能就会让你摸不着头脑,要讲清一个技术的原理,我们只要从三大方面着手就行,是什么、为什么、怎么做。这个问题可能不会问的这么泛,可能会问更细一点,比如:什么是虚拟DOM,他有什么优缺点,如何实现虚拟DOM。什么是虚拟 DOM
虚拟 DOM 实际上它只是一层对真实 DOM 的抽象,以 JavaScript 对象 (VNode 节点) 作为基础的树,用对象的属性来描述节点,最终可以通过一系列操作使这棵树映射到真实环境上。
虚拟 DOM 优缺点
优点:改善大规模 DOM 操作的性能、规避 XSS 风险、能以较低的成本实现跨平台开发。
缺点:内存占用较高,因为需要模拟整个网页的真实 DOM。高性能应用场景存在难以优化的情况,类似像 Google Earth 一类的高性能前端应用在技术选型上往往不会选择 React。
如何实现虚拟 DOM
1、h 函数,用 JS 对象模拟 DOM 树
2、render 函数,实现渲染,从 Virtual DOM 映射到真实 DOM
3、mount 函数,实现挂载
4、diff 算法:比较两棵虚拟 DOM 树的差异
那你再说说 React 的 Diff 算法是怎么实现的吧
8、React 的 diff 算法
参考答案
diff 算法是一种对比两个树差异的一种算法,那在 React 里就是对比新旧树的差异了。那么我们可以说 React 中 Diff 算法的本质是:
对比 current Fiber
如果该 DOM 节点已在页面中,current Fiber 代表该 DOM 节点对应的 Fiber 节点。
和 JSX 对象
即 ClassComponent 的 render 方法的返回结果,或 FunctionComponent 的调用结果。JSX 对象中包含描述 DOM 节点的信息。
并且生成 workInProgress Fiber。
如果该 DOM 节点将在本次更新中渲染到页面中,workInProgress Fiber 代表该 DOM 节点对应的 Fiber 节点。
React 对 diff 算法的优化,毕竟要完全对比两棵树的复杂度是很大的,所以 React 的 diff 算法预设了三个限制:
只对同级元素进行 Diff。如果一个 DOM 节点在前后两次更新中跨越了层级,那么 React 不会尝试复用他。
两个不同类型的元素会产生出不同的树。如果元素由 div 变为 p,React 会销毁 div 及其子孙节点,并新建 p 及其子孙节点。
开发者可以通过 key prop 来暗示哪些子元素在不同的渲染下能保持稳定。
------ 技术栈分界线 ------
以上都是多少都可以问一问的题,下面的就涉及技术栈了。你可以理解为根据公司技术栈出题。也可以理解为和 React 强相关的技术栈出题。
TIP
篇幅原因,只是挑选了一些面试中和 React 相关性比较强,且比较常见的问题。
class
下面两道问题一般是公司存在 class 语法代码可能会问的题。 打个比方:公司项目旧代码是用 class 写的,希望你来了能维护或者重构就会问一些 class 的知识。
9、class 与 hooks 的区别/优劣
参考答案
共同点是两者的功能和效果都是一样的,类组件和函数组件都可以作为基础组件展示 UI。
他们的设计理念不一样,一个是面向对象,一个是函数式。面向对象的核心是继承、生命周期等这些。而函数式 immutable(不变)、没有副作用、引用透明等这些特点。
Hooks 更优的原因: Hooks 有确定的输入输出,没有 this 指向问题,也不用 renderprops 或者 Hoc 去解决复用状态逻辑的问题,它是一个组合的思想,组合更优于继承。class 组件业务逻辑散落在生命周期中,Hooks 则淡化了生命周期的概念。Hooks 的函数组件可以提供比原先更细粒度的逻辑组织与复用,且能更好地适用于时间切片与并发模式。
10、聊聊 class 与 hook 的生命周期
参考答案
分为三个阶段 挂载->更新-> 卸载 (Error)详细请看表格,其中标注 UNSAFE_ 的都是被弃用的,通过这个表格可知道各方法所处的时期:
Mount | Update | Unmount | Error | |
---|---|---|---|---|
Render 阶段 | construct、getDerivedStateFromProps、UNSAFE_componentWillMount、Render | UNSAFE_componentWillReceiveProps、getDerivedStateFromProps、shouldComponentUpdate、UNSAFE_componentWillUpdate、Render | getDerivedStateFromError | |
commit 阶段 | componentDidMount、getSnapshotBeforeUpdate | componentDidUpdate | componentWillUnmount | componentDidCatch |
对于异步请求,应该放在 componentDidMount 中去操作。
constructor:可以放,但从设计上而言不推荐。constructor 主要用于初始化 state 与函数绑定,不承载业务逻辑且现在已经很少使用了。componentWillMount:已被标记废弃,在新的异步渲染架构下会触发多次渲染,容易引发 Bug。
getDerivedStateFromProps 容易编写反模式代码,使受控组件与非受控组件区分模糊。
UNSAFE_componentWillMount 被标记弃用,主要原因是新的异步渲染架构会导致它被多次调用。所以网络请求及事件绑定代码应移至 componentDidMount 中。
UNSAFE_componentWillReceiveProps 被标记弃用,被 getDerivedStateFromProps 所取代,主要原因是性能问题。
shouldComponentUpdate 通过返回 true 或者 false 来确定是否需要触发新的渲染。主要用于性能优化。
UNSAFE_componentWillUpdate 同样是由于新的异步渲染机制,而被标记废弃,原先的逻辑可结合 getSnapshotBeforeUpdate 与 componentDidUpdate 改造使用。
如果在 componentWillUnmount 函数中忘记解除事件绑定,取消定时器等清理操作,容易引发 bug。
如果没有添加错误边界处理,当渲染发生异常时,用户将会看到一个无法操作的白屏,所以一定要添加。
vue
还有的公司 React 和 Vue 技术栈都有,你过来可能主要还是写 React 但是也希望你会一些 Vue。那么下面几道题你很可能会遇到。
情景导入:如果简历没写我可能会问,之前有写过 Vue 项目嘛?(有写过啊,就直接问,没写过就跳过这个技术栈)简历有写的话我就会说,我看你的 xxx 项目是用 Vue 写的,那。。。
11、你觉得 React 与 Vue 的区别主要在哪呢?
参考答案
1、各自推崇的/核心思想
React 推崇函数式编程(纯组件),数据不可变以及单向数据流。函数式编程最大的好处是其无副作用(稳定性)和确定的输入输出(可测试性),所以通常说 React 适合大型应用。
Vue 推崇灵活易用(渐进式开发体验),数据可变,双向数据绑定。尽可能的降低了前端开发的门槛。
2、写法差异
React 没有太多的概念和 api,推崇 all in js,使用 JSX 和 部分 hook 就能开发。通过原生 JS 实现模板中的常见语法,比如插值,条件,循环等,都是通过 JS 语法实现的。更加原生。要求更高的 js 能力,以及对设计模式(也可以说开发经验)的掌握。
Vue 是基于 template 模板 + options API。引入了较多的概念和 api,需要理解或记住大部分概念和 api 才能编写。更要求熟练使用官方概念和 api。
3、数据流
React 提倡的是单向数据流,数据不可变,需要 setState 驱动新的 State 替换老的 State。
Vue 数据被观测是双向绑定的,省去了数据手动处理更加便捷。
4、Render(渲染过程)
React 在应用的状态被改变时,全部子组件都会重新渲染。通过 shouldComponentUpdate 这个生命周期方法可以进行控制,但 Vue 将此视为默认的优化。
Vue 可以更快地计算出 Virtual DOM 的差异,这是由于它在渲染过程中,会跟踪每一个组件的依赖关系,不需要重新渲染整个组件树。
5、事件机制
React 原生事件被包装,所有事件都冒泡到顶层 document 监听,然后在这里合成事件下发。基于这套,可以跨端使用事件机制,而不是和 Web DOM 强绑定。React 组件上无事件,父子组件通信使用 props。
Vue 原生事件使用标准 Web 事件,Vue 组件自定义事件机制,是父子组件通信基础,Vue 合理利用了 snabbdom 库的模块插件
6、
React 主要使用 diff 队列保存需要更新哪些 DOM,得到 patch 树,再统一操作批量更新 DOM。
Vue Diff 使用双向链表,边对比,边更新 DOM。
7、
注意两者处于不同赛道,不存在谁取代谁这一说法,别踩一捧一。
React 只关注底层,上层应用解决方案交给社区,造就了 React 社区繁荣,同时 React 团队有更多时间专注底层。未来 React 的发展依然会在 函数式编程 这个核心思想的下进行升级。
Vue 提供了一揽子全家桶解决方案,比如 Vuex、Vue-Router、Vue-CLI、Vutur 工具等。减少选择困难症,只需认准官方给出的解决方案即可。Vue 依然会定位简单易上手(渐进式开发),依然是考虑通过依赖收集来实现数据可变。
12、vuex 与 redux 的区别
参考答案
从使用上来说
在 Vuex 中,$store被直接注入到了组件实例中,因此可以比较灵活的使用:使用dispatch、commit提交更新,通过mapState或者直接通过this.$store 来读取数据。
在 Redux 中,我们每一个组件都需要显式的用 connect 把需要的 props 和 dispatch 连接起来。
Vuex 更加灵活一些,组件中既可以 dispatch action,也可以 commit updates,而 Redux 中只能进行 dispatch,不能直接调用 reducer 进行修改。
从实现原理来说
Redux 使用的是不可变数据,而 Vuex 的数据是可变的。
Redux 每次都是用新 state 替换旧 state,而 Vuex 是直接修改。
Redux 在检测数据变化的时候,是通过 diff 的方式比较差异的,而 Vuex 其实和 Vue 的原理一样,是通过 getter/setter 来比较的。
状态/路由
13、flux 状态管理
参考答案
Flux 是一种基于单向数据流的架构。架构如图所示:
具体流程:Store 存储了视图层所有的数据,当 Store 变化后会引起 View 层的更新。如果在视图层触发 Action,比如点击一个按钮,当前的页面数据值会发生变化。Action 会被 Dispatcher 进行统一的收发处理,传递给 Store 层。由于 Store 层已经注册过相关 Action 的处理逻辑,处理对应的内部状态变化后,会触发 View 层更新。
14、简述 redux 状态管理
参考答案
核心设计
包含了三大原则:单一数据源、纯函数 Reducer、State 是只读的。
一个核心点是处理“副作用”。
- 第一类是在 Dispatch 的时候会有一个 middleware 中间件层,拦截分发的 Action 并添加额外的复杂行为,还可以添加副作用。
- 第二类是允许 Reducer 层中直接处理副作用,采取该方案的有 React Loop,React Loop 在实现中采用了 Elm 中分形的思想,使代码具备更强的组合能力。
AJAX 请求等异步工作,或不是纯函数产生的第三方的交互都被认为是 “副作用”
数据流
Redux 中整个数据流的方案与 Flux 大同小异。
首先是 dispatch 一个 action。然后 reducer 会收到这个 action, 根据这个 action 对状态进行修改。状态修改以后会被处理容器捕捉到。从而对相关的界面进行更新。
- Store,Store 存放应用程序的状态,并且有帮助函数来访问这些状态。Store 可以用来聆听变化和发送 action。Store 只有一个。
- Reducers,数据的状态是通过 reducer 函数来改变的。
- Actions,Actions 代表的是一个对象。有两部分,一个是 action 本身,另一个就是它的 payload。简单说就是对哪些数据进行哪些操作。
- React-Redux,Redux 本身和 React 没有关系,只是数据处理中心,是 React-Redux 让他们联系在一起。React-rRedux 提供两个方法:connect 和 Provider。
15、mobx 和 redux 有什么区别?
参考答案
- 核心的差异:Redux 是单向数据流,Mobx 则是通过监听数据的属性变化,直接在数据上更改来触发 UI 的渲染。
- redux 是只读的,不能直接去修改它,而是应该返回一个新的状态,同时使用纯函数。mobx 中的状态是可变的,可以直接对其进行修改。
- mobx 结果也难以预测,调试会比较困难。redux 提供能够进行时间回溯的开发工具,同时其纯函数以及更少的抽象,让调试变得更加的容易
16、React-Router 的实现原理是什么?
参考答案
基于 history 库来实现不同的客户端路由实现思想,并且能够保存历史记录等,磨平浏览器差异,上层无感知。
不同的客户端路由实现思想:1、基于 hash 的路由:通过监听 hashchange 事件,感知 > hash 的变化。通过 location.hash=xxx 改变 hash 。2、基于 H5 history 路由:通过自定义事件触发实现监听 url 的变化。可以通过 history.pushState 和 resplaceState 等改变 url ,会将 URL 压入堆栈,同时能够应用 history.go() 等 API。
通过维护的列表,在每次 URL 发生变化的回收,通过配置的 路由路径,匹配到对应的 Component,并且 render。
17、React-Router 实现路由切换
参考答案
有几种方式:使用
<Route>
,会比较<Route>
的 path 属性和当前地址的 pathname 实现路由切换。使用
<Switch>
会遍历其所有的子<Route>
元素,并仅渲染与当前地址匹配的第一个元素。使用
<Link>
、<NavLink>
、<Redirect>
,会在你的应用程序中创建链接,通过 to 属性与当前地址匹配。
18、React-Router 如何获取 URL 的参数和历史对象?
参考答案
- get 传值:通过 location.search 获取 url 获取到一个字符串'?id='1111',通过浏览器的 URLSearchParams api 或自封装字符串解析方法解析出 id 的值。
- 动态路由传值:通过 match.params.id 或者 useParams(Hooks)取得 url 中的动态路由 id 部分的值
- 通过 query 或 state 传值:to 属性传递对象或数组时,通过 location.state 或 location.query 来获取即可,但是存在缺点就是,只要刷新页面参数就会丢失。
------ 设计模式/代码优化 ------
React 是对设计模式的能力很有要求的,这里只列举和 React 强相关的设计模式。
TODO
更详细的设计模式文章,待更新至设计模式专栏
19、HOC
参考答案
高阶组件(HOC)是 React 中用于复用组件逻辑的一种高级技巧。HOC 自身不是 React API 的一部分,它是一种基于 React 的组合特性而形成的设计模式。
在我们的应用程序中,我们经常希望在多个组件中使用相同的逻辑。此逻辑可以包括将特定样式应用于组件、要求授权或添加全局状态。
能够在多个组件中重用相同逻辑的一种方法就是使用高阶组件模式。这种模式允许我们在整个应用程序中重用组件逻辑。
具体而言,高阶组件是参数为组件,返回值为新组件的函数。HOC 包含我们想要应用于作为参数传递的组件的某些逻辑。应用该逻辑后,HOC 返回带有附加逻辑的元素。
// HOC模式例子
function withStyles(Component) {
return (props) => {
const style = { padding: '0.2rem', margin: '1rem' }
return <Component style={style} {...props} />
}
}
function Button() {
return <button>Click me!</button>
}
function Text() {
return <p>Hello World!</p>
}
const StyledButton = withStyles(Button)
const StyledText = withStyles(Text)
// HOC模式例子
function withStyles(Component) {
return (props) => {
const style = { padding: '0.2rem', margin: '1rem' }
return <Component style={style} {...props} />
}
}
function Button() {
return <button>Click me!</button>
}
function Text() {
return <p>Hello World!</p>
}
const StyledButton = withStyles(Button)
const StyledText = withStyles(Text)
20、Render Props
参考答案
术语 “render prop” 是指一种在 React 组件之间使用一个值为函数的 prop 共享代码的简单技术
具有 render prop 的组件接受一个返回 React 元素的函数,并在组件内部通过调用此函数来实现自己的渲染逻辑。
// Render Props模式例子
<DataProvider render={data => <h1>Hello {data.target}</h1>} />
// Render Props模式例子
<DataProvider render={data => <h1>Hello {data.target}</h1>} />
21、非受控组件
参考答案
在大多数情况下,推荐使用 受控组件 来处理表单数据。在一个受控组件中,表单数据是由 React 组件来管理的。另一种替代方案是使用非受控组件,这时表单数据将交由 DOM 节点来处理。
因为非受控组件将真实数据储存在 DOM 节点中,所以在使用非受控组件时,有时候反而更容易同时集成 React 和非 React 代码。如果你不介意代码美观性,并且希望快速编写代码,使用非受控组件往往可以减少你的代码量。否则,你应该使用受控组件。
// 非受控组件例子
class NameForm extends React.Component {
constructor(props) {
super(props);
this.handleSubmit = this.handleSubmit.bind(this);
this.input = React.createRef();
}
handleSubmit(event) {
alert('A name was submitted: ' + this.input.current.value);
event.preventDefault();
}
render() {
return (
<form onSubmit={this.handleSubmit}>
<label>
Name:
<input type="text" ref={this.input} />
</label>
<input type="submit" value="Submit" />
</form>
);
}
}
// 非受控组件例子
class NameForm extends React.Component {
constructor(props) {
super(props);
this.handleSubmit = this.handleSubmit.bind(this);
this.input = React.createRef();
}
handleSubmit(event) {
alert('A name was submitted: ' + this.input.current.value);
event.preventDefault();
}
render() {
return (
<form onSubmit={this.handleSubmit}>
<label>
Name:
<input type="text" ref={this.input} />
</label>
<input type="submit" value="Submit" />
</form>
);
}
}
------ 性能优化 ------
22、如何发现以及分析性能问题
参考答案
肉眼可见的性能问题,如白屏、卡顿、加载时间很长,一般是要答带数据的优化。lightHouse 的评分,可以作为优化依据。
React 性能检测的一些工具也可以:Profiler 能测出 reRender 的耗时,平常对数据要求不严也就是不需要准确测出 rerender 的时间的话可以使用 ReactDevtool。
浏览器开发者工具可以检测到大部分的性能数据:
网络面板的一些优化方案:
- 阻塞或者排队,由于一个域名最多维护 6 个链接,可以做域名分片或者多个域名。
- 网络原因可以用 CDN 缓存
- 下载时间过长可以压缩或者 webpack 打包优化。
performance:
- FPS (Frames Per Second)每秒传输帧数,发现页面帧速图表出现红色块,代表一帧所需时间过长->卡顿。
- CPU 图表显示占得面积太大,可能某个 js 占用太多主线程时间
- V8 内存使用凸显一直上升,内存泄漏可能存在
23、怎么做性能优化
参考答案
1、写 React 代码的优化
减少计算
- 增加 key
- commit 阶段减少耗时操作
- 一些 hook : useMemo、useCallback、React.Memo 4. setState 将多个合并,或者用 ustable_batchedUpdate 批量更新
精细化渲染
- 优先用户响应,耗时任务放到下一个宏任务。“关闭弹窗类似的场景”
- usecontext 跳过中间组件 ,发布订阅模式,redux
- useMemo 也可以跳过 render
控制范围
- 防抖节流
- 懒渲染("虚拟列表")、懒加载(webPack)
- 避免在 didMount 或 didUpdate 中更新 state
2、webpack 的优化(除去开发环境的优化)
打包速度
- onOf、external 可以跳过 loader 的查找,除去一些不打包
- babel 缓存,可以对运行结果缓存
- 多进程打包 thread-loader
运行性能
- 文件资源缓存 hash、chunkhash、contenthash,这几个的区别也要明白
- treeShacking
- code split
- 懒加载、预加载(某些浏览器不支持)
- 离线可访问 pwa (也是一种优化,但用的不多)
3、网络等优化
- CDN 缓存
- 域名分片
- 文件压缩
- DNS 预解析,提前解析之后可能会用到的域名
最后的话
⭐️虽然说现在大环境不好,工作难找但还是衷心希望各位准备面试的小伙伴面试顺利~,收割 offer,我们一起加油吧 🤝!还有就是新年快乐 ❤️ ~
🚀 文章还有不足,如果你有意向,很欢迎你可以加入我们,文章后续会持续更新~