React中11个设计的瑕疵

2023-12-18 15:58

本文主要是介绍React中11个设计的瑕疵,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

React 已经很优秀了,但依旧有很多可以改善的小细节。

1. 技术选型

React/Vue/Anguler.都是一条单行道,当我们这个项目做好了框架的选型之后,你基本在几年内就没法切换了,顶多只能切换同类型的框架,比如 React,在移动端我们可以用 Preact。

2. 学习成本

时至今日,如果你在 NPM 上搜索 React 你会发现已经有91008个包了。生态繁荣是好事,但也意味着同一个问题可能会有很多种解决方案。这在无意间就提升了我们的学习成本。

另外,React每年都保持着一个高频率的更新,比如今年的 hook 对于前端来说又是一次理念上的革新。React 是改善 UI 体验的领头者,也是在前端多种概念的创造者。

3. 兼容性问题,但问题不大

React 无法兼容低版本的老式浏览器,然而国内经常还是看见有这部分的需求。

4. this 问题

绑定事件的 this 问题,在 react 存在很久了。早在前端对于 this 的理解参差不齐的时候,很容易就在这里采坑出错。

现在我们常用的是在构造函数中 bind 绑定,也有直接箭头函数绑定,还出现了很多 autobind 这样的绑定this工具库。但为什么 React 不能帮我们自动绑定 this 呢?

而官方不帮我们自动绑定 this,主要归咎于 JS 自身的语法问题。

5. setState 异步 or 同步?

我们都知道 React 为了性能考虑 setState 是异步更新的。但是如果脱离了 react 的掌控范围之内则是同步的。

class App extends React.Component {state = {count: 0};componentDidMount() {const btn = document.getElementById('test');btn.addEventListener('click', this.handleClick); // 立即执行}handleClick = () => {this.setState({count: this.state.count + 1});};render() {return (<button id="test" onClick={this.handleClick}>click</button>);}
}

如果在项目中分散了各种这样混合的代码,很可能让你获取到不被期望的state,难以维护。

6. setState callback hell

想象一下,时常我们需要获取最新的状态而做某些事情,那我们就只能在 setState 的 callback里来做。但是为什么 react 不直接提供 promise 的版本呢?

class CallbackHell extends React.component {handleClick = () => {this.setState({}, () => {doSomething()this.setState({}, () => {doOtherSomething()})})}
}

搜一下 PR 不难发现,其实早在2017年就有人提过了类似的问题。https://github.com/facebook/react/pull/9989/commits/b5da0b3aff4ecbbdff4ba264f2f6ee33afeb4899

PS:笔者的猜测可能是因为 react 官方不希望把这些 callback 放入 microtask 里去执行。可以在下方留言表达你的看法。

7. 合成事件

在 React 中,我们获取的事件并不是原生事件而是合成事件,合成事件初衷是为了提升性能。但也会带来一些问题。

class SyntheticEvent extends React.component {handleClick = (e) => {console.log(e);setTimeout(() => {console.log(e); // 无法访问})}
}

那我们要如何才能访问到呢?React 给我们提供了一个接口。Event Pooling

class SyntheticEvent extends React.component {handleClick = (e) => {console.log(e);e.persist(); // 放弃合成事件setTimeout(() => {console.log(e); // 正常访问})}
}

除了异步中不能访问 event 以外,还有事件冒泡的问题。

class SyntheticEvent extends React.Component {componentDidMount() {const btn = document.getElementById('test')!;btn.addEventListener('click', () => {console.log('document bind');});}handleClick = (e) => {console.log('点击了');e.stopPropagation();};render() {return (<button id="test" onClick={this.handleClick}>click</button>);}
}

stopPropagation 是没法阻止我们冒泡到 document 的。这是因为 React 对事件的处理都是冒泡在 document 在执行。但如果我们真的需要阻止这样的我们应该如何做呢?

  handleClick = (e) => {console.log('点击了');e.nativeEvent.stopImmediatePropagation();};

我们可以通过 nativeEvent 拿到原生事件,然后调用原生的stopImmediatePropagation来阻止document 上的事件。

最后,react 的合成了所有的事件是否是最优化呢?

在类 React 框架中有一个性能最优化的框架叫 Inferno. 它对事件的处理则是处理部分的事件作为合成事件,其余依旧为原生事件。源码在这里

8. 事件绑定传参问题

当我们想在一个事件传递多个参数的时候,这在 React 里也是非常的恶心。通常我们是用箭头函数或者直接用bind(this, params)。

而在这一点在 inferno 里就做的很棒,提供了一个linkEvent的接口,不仅解决了 params的问题,还能解决 this 的问题。

import { linkEvent, Component } from 'inferno';function handleClick(instance, event) {instance.setState({ data: event.target.value });
}class MyComponent extends Component {render () {

return <div><input type="text" onClick={ linkEvent(this, handleClick) } /><div>;

}
}

 

https://github.com/infernojs/inferno/blob/master/README.md#linkevent-package-inferno

9. render()

我敢打赌,99%的组件我们至少都需要用到props。那为什么React官方团队不能自动的每次把 props 和 state 都自动的注入到 render 方法里呢?像这样

render(props, state) {return <div></div>
}

10. 富交互应用

试想一下假设你需要实现一个类似微信读书的应用。上面有划词标记,划词备忘录,划词锚点,自定义选区编辑等大量需要跟DOM打交道的需求。这时候我们单纯用VM的方式其实非常难实现。无论如何我们都会多生命周期进行大量的DOM事件操作。这对于React来说,或者别的MVVM框架来说都并不是那么的擅长。

也就是说,任何需要我们必须大量操作DOM的需求,我们虽然都能实现,但是依旧是以一种很“不舒服”的方式进行实现。

11. 组件复用/逻辑复用

组件复用和逻辑复用,一直是React长久以来的探索。从最早的mixin -> HOC && Render props。一直都是React的软肋。

mixin就不说了,在中大型应用中使用mixin就是噩梦。天知道你会被谁注入了什么。

HOC是我们长久以来逻辑复用的最佳实践,然而大量使用HOC,导致我们项目无意间多出了许许多多无用层级的DOM,无意义的内存浪费。对于强迫症来说审查元素进行调试也是一件非常恶心的事情。

而render props也只是另一种围魏救赵的方式罢了,虽然也完成了我们组件复用和逻辑复用的使命,但一不小心我们就可能陷入了render callback hell。

而这些所有的问题,我们都可以在hook里被更完美更优雅的解决。

React已经非常优秀了,但它还再变得更好。

原文https://mp.weixin.qq.com/s/w1VEMNS2KC-hsyds1gvlnA 

这篇关于React中11个设计的瑕疵的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/508982

相关文章

全面解析HTML5中Checkbox标签

《全面解析HTML5中Checkbox标签》Checkbox是HTML5中非常重要的表单元素之一,通过合理使用其属性和样式自定义方法,可以为用户提供丰富多样的交互体验,这篇文章给大家介绍HTML5中C... 在html5中,Checkbox(复选框)是一种常用的表单元素,允许用户在一组选项中选择多个项目。本

HTML5 搜索框Search Box详解

《HTML5搜索框SearchBox详解》HTML5的搜索框是一个强大的工具,能够有效提升用户体验,通过结合自动补全功能和适当的样式,可以创建出既美观又实用的搜索界面,这篇文章给大家介绍HTML5... html5 搜索框(Search Box)详解搜索框是一个用于输入查询内容的控件,通常用于网站或应用程

CSS3中的字体及相关属性详解

《CSS3中的字体及相关属性详解》:本文主要介绍了CSS3中的字体及相关属性,详细内容请阅读本文,希望能对你有所帮助... 字体网页字体的三个来源:用户机器上安装的字体,放心使用。保存在第三方网站上的字体,例如Typekit和Google,可以link标签链接到你的页面上。保存在你自己Web服务器上的字

html 滚动条滚动过快会留下边框线的解决方案

《html滚动条滚动过快会留下边框线的解决方案》:本文主要介绍了html滚动条滚动过快会留下边框线的解决方案,解决方法很简单,详细内容请阅读本文,希望能对你有所帮助... 滚动条滚动过快时,会留下边框线但其实大部分时候是这样的,没有多出边框线的滚动条滚动过快时留下边框线的问题通常与滚动条样式和滚动行

MyBatis设计SQL返回布尔值(Boolean)的常见方法

《MyBatis设计SQL返回布尔值(Boolean)的常见方法》这篇文章主要为大家详细介绍了MyBatis设计SQL返回布尔值(Boolean)的几种常见方法,文中的示例代码讲解详细,感兴趣的小伙伴... 目录方案一:使用COUNT查询存在性(推荐)方案二:条件表达式直接返回布尔方案三:存在性检查(EXI

使用vscode搭建pywebview集成vue项目实践

《使用vscode搭建pywebview集成vue项目实践》:本文主要介绍使用vscode搭建pywebview集成vue项目实践,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录环境准备项目源码下载项目说明调试与生成可执行文件核心代码说明总结本节我们使用pythonpywebv

使用Python和Tkinter实现html标签去除工具

《使用Python和Tkinter实现html标签去除工具》本文介绍用Python和Tkinter开发的HTML标签去除工具,支持去除HTML标签、转义实体并输出纯文本,提供图形界面操作及复制功能,需... 目录html 标签去除工具功能介绍创作过程1. 技术选型2. 核心实现逻辑3. 用户体验增强如何运行

CSS 样式表的四种应用方式及css注释的应用小结

《CSS样式表的四种应用方式及css注释的应用小结》:本文主要介绍了CSS样式表的四种应用方式及css注释的应用小结,本文通过实例代码给大家介绍的非常详细,详细内容请阅读本文,希望能对你有所帮助... 一、外部 css(推荐方式)定义:将 CSS 代码保存为独立的 .css 文件,通过 <link> 标签

使用Vue-ECharts实现数据可视化图表功能

《使用Vue-ECharts实现数据可视化图表功能》在前端开发中,经常会遇到需要展示数据可视化的需求,比如柱状图、折线图、饼图等,这类需求不仅要求我们准确地将数据呈现出来,还需要兼顾美观与交互体验,所... 目录前言为什么选择 vue-ECharts?1. 基于 ECharts,功能强大2. 更符合 Vue

Vue中插槽slot的使用示例详解

《Vue中插槽slot的使用示例详解》:本文主要介绍Vue中插槽slot的使用示例详解,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录一、插槽是什么二、插槽分类2.1 匿名插槽2.2 具名插槽2.3 作用域插槽三、插槽的基本使用3.1 匿名插槽