React并发机制揭秘
.markdown-body { line-height: 1.75; font-weight: 400; font-size: 15px; overflow-x: hidden; color: rgba(43, 43, 43, 1); font-family: -apple-system, system-ui, BlinkMacSystemFont, Helvetica Neue, PingFang SC, Hiragino Sans GB, Microsoft YaHei, Arial, sans-serif; background-image: linear-gradient(90deg, rgba(159, 219, 252, 0.15) 3%, rgba(0, 0, 0, 0) 0), linear-gradient(1turn, rgba(159, 219, 252, 0.15) 3%, rgba(0, 0, 0, 0) 0); background-size: 20px 20px; background-position: center }
.markdown-body h1, .markdown-body h2, .markdown-body h3, .markdown-body h4, .markdown-body h5, .markdown-body h6 { padding: 30px 0; margin-top: 35px; margin-bottom: 10px; color: rgba(77, 208, 225, 1) }
.markdown-body h1 { font-size: 30px; text-align: center; position: relative; width: max-content; margin: 0 auto }
.markdown-body h1:before { position: absolute; content: ""; z-index: -1; top: -20px; height: 100%; width: 100px; left: 0; right: 0; margin: 0 auto; background: url("") center / 64px 64px no-repeat; opacity: 0.84 }
.markdown-body h1:after { position: absolute; content: ""; width: 150%; left: -25%; height: 50%; bottom: 12px; border-radius: 50%; background: linear-gradient(rgba(0, 0, 0, 0) 80%, rgba(77, 208, 225, 0.8)); opacity: 0.6; animation: 6s linear infinite h1animate }
@keyframes h1Animate { 0% { background-position: right bottom } 50% { background-position: right } 100% { background-position: right bottom } }
.markdown-body h2 { display: block; border-bottom: 4px solid rgba(77, 208, 225, 1); position: relative; font-size: 24px; padding: 12px 32px; margin: 30px 0 }
.markdown-body h2:before { width: 24px; height: 24px; left: 0; top: 0; margin: auto; background-size: 24px 24px; background-image: url("") }
.markdown-body h2:after, .markdown-body h2:before { content: ""; display: block; position: absolute; bottom: 0 }
.markdown-body h2:after { right: 0; width: 400px; height: 10px; border-top-right-radius: 24px; background: linear-gradient(90deg, rgba(255, 255, 255, 1), rgba(77, 208, 225, 1)); max-width: 50vw }
.markdown-body h3 { margin: 30px 0; font-size: 18px; position: relative; padding: 4px 32px; width: max-content }
.markdown-body h3:before { border-bottom: 2px solid rgba(77, 208, 225, 1); width: 100%; content: ""; display: block; height: 28px; position: absolute; left: 0; top: 0; bottom: -2px; margin: auto; background-size: 28px 28px; background-image: url(""); background-repeat: no-repeat; animation: 2s infinite alternate h3animationbefore }
@keyframes h3AnimationBefore { 0% { width: 28px } 25% { width: 100% } 50% { width: 100% } 100% { width: 100% } }
.markdown-body h3:after { content: ""; display: block; width: 28px; height: 28px; position: absolute; border: 2px solid rgba(77, 208, 225, 1); border-radius: 50%; right: -15px; top: 0; bottom: 0; margin: auto; background-size: 28px 28px; background-image: url(""); animation: 2s infinite alternate h3animationafter }
@keyframes h3AnimationAfter { 0% { } 10% { } 50% { transform: rotate(-1turn) } 100% { transform: rotate(-1turn) } }
.markdown-body h4 { font-size: 16px }
.markdown-body h5 { font-size: 15px }
.markdown-body h6 { margin-top: 5px }
.markdown-body p { line-height: inherit; margin: 22px 0; letter-spacing: 2px; font-size: 14px; word-spacing: 2px }
.markdown-body img { max-width: 80%; border-radius: 6px; display: block; margin: 20px auto !important; object-fit: contain; box-shadow: 0 0 16px rgba(110, 110, 110, 0.45) }
.markdown-body figcaption { display: block; font-size: 13px; color: rgba(43, 43, 43, 1) }
.markdown-body figcaption:before { content: ""; background-image: url(""); display: inline-block; width: 18px; height: 18px; background-size: 18px; background-repeat: no-repeat; background-position: center; margin-right: 5px; margin-bottom: -5px }
.markdown-body hr { border-top: 1px solid rgba(77, 208, 225, 1); border-right: none; border-bottom: none; border-left: none; margin-top: 32px; margin-bottom: 32px }
.markdown-body del { color: rgba(77, 208, 225, 1) }
.markdown-body code { border-radius: 2px; overflow-x: auto; background-color: rgba(77, 208, 225, 0.08); color: rgba(38, 198, 218, 1); padding: 0.195em 0.4em }
.markdown-body pre { font-family: Menlo, Monaco, Consolas, Courier New, monospace; overflow: auto; position: relative; line-height: 1.75; box-shadow: 0 0 8px rgba(110, 110, 110, 0.45); border-radius: 4px; margin: 16px }
.markdown-body pre:before { content: ""; display: block; height: 30px; width: 100%; margin-bottom: -7px; background: url("") 10px 10px / 40px no-repeat }
.markdown-body pre>code { font-size: 12px; padding: 15px 12px; margin: 0; word-break: normal; display: block; overflow-x: auto; color: rgba(51, 51, 51, 1); background: rgba(248, 248, 248, 1) }
.markdown-body a { color: rgba(77, 208, 225, 1); border-bottom: 1px solid rgba(77, 208, 225, 1); font-weight: 400; text-decoration: none; margin: 0 4px }
.markdown-body a:active, .markdown-body a:hover { background-color: rgba(77, 208, 225, 0.1) }
.markdown-body strong { color: rgba(38, 198, 218, 1) }
.markdown-body strong:before { content: "「" }
.markdown-body strong:after { content: "」" }
.markdown-body em { font-style: normal; color: rgba(77, 208, 225, 1); font-weight: 700 }
.markdown-body table { display: inline-block !important; font-size: 12px; width: auto; max-width: 100%; overflow: auto; border: 1px solid rgba(246, 246, 246, 1) }
.markdown-body thead { background: rgba(246, 246, 246, 1); color: rgba(0, 0, 0, 1); text-align: left }
.markdown-body tr:nth-child(2n) { background-color: rgba(77, 208, 225, 0.05) }
.markdown-body td, .markdown-body th { padding: 12px 7px; line-height: 24px }
.markdown-body td { min-width: 120px }
.markdown-body blockquote { margin: 2em 0; padding: 24px 32px; border-left: 4px solid rgba(38, 198, 218, 1); background: rgba(77, 208, 225, 0.15); position: relative }
.markdown-body blockquote:before { content: "❝"; top: 8px; left: 8px; color: rgba(77, 208, 225, 1); font-size: 30px; line-height: 1; font-weight: 700; position: absolute; opacity: 0.7 }
.markdown-body blockquote:after { content: "❞"; font-size: 30px; position: absolute; right: 8px; bottom: 0; color: rgba(77, 208, 225, 1); opacity: 0.7 }
.markdown-body blockquote p { color: rgba(89, 89, 89, 1); line-height: 2 }
.markdown-body ol, .markdown-body ul { color: rgba(89, 89, 89, 1); padding-left: 28px }
.markdown-body ol li, .markdown-body ul li { margin-bottom: 0; list-style: inherit }
.markdown-body ol li .task-list-item, .markdown-body ul li .task-list-item { list-style: none }
.markdown-body ol li .task-list-item ol, .markdown-body ol li .task-list-item ul, .markdown-body ul li .task-list-item ol, .markdown-body ul li .task-list-item ul { margin-top: 0 }
.markdown-body ol ol, .markdown-body ol ul, .markdown-body ul ol, .markdown-body ul ul { margin-top: 3px }
.markdown-body ol li { padding-left: 6px }
@media (max-width: 720px) { .markdown-body h1 { font-size: 24px } .markdown-body h2 { font-size: 20px } .markdown-body h3 { font-size: 18px } }.markdown-body pre, .markdown-body pre>code.hljs { color: rgba(169, 183, 198, 1); background: rgba(40, 43, 46, 1) }
.hljs-bullet, .hljs-literal, .hljs-number, .hljs-symbol { color: rgba(104, 151, 187, 1) }
.hljs-deletion, .hljs-keyword, .hljs-selector-tag { color: rgba(204, 120, 50, 1) }
.hljs-link, .hljs-template-variable, .hljs-variable { color: rgba(98, 151, 85, 1) }
.hljs-comment, .hljs-quote { color: rgba(128, 128, 128, 1) }
.hljs-meta { color: rgba(187, 181, 41, 1) }
.hljs-addition, .hljs-attribute, .hljs-string { color: rgba(106, 135, 89, 1) }
.hljs-section, .hljs-title, .hljs-type { color: rgba(255, 198, 109, 1) }
.hljs-name, .hljs-selector-class, .hljs-selector-id { color: rgba(232, 191, 106, 1) }
.hljs-emphasis { font-style: italic }
.hljs-strong { font-weight: 700 }
React 18.2.0 引入了一系列并发机制的新特性,旨在帮助各位开发者更好地控制和优化应用程序的性能和用户体验。
本文掌门人将深入探讨这些新特性,并通过实际应用示例展示如何利用它们构建更高效、更响应式的应用程序。
一、自动批处理
React 现在可以自动将多个状态更新操作批量处理,以减少不必要的渲染和提高性能。这意味着,当你在一个事件处理程序中连续调用多个状态更新函数时,React 会将它们合并为一次更新,从而减少渲染次数。
计数器
在下面这个案例中,当用户点击按钮时,handleClick
函数会连续调用三次 setCount
。由于 React 的自动批处理,这三次更新会被合并为一次更新,从而减少渲染次数。
import { useState } from 'react';
function Counter() {
const [count, setCount] = useState(0);
const handleClick = () => {
setCount(count + 1);
setCount(count + 2);
setCount(count + 3);
};
return (
<div>
<p>Count: {count}</p>
<button onClick={handleClick}>Increment by 3</button>
</div>
);
}
二、异步渲染
React现在可以支持异步渲染,这意味着 React 可以在渲染过程中暂停和恢复,从而允许其他任务(如用户输入或动画)优先执行。这有助于提高应用程序的响应性和性能。
异步数据获取
在下面这个案例中,AsyncComponent
组件在挂载时会异步获取数据。由于 React 的异步渲染,当数据还未获取到时,组件会显示 "Loading...",而不会阻塞其他任务的执行。
import { useState, useEffect } from 'react';
function AsyncComponent() {
const [data, setData] = useState(null);
useEffect(() => {
fetchData().then((fetchedData) => {
setData(fetchedData);
});
}, []);
if (data === null) {
return <p>Loading...</p>;
}
return <div>{data}</div>;
}
async function fetchData() {
// 模拟异步数据获取
await new Promise((resolve) => setTimeout(resolve, 1000));
return 'Async data';
}
三、useDeferredValue:延迟值更新
useDeferredValue
是一个新的 Hook,它允许你在渲染过程中延迟某些值的更新,直到下一次渲染。这对于实现响应式 UI 和避免不必要的渲染非常有用。
输入框防抖
在下面这个示例中,当用户在输入框中输入文本时,useDeferredValue
会延迟更新 deferredText
,直到 500 毫秒后的下一次渲染。这有助于提高应用程序的响应性和性能。
import { useState, useDeferredValue } from 'react';
function DebounceInput() {
const [text, setText] = useState('');
const deferredText = useDeferredValue(text, { timeoutMs: 500 });
const handleChange = (e) => {
setText(e.target.value);
};
return (
<div>
<input type="text" value={text} onChange={handleChange} />
<p>Debounced value: {deferredText}</p>
</div>
);
}
四、useTransition:创建过渡状态
useTransition
是一个新的 Hook,它允许你在状态更新过程中创建一个过渡(transition),从而实现更细粒度的更新控制。你可以使用这个 Hook 来延迟某些状态更新,直到过渡完成。这对于实现更好的用户体验和性能优化非常有用。
加载状态与过渡效果
在下面这个示例中,当用户点击按钮时,handleClick
函数会使用 startTransition
创建一个过渡,并在过渡期间延迟更新 count
。isPending
变量表示过渡是否正在进行中,当过渡正在进行时,会显示 "Loading..."。
import { useState, useTransition } from 'react';
function TransitionExample() {
const [count, setCount] = useState(0);
const [isPending, startTransition] = useTransition();
const handleClick = () => {
startTransition(() => {
setCount(count + 1);
});
};
return (
<div>
<p>Count: {count}</p>
<button onClick={handleClick}>Increment</button>
{isPending && <p>Loading...</p>}
</div>
);
}
五、Suspense 配置优化
React新版本引入了新的 Suspense 配置选项,如 timeoutMs
和 fallbackAfterSuspense
。这些选项允许你更精细地控制 Suspense 的行为,例如设置挂起时间和在挂起后显示备用内容。
分页加载与错误处理
在下面这个示例中,App
组件使用 Suspense
包裹了两个异步加载的页面组件 Page1
和 Page2
。通过设置 timeoutMs
为 1000 毫秒,我们可以在 1 秒后显示 fallbackAfterSuspense
属性指定的备用内容 "Loading with suspense..."。同时,当 Page2
加载失败时,Suspense 会捕获错误并显示默认的错误边界内容
import { Suspense } from 'react';
function App() {
return (
<Suspense
fallback={<div>Loading...</div>}
timeoutMs={1000}
fallbackAfterSuspense={<div>Loading with suspense...</div>}
>
<Page1 />
<Page2 />
</Suspense>
);
}
function Page1() {
// 模拟异步数据获取
return new Promise((resolve) => setTimeout(() => resolve(<div>Page 1</div>), 2000));
}
function Page2() {
// 模拟异步数据获取失败
return new Promise((_, reject) => setTimeout(() => reject(new Error('Error loading Page 2')), 2000));
}
六、startTransition:手动创建过渡
startTransition
是一个新的函数,它允许你在状态更新过程中创建一个过渡(transition)。你可以使用这个函数来延迟某些状态更新,直到过渡完成。这对于实现更好的用户体验和性能优化非常有用。
动画与过渡的协同
在下面这个示例中,当用户点击按钮时,handleClick
函数会使用 startTransition
创建一个过渡,并在过渡期间平滑地移动盒子。通过结合 CSS 动画和 React 过渡,可以实现更自然、更流畅的交互体验。
import { useState, startTransition } from 'react';
function AnimationTransitionExample() {
const [position, setPosition] = useState(0);
const handleClick = () => {
// 使用 startTransition 创建过渡
startTransition(() => {
setPosition(position + 100);
});
};
return (
<div style={{ position: 'relative' }}>
<div
style={{
position: 'absolute',
left: position,
transition: 'left 0.5s ease-out',
}}
>
Box
</div>
<button onClick={handleClick}>Move box</button>
</div>
);
}
七、useId 与 useSyncExternalStore:与外部数据源同步
这两个新的 Hooks 提供了更好的支持,以便在 React 组件和外部数据源之间同步数据。useId
用于生成唯一的 ID,而 useSyncExternalStore
用于在组件和外部数据源之间同步数据。
实时聊天应用
在下面这个示例中,ChatRoom
组件使用 useSyncExternalStore
从外部数据源(chatStore
)获取聊天消息列表。当外部数据源的数据发生变化时,组件会自动重新渲染。同时,每个 ChatMessage
组件都使用 useId
生成了一个唯一的 ID,以便在 DOM 中正确引用它们。
import { useId, useSyncExternalStore } from 'react';
function ChatMessage({ message }) {
const id = useId();
return <div id={`message-${id}`}>{message}</div>;
}
function ChatRoom() {
const messages = useSyncExternalStore(
() => chatStore.subscribe,
() => chatStore.getMessages()
);
return (
<div>
{messages.map((message) => (
<ChatMessage key={message.id} message={message.text} />
))}
</div>
);
}
const chatStore = {
messages: [],
subscribers: new Set(),
subscribe(callback) {
this.subscribers.add(callback);
return () => {
this.subscribers.delete(callback);
};
},
getMessages() {
return this.messages;
},
addMessage(message) {
this.messages.push(message);
this.subscribers.forEach((callback) => callback());
},
};
总结以及预测
React新版本的并发机制提供了一系列新特性,包括自动批处理、异步渲染、新的 Hooks 和 Suspense 配置等。这些特性旨在帮助开发者更好地控制和优化应用程序的性能和用户体验。通过这些新特性我看到了React在性能优化和用户体验方面的巨大潜力。
下面是掌门人对于React并发机制的一些预测,若是大家与我想法不一致,还请嘴下留情。
- 更细粒度的并发控制:未来的React可能会提供更细粒度的并发控制,允许开发者根据具体场景定制渲染策略,以实现更高的性能和更低的开销。
- 集成Web Workers:Web Workers为JavaScript提供了在单独线程中运行代码的能力,从而避免了主线程的阻塞。未来,React可能会更好地集成Web Workers,以实现更高效的并发处理。
- 更强大的Suspense功能:Suspense是React并发机制的核心组件之一,未来的版本可能会增强其功能,例如支持多个等待状态、更细粒度的错误处理等。
- 优化的数据流管理:React的上下文API和数据流管理一直是开发者关注的焦点。未来的React可能会进一步优化这些方面,以提供更高效、更简洁的数据流解决方案。
- 更好的服务器端渲染支持:随着服务器端渲染(SSR)在现代Web应用中的重要性日益增加,React可能会提供更好的SSR支持,包括更快的首屏加载时间、更好的SEO优化等。
从长远来看,React的发展将更加注重性能、可维护性和开发者体验,React目前的发展趋势应该是将自身打造为一个全栈底层的基座,同时,随着物联网设备的普及,React也可能会拓展其在该领域的应用。
当然,上述想法纯属瞎猜,大家听个乐呵就行。
React并发机制揭秘的更多相关文章
- (第二章)Java并发机制的底层实现原理
一.概述 Java代码在编译后会变成Java字节码,字节码被类加载器加载到JVM里,JVM执行字节码,最终需要转化为汇编指令在CPU上执行,Java中所使用的并发机制依赖于JVM的实现和CPU的指令. ...
- storm的并发机制
storm的并发机制 storm计算支持在多台机器上水平扩容,通过将计算切分为多个独立的tasks在集群上并发执行来实现. 一个task可以简单地理解:在集群某节点上运行的一个spout或者bolt实 ...
- 【Java基础】线程和并发机制
前言 在Java中,线程是一个很关键的名词,也是很高频使用的一种资源.那么它的概念是什么呢,是如何定义的,用法又有哪些呢?为何说Android里只有一个主线程呢,什么是工作线程呢.线程又存在并发,并发 ...
- Go语言并发机制初探
Go 语言相比Java等一个很大的优势就是可以方便地编写并发程序.Go 语言内置了 goroutine 机制,使用goroutine可以快速地开发并发程序, 更好的利用多核处理器资源.这篇文章学习 g ...
- Storm并发机制详解
本文可作为 <<Storm-分布式实时计算模式>>一书1.4节的读书笔记 在Storm中,一个task就可以理解为在集群中某个节点上运行的一个spout或者bolt实例. 记住 ...
- 第二十节: 深入理解并发机制以及解决方案(锁机制、EF自有机制、队列模式等)
一. 理解并发机制 1. 什么是并发,并发与多线程有什么关系? ①. 先从广义上来说,或者从实际场景上来说. 高并发通常是海量用户同时访问(比如:12306买票.淘宝的双十一抢购),如果把一个用户看做 ...
- 《Java并发编程的艺术》Java并发机制的底层实现原理(二)
Java并发机制的底层实现原理 1.volatile volatile相当于轻量级的synchronized,在并发编程中保证数据的可见性,使用 valotile 修饰的变量,其内存模型会增加一个 L ...
- python进阶(一) 多进程并发机制
python多进程并发机制: 这里使用了multprocessing.Pool进程池,来动态增加进程 #coding=utf-8 from multiprocessing import Pool im ...
- 那些年读过的书《Java并发编程的艺术》一、并发编程的挑战和并发机制的底层实现原理
一.并发编程的挑战 1.上下文切换 (1)上下文切换的问题 在处理器上提供了强大的并行性就使得程序的并发成为了可能.处理器通过给不同的线程分配不同的时间片以实现线程执行的自动调度和切换,实现了程序并行 ...
- Java并发机制和底层实现原理
Java代码在编译后会变成Java字节码,字节码被类加载器加载到JVM里,JVM执行字节码转化为汇编指令在CPU上执行.Java中的并发机制依赖于JVM的实现和CPU的指令. Java语言规范第三版中 ...
随机推荐
- Vulnhub-kioptix2014靶机getshell及提权
靶机搭建 点击扫描虚拟机 然后扫描文件夹即可 信息收集 扫描ip nmap扫描得到目标靶机ip nmap -sn 192.168.108.0/24 故 攻击机:192.168.108.130 目标靶机 ...
- Markdown 编写技巧汇总(二)
继续上篇汇总 附-上篇汇总,接着做更加高级一点的应用技巧. [1]列表与引言嵌套 两者嵌套,如: > 我是一行文本 > 1. 文本1 > 2. 文本2 > 1. 文本 > ...
- RCE_STUDY
概念 RCE(Remote code execution)远程代码执行漏洞,RCE又分命令执行和代码执行. RCE-远程代码执行:远程执行PHP代码 RCE-远程命令执行:远程执行Linux或者Win ...
- 【P2】MARS使用/MIPS汇编
课上 T1 在n位数中删除N个数使剩下的(n-N)位数最大 写得似乎过于谨慎而慢了,没出现寄存器打错的问题,一遍过了 T2 拆分数字 将输入整数N拆分为几个数相加的形式,按拆分项数降序排列,每项按数字 ...
- Pwnable_orw
题源 题解 保护 只开启了栈保护 分析 进入ida分析 main函数如下 seccomp (Secure Computing Mode)是一种 Linux 内核安全机制,它可以 限制进程可执行的系统调 ...
- ORACLE SQL中执行先后次序的问题
分享一个经验 需求:Oracle中,根据COST优先级取最优先的一条记录脚本: select ... from ... where ... and rownum=1 order by cost 实际不 ...
- 使用Python完成设备巡检
在企业网络中,设备巡检是保持网络稳定性和安全性的核心任务.无论是路由器.交换机,还是防火墙和服务器等设备,都需要定期进行巡检,以确保网络设施的正常运行.然而,传统的设备巡检通常是通过手动登录设备.查看 ...
- OpenHarmony 开源鸿蒙北向开发——hdc工具安装
hdc(OpenHarmony Device Connector)是为开发人员提供的用于设备连接调试的命令行工具,该工具需支持部署在 Windows/Linux/Mac 等系统上与 OpenHar ...
- 基础指令:sudo提权、通配符、特殊符号、stat命令、id命令、正则表达式
目录 4.4 sudo提权 授权的两种方法 4.4 通配符-查文件 4.5 特殊符号 4.6 stat输出文件的详细内容 4.7 id命令查看用户基本信息 4.8 正则表达式 4.8.1 符号 ^ 4 ...
- linux重启后,启动docker和docker对应的服务
我的项目部署在docker上,linux关闭之后,项目要重启,在此做一个记录1.启动linux之后,执行docker images或者docker ps,如果出现下面的错误Cannot connect ...