上下文、Redux 还是组合?
这篇文章最初发布于2023年2月23日@我的博客页面
我是受到最近科技公司裁员影响的开发人员之一。所以,我开始用 React 面试前端职位。
在其中一家公司,我在反应中遇到了一个经典的道具钻孔问题,
并被要求解决它。为了简单起见,给出的问题就像
这个:
export default function App() { const [user, setUser] = React.useState(null); const handleLogin = () => setUser(userDetails); return ( <div className="App"> Company Logo <div> {user ? ( <Dashboard user={user} /> ) : ( <button onClick={handleLogin}>Login</button> )} </div> </div> ); } function Dashboard({ user }) { return ( <div> <DashboardNav user={user} /> </div> ); } function DashboardNav({ user }) { return ( <div> <WelcomeUser user={user} /> <UserRole user={user} /> </div> ); } function WelcomeUser({ user }) { return <div>Welcome {user.name}</div>; } function UserRole({ user }) { return <div>Role {user.role}</div>; }
正如您所观察到的,我们将 user 属性从 App 组件传递到
子组件WelcomeUser 和UserRole。中间组件 Dashboard 和 DashboardNav 只是转发 props,并没有真正使用它。
这是 React 中的一个经典 prop 钻孔 问题。
有趣的是,面试官要求通过 React Context API 来解决问题
或使用 Redux。
通过 React 上下文解决
使用 context API 来解决这个问题,代码如下。
const UserContext = React.createContext(undefined); export default function App() { const [user, setUser] = React.useState(null); const handleLogin = () => setUser(userDetails); return ( <div className="App"> Company Logo: Context <div> {user ? ( <UserContext.Provider value={user}> <Dashboard /> </UserContext.Provider> ) : ( <button onClick={handleLogin}>Login</button> )} </div> </div> ); } function Dashboard() { return ( <div> <DashboardNav /> </div> ); } function DashboardNav() { return ( <div> <WelcomeUser /> <UserRole /> </div> ); } function WelcomeUser() { const user = React.useContext(UserContext); return <div>Welcome {user.name}</div>; } function UserRole() { const user = React.useContext(UserContext); return <div>Role {user.role}</div>; }
我们正在创建 UserContext 并使用 Provider 包装仪表板,以便
我们可以将我们想要的 props 传递给深度嵌套的子组件。这个解决方案
有效。
通过 Redux 解决
所以,如果我们要走经典的 redux 路线,我们需要创建一个类似的
使用单个全局存储来构造和包装所有内容,其中包含用户
目的。
解决方案代码将包含大量样板文件,因为我们使用 redux 来
解决一个简单的问题。
我只是给出了下面代码的要点,但如果你真的想探索完整的
代码,在这里:用 redux 解决。
export default function App() { return ( <Provider store={store}> <ReduxConnectedApp /> </Provider> ); } function ReduxApp({ user, setUser }) { const handleLogin = () => setUser(userDetails); return ( <div className="App"> Company Logo: Redux <div> {user ? <Dashboard /> : <button onClick={handleLogin}>Login</button>} </div> </div> ); } function Dashboard() { return ( <div> <DashboardNav /> </div> ); } function DashboardNav() { return ( <div> <ConnectedWelcomeUser /> <ConnectedUserRole /> </div> ); } function WelcomeUser({ user }) { return <div>Welcome {user.name}</div>; } const mapStateToPropsWelcomeUser = (state) => ({ user: state }); const ConnectedWelcomeUser = connect(mapStateToPropsWelcomeUser)(WelcomeUser); function UserRole({ user }) { return <div>Role {user.role}</div>; } const mapStateToPropsUserRole = (state) => ({ user: state }); const ConnectedUserRole = connect(mapStateToPropsUserRole)(UserRole);
我们已经连接了需要访问全局状态的组件
存储在redux中。
我是怎么解决的
阅读了有关反应组合的内容后,我通过制作解决了问题
使用 Children 道具,看起来像这样
export default function AppSolution() { const [user, setUser] = React.useState(null); const handleLogin = () => setUser(userDetails); return ( <div className="App"> Company Logo <div> {user ? ( <Dashboard> <DashboardNav> <WelcomeUser user={user} /> <UserRole user={user} /> </DashboardNav> </Dashboard> ) : ( <button onClick={handleLogin}>Login</button> )} </div> </div> ); } function Dashboard({ children }) { return <div>{children}</div>; } function DashboardNav({ children }) { return <div>{children}</div>; } function WelcomeUser({ user }) { return <div>Welcome {user.name}</div>; } function UserRole({ user }) { return <div>Role {user.role}</div>; }
如果你仔细想想,这是解决这个问题的简单方法,无需介绍
任何复杂性,如 createContext 或 react-redux。我们还获得其他好处,例如
作为
- 将来,如果我们在 Dashboard 中引入任何状态并对其进行操作,我们的 DashboardNav 永远不会重新渲染。
- 通过提供道具仅提供给所需的组件,我们拥有良好的可见性 用户的所有消费者,无需在组件(文件)之间导航 寻找他们。
这种模式并不新鲜,并且已经在 React 社区中讨论过。一个这样好的演练是在 React 中使用组合来避免“Prop Drilling”
结论
但是,我收到面试官的反馈,是这样的
受访者没有正确理解问题,无法提供预期的解决方案。
我猜原因要么是面试官没有意识到这种模式,要么是我选择以一种没有被要求的方式解决问题。
话虽如此,我现在有动力写更多关于 React 中有趣的模式,希望它能吸引更广泛的受众。
分享这篇文章让我知道您的想法。
参考文献
- Codesandbox 解决方案
- 使用上下文之前 - React 文档
- 在 React 中使用组合来避免“Prop Drilling”
- 博客答案:React 渲染行为(大部分)完整指南
- 博客回答:为什么 React Context 不是“状态管理”工具(以及为什么它不取代 Redux)
以上是上下文、Redux 还是组合?的详细内容。更多信息请关注PHP中文网其他相关文章!

热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

Video Face Swap
使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热门文章

热工具

记事本++7.3.1
好用且免费的代码编辑器

SublimeText3汉化版
中文版,非常好用

禅工作室 13.0.1
功能强大的PHP集成开发环境

Dreamweaver CS6
视觉化网页开发工具

SublimeText3 Mac版
神级代码编辑软件(SublimeText3)

JavaScript是现代Web开发的基石,它的主要功能包括事件驱动编程、动态内容生成和异步编程。1)事件驱动编程允许网页根据用户操作动态变化。2)动态内容生成使得页面内容可以根据条件调整。3)异步编程确保用户界面不被阻塞。JavaScript广泛应用于网页交互、单页面应用和服务器端开发,极大地提升了用户体验和跨平台开发的灵活性。

JavaScript的最新趋势包括TypeScript的崛起、现代框架和库的流行以及WebAssembly的应用。未来前景涵盖更强大的类型系统、服务器端JavaScript的发展、人工智能和机器学习的扩展以及物联网和边缘计算的潜力。

不同JavaScript引擎在解析和执行JavaScript代码时,效果会有所不同,因为每个引擎的实现原理和优化策略各有差异。1.词法分析:将源码转换为词法单元。2.语法分析:生成抽象语法树。3.优化和编译:通过JIT编译器生成机器码。4.执行:运行机器码。V8引擎通过即时编译和隐藏类优化,SpiderMonkey使用类型推断系统,导致在相同代码上的性能表现不同。

JavaScript是现代Web开发的核心语言,因其多样性和灵活性而广泛应用。1)前端开发:通过DOM操作和现代框架(如React、Vue.js、Angular)构建动态网页和单页面应用。2)服务器端开发:Node.js利用非阻塞I/O模型处理高并发和实时应用。3)移动和桌面应用开发:通过ReactNative和Electron实现跨平台开发,提高开发效率。

Python更适合初学者,学习曲线平缓,语法简洁;JavaScript适合前端开发,学习曲线较陡,语法灵活。1.Python语法直观,适用于数据科学和后端开发。2.JavaScript灵活,广泛用于前端和服务器端编程。

本文展示了与许可证确保的后端的前端集成,并使用Next.js构建功能性Edtech SaaS应用程序。 前端获取用户权限以控制UI的可见性并确保API要求遵守角色库

从C/C 转向JavaScript需要适应动态类型、垃圾回收和异步编程等特点。1)C/C 是静态类型语言,需手动管理内存,而JavaScript是动态类型,垃圾回收自动处理。2)C/C 需编译成机器码,JavaScript则为解释型语言。3)JavaScript引入闭包、原型链和Promise等概念,增强了灵活性和异步编程能力。

我使用您的日常技术工具构建了功能性的多租户SaaS应用程序(一个Edtech应用程序),您可以做同样的事情。 首先,什么是多租户SaaS应用程序? 多租户SaaS应用程序可让您从唱歌中为多个客户提供服务
