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中的Microdata与历史记录管理详解

《HTML5中的Microdata与历史记录管理详解》Microdata作为HTML5新增的一个特性,它允许开发者在HTML文档中添加更多的语义信息,以便于搜索引擎和浏览器更好地理解页面内容,本文将探... 目录html5中的Mijscrodata与历史记录管理背景简介html5中的Microdata使用M

html5的响应式布局的方法示例详解

《html5的响应式布局的方法示例详解》:本文主要介绍了HTML5中使用媒体查询和Flexbox进行响应式布局的方法,简要介绍了CSSGrid布局的基础知识和如何实现自动换行的网格布局,详细内容请阅读本文,希望能对你有所帮助... 一 使用媒体查询响应式布局        使用的参数@media这是常用的

HTML5表格语法格式详解

《HTML5表格语法格式详解》在HTML语法中,表格主要通过table、tr和td3个标签构成,本文通过实例代码讲解HTML5表格语法格式,感兴趣的朋友一起看看吧... 目录一、表格1.表格语法格式2.表格属性 3.例子二、不规则表格1.跨行2.跨列3.例子一、表格在html语法中,表格主要通过< tab

Vue3组件中getCurrentInstance()获取App实例,但是返回null的解决方案

《Vue3组件中getCurrentInstance()获取App实例,但是返回null的解决方案》:本文主要介绍Vue3组件中getCurrentInstance()获取App实例,但是返回nu... 目录vue3组件中getCurrentInstajavascriptnce()获取App实例,但是返回n

JS+HTML实现在线图片水印添加工具

《JS+HTML实现在线图片水印添加工具》在社交媒体和内容创作日益频繁的今天,如何保护原创内容、展示品牌身份成了一个不得不面对的问题,本文将实现一个完全基于HTML+CSS构建的现代化图片水印在线工具... 目录概述功能亮点使用方法技术解析延伸思考运行效果项目源码下载总结概述在社交媒体和内容创作日益频繁的

前端CSS Grid 布局示例详解

《前端CSSGrid布局示例详解》CSSGrid是一种二维布局系统,可以同时控制行和列,相比Flex(一维布局),更适合用在整体页面布局或复杂模块结构中,:本文主要介绍前端CSSGri... 目录css Grid 布局详解(通俗易懂版)一、概述二、基础概念三、创建 Grid 容器四、定义网格行和列五、设置行

前端下载文件时如何后端返回的文件流一些常见方法

《前端下载文件时如何后端返回的文件流一些常见方法》:本文主要介绍前端下载文件时如何后端返回的文件流一些常见方法,包括使用Blob和URL.createObjectURL创建下载链接,以及处理带有C... 目录1. 使用 Blob 和 URL.createObjectURL 创建下载链接例子:使用 Blob

Vuex Actions多参数传递的解决方案

《VuexActions多参数传递的解决方案》在Vuex中,actions的设计默认只支持单个参数传递,这有时会限制我们的使用场景,下面我将详细介绍几种处理多参数传递的解决方案,从基础到高级,... 目录一、对象封装法(推荐)二、参数解构法三、柯里化函数法四、Payload 工厂函数五、TypeScript

Vue3使用router,params传参为空问题

《Vue3使用router,params传参为空问题》:本文主要介绍Vue3使用router,params传参为空问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录vue3使用China编程router,params传参为空1.使用query方式传参2.使用 Histo

CSS Padding 和 Margin 区别全解析

《CSSPadding和Margin区别全解析》CSS中的padding和margin是两个非常基础且重要的属性,它们用于控制元素周围的空白区域,本文将详细介绍padding和... 目录css Padding 和 Margin 全解析1. Padding: 内边距2. Margin: 外边距3. Padd