”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > 前端开发 + 数据结构和算法:DSA 如何为您的 React 应用程序提供动力 ⚡

前端开发 + 数据结构和算法:DSA 如何为您的 React 应用程序提供动力 ⚡

发布于2024-11-03
浏览:304

专注于前端的面试通常根本不关心 DSA。

对于我们这些记得在学校/大学学习过 DSA 的人来说,所有的例子都感觉纯粹是算法(有充分的理由),但几乎没有任何例子或指导来说明我们每天使用的产品如何利用这个概念。

“我需要这个吗?”
你已经问过很多次这个问题了,不是吗? ?

以下是您今天可以在 React 应用程序中利用的一些数据结构! ?

目录

  1. 介绍
  2. 数组:状态管理的首选
  3. 对象和哈希图:标准化数据存储以提高效率
  4. 双向链表:使用上下文导航
  5. 堆栈:具有不可变行为的撤消/重做功能
  6. 队列:管理顺序 API 调用
  7. 树:渲染递归组件
  8. 图表:构建复杂的数据关系和导航
  9. 结论

相关阅读:

1. 数组?:状态管理的首选

数组在 React 中无处不在。如果您需要帮助了解 .map() 或 .filter() 的工作原理,您可能会太早看到这篇文章了!但不用担心,一旦您熟悉了这些数组方法,您就会发现它们对于渲染列表、管理组件状态和转换数据是多么重要。

2. 对象和哈希映射?️:标准化数据存储以提高效率

在 React 应用程序中,当您处理大量实体(例如用户或帖子)时,将数据规范化为对象(哈希图)可以使读取和更新更加高效。您无需使用深层嵌套结构,而是通过 ID 映射实体。

示例从具有 ID 的规范化存储中读取

const postsById = {
  1: { id: 1, title: 'First Post', content: 'Content of first post' },
  2: { id: 2, title: 'Second Post', content: 'Content of second post' }
};

const postIds = [1, 2];

function PostList() {
  return (
    
{postIds.map(id => ( ))}
); } function Post({ post }) { return (

{post.title}

{post.content}

); }

这种模式可以实现高效的数据访问,特别是对于需要快速更新或读取而无需重新渲染整个集合的大型数据集。

3. 双向链表?:带上下文的导航

当您需要上一个和下一个元素的上下文时,双向链表非常有用 - 想象一下导航照片库,其中每个图像都显示其相邻图像以供参考。我们不使用索引,而是将当前节点直接存储在组件状态中。

示例用于在具有上下文的元素之间导航的双向链表

class Node {
  constructor(value) {
    this.value = value;
    this.next = null;
    this.prev = null;
  }
}

class DoublyLinkedList {
  constructor() {
    this.head = null;
    this.tail = null;
  }

  add(value) {
    const newNode = new Node(value);
    if (!this.head) {
      this.head = newNode;
      this.tail = newNode;
    } else {
      this.tail.next = newNode;
      newNode.prev = this.tail;
      this.tail = newNode;
    }
  }
}

const imageList = new DoublyLinkedList();
imageList.add({ id: 1, src: 'image1.jpg', alt: 'First Image' });
imageList.add({ id: 2, src: 'image2.jpg', alt: 'Second Image' });
imageList.add({ id: 3, src: 'image3.jpg', alt: 'Third Image' });

function Gallery() {
  const [currentNode, setCurrentNode] = useState(imageList.head);

  return (
    
{currentNode.prev && ( {currentNode.prev.value.alt} )} {currentNode.value.alt} {currentNode.next && ( {currentNode.next.value.alt} )}
); }

在此 React 组件中:

  • 当前节点存储在状态中,并且UI根据是否有上一个或下一个节点进行更新。
  • 这些按钮使用户能够向前和向后导航列表,如果没有更多节点可移动到,则禁用。
  • 此结构通过周围元素的上下文模拟实时导航,通常用于轮播、媒体库或播放列表等 UI 组件中。

4. Stacks ?:具有不可变行为的撤消/重做功能

堆栈允许您使用后进先出 (LIFO) 逻辑有效地管理撤消/重做操作。通过使用不可变操作(concat、slice),我们可以确保状态保持不变。

示例使用不可变的推送和弹出撤消/重做

const [undoStack, setUndoStack] = useState([]);
const [redoStack, setRedoStack] = useState([]);
const [formState, setFormState] = useState({ name: '', email: '' });

const updateForm = (newState) => {
  setUndoStack(prev => prev.concat([formState]));  // Immutable push
  setRedoStack([]);  // Clear redo stack
  setFormState(newState);
};

const undo = () => {
  if (undoStack.length > 0) {
    const lastState = undoStack.at(-1);
    setUndoStack(prev => prev.slice(0, -1));  // Immutable pop
    setRedoStack(prev => prev.concat([formState]));  // Move current state to redo
    setFormState(lastState);
  }
};

const redo = () => {
  if (redoStack.length > 0) {
    const lastRedo = redoStack.at(-1);
    setRedoStack(prev => prev.slice(0, -1));  // Immutable pop
    setUndoStack(prev => prev.concat([formState]));  // Push current state to undo
    setFormState(lastRedo);
  }
};

5. 队列?:管理顺序 API 调用

队列以先进先出 (FIFO) 方式运行,非常适合确保以正确的顺序处理 API 调用或通知等任务。

示例API 调用排队

const [apiQueue, setApiQueue] = useState([]);

const enqueueApiCall = (apiCall) => {
  setApiQueue(prevQueue => prevQueue.concat([apiCall]));  // Immutable push
};

const processQueue = () => {
  if (apiQueue.length > 0) {
    const [nextCall, ...restQueue] = apiQueue;
    nextCall().finally(() => setApiQueue(restQueue));  // Immutable pop
  }
};

6. 树?:渲染递归组件

在处理嵌套组件时,React 中通常会使用树注释线程文件夹结构菜单

示例递归渲染评论树

const commentTree = {
  id: 1,
  text: "First comment",
  children: [
    { id: 2, text: "Reply to first comment", children: [] },
    { id: 3, text: "Another reply", children: [{ id: 4, text: "Nested reply" }] }
  ]
};

function Comment({ comment }) {
  return (
    

{comment.text}

{comment.children?.map(child => (
))}
); }

另一篇可能与您相关的热门帖子:

7. 图表?:构建复杂的数据关系和导航

示例1多个视图之间的路由
您可以将页面之间的路线表示为图表,从而确保 SPA 中灵活的导航路径。

const routesGraph = {
  home: ['about', 'contact'],
  about: ['home', 'team'],
  contact: ['home'],
};

function navigate(currentRoute, targetRoute) {
  if (routesGraph[currentRoute].includes(targetRoute)) {
    console.log(`Navigating from ${currentRoute} to ${targetRoute}`);
  } else {
    console.log(`Invalid route from ${currentRoute} to ${targetRoute}`);
  }
}

示例2用户关系建模
图表非常适合建模社交联系或多个实体互连的任何类型的关系。

const usersGraph = {
  user1: ['user2', 'user3'],
  user2: ['user1', 'user4'],
  user3: ['user1'],
  user4: ['user2']
};

function findConnections(userId) {
  return usersGraph[userId] || [];
}

console.log(findConnections('user1'));  // Outputs: ['user2', 'user3']

注意:我们使用图表来显示中间件中审阅者的依赖关系。

TL;DR——学校的教训得到了回报

这些 DSA 类在当时可能感觉很抽象,但数据结构正在为你周围的 React 世界提供动力。

对象、堆栈、队列、链表、树和图不仅仅是理论——它们是您每天构建的干净、高效和可扩展应用程序的支柱。

因此,下次您管理队列中的状态或处理复杂的 UI 逻辑时,请记住:您从学校起就一直在接受这方面的培训。 ?

让我知道您最常使用哪些数据结构!

版本声明 本文转载于:https://dev.to/jayantbh/frontend-dev-data-structures-algorithms-how-dsa-can-power-your-react-app-491a?1如有侵犯,请联系[email protected]删除
最新教程 更多>
  • 寻找经济实惠的同日格兰尼公寓(带 Pillar Build Granny Flats)
    寻找经济实惠的同日格兰尼公寓(带 Pillar Build Granny Flats)
    在 Pillar Build Granny Flats,我们为您提供祖母屋解决方案的精英服务,满足您的独特需求。无论是房主、承包商还是投资者,我们都可以帮助您在当天购买后院公寓,效果非常好,为您节省宝贵的时间,而且不用说,预算也很实惠。我们的祖母房建造者将在每一步工作,以确保您的项目以最精确和细心的...
    编程 发布于2024-11-05
  • 如何使用 botoith Google Colab 和 AWS 集成
    如何使用 botoith Google Colab 和 AWS 集成
    您有没有想过,在实施AWS Lambda时,想要一一确认代码的运行情况? 您可能认为在 AWS 控制台上实施很痛苦,因为您必须运行 Lambda 函数并且每次都会产生成本。 因此,我将向您展示您的担忧的解决方案。 它是通过 Google Colab 和 AWS 集成实现的。 步骤如下: ...
    编程 发布于2024-11-05
  • (高性能 Web 应用程序的要求
    (高性能 Web 应用程序的要求
    “高性能网络应用程序”或“前端”到底是什么? 自从 Internet Explorer 时代衰落以来,JavaScript 生态系统变得越来越强大,“前端”一词已成为高性能、现代 Web 客户端的代名词。这个“前端”世界的核心是 React。事实上,在前端开发中不使用 React 常常会让一个人看...
    编程 发布于2024-11-05
  • 如何将单个输入字段设置为分区输入?
    如何将单个输入字段设置为分区输入?
    将输入字段设置为分区输入有多种方法可用于创建一系列分区输入字段。一种方法利用“字母间距”来分隔单个输入字段内的字符。此外,“background-image”和“border-bottom”样式可以进一步增强多个输入字段的错觉。CSS Snippet以下 CSS 代码演示了如何创建所需的效果:#pa...
    编程 发布于2024-11-05
  • 用 Go 构建一个简单的负载均衡器
    用 Go 构建一个简单的负载均衡器
    负载均衡器在现代软件开发中至关重要。如果您曾经想知道如何在多个服务器之间分配请求,或者为什么某些网站即使在流量大的情况下也感觉更快,答案通常在于高效的负载平衡。 在这篇文章中,我们将使用 Go 中的循环算法构建一个简单的应用程序负载均衡器。这篇文章的目的是逐步了解负载均衡器的工作原理。 ...
    编程 发布于2024-11-05
  • 如何以超链接方式打开本地目录?
    如何以超链接方式打开本地目录?
    通过超链接导航本地目录尝试在链接交互时启动本地目录视图时,您可能会遇到限制。然而,有一个解决方案可以解决这个问题,并且可以在各种浏览器之间无缝工作。实现方法因为从 HTML 页面直接打开路径或启动浏览器是由于安全原因受到限制,更可行的方法是提供可下载的链接(.URL 或 .LNK)。推荐路径:.UR...
    编程 发布于2024-11-05
  • 为什么 Makefile 会抛出 Go 命令的权限被拒绝错误?
    为什么 Makefile 会抛出 Go 命令的权限被拒绝错误?
    运行 Go 时 Makefile 中出现权限被拒绝错误通过 Makefile 运行 Go 命令时可能会遇到“权限被拒绝”错误,即使你可以直接执行它们。这种差异是由于 GNU make 中的问题引起的。原因:当您的 PATH 上有一个目录包含名为“go.gnu”的子目录时,就会出现此错误。 ”例如,如...
    编程 发布于2024-11-05
  • parseInt 函数中 Radix 参数的意义是什么?
    parseInt 函数中 Radix 参数的意义是什么?
    parseInt 函数中 Radix 的作用parseInt 函数将字符串转换为整数。然而,它并不总是采用以 10 为基数的数字系统。要指定所需的基数,请使用基数参数。理解基数基数是指单个数字表示的值的数量。例如,十六进制的基数为 16,八进制的基数为 8,二进制的基数为 2。为什么使用基数?需要当...
    编程 发布于2024-11-05
  • 在空数据集上使用 MySQL 的 SUM 函数时如何返回“0”而不是 NULL?
    在空数据集上使用 MySQL 的 SUM 函数时如何返回“0”而不是 NULL?
    当不存在任何值时如何从 MySQL 的 SUM 函数中检索“0”MySQL 中的 SUM 函数提供了一种方便的方法来聚合数值价值观。但是,当查询期间没有找到匹配的行时,SUM 函数通常返回 NULL 值。对于某些用例,可能更需要返回“0”而不是 NULL。利用 COALESCE 解决问题此问题的解决...
    编程 发布于2024-11-05
  • 如何使用 JavaScript 将链接保留在同一选项卡中?
    如何使用 JavaScript 将链接保留在同一选项卡中?
    在同一选项卡和窗口中导航链接您可能会遇到想要在同一窗口和选项卡中打开链接的情况作为当前页面。但是,使用 window.open 函数通常会导致在新选项卡中打开链接。为了解决这个问题,您可以使用 name 属性,如下所示:window.open("https://www.youraddres...
    编程 发布于2024-11-05
  • 如何解决Python中的循环依赖?
    如何解决Python中的循环依赖?
    Python 中的循环依赖使用 Python 模块时遇到循环依赖可能是一个令人沮丧的问题。在这个特定场景中,我们有两个文件,node.py 和 path.py,分别包含 Node 和 Path 类。最初,path.py 使用 from node.py import * 导入 node.py。但是,在...
    编程 发布于2024-11-05
  • MariaDB 与 MySQL:开发人员需要了解什么
    MariaDB 与 MySQL:开发人员需要了解什么
    MariaDB 和 MySQL 是著名的开源 RDBMS,但尽管它们有着共同的历史,但它们在功能和性能方面却有所不同。本文快速强调了主要差异,帮助开发人员决定哪个数据库最适合他们的需求。 差异和示例 存储引擎,MariaDB 对 Aria 和 MyRocks 等引擎的扩展支持提供了比...
    编程 发布于2024-11-05
  • 为什么我的 Goroutine 递增变量会产生意外的结果?
    为什么我的 Goroutine 递增变量会产生意外的结果?
    这是编译器优化的结果吗?在此代码片段中,启动了一个 goroutine 并重复递增变量 i:package main import "time" func main() { i := 1 go func() { for { ...
    编程 发布于2024-11-05
  • 利用 AI 快速学习 Node.js - 第 4 天
    利用 AI 快速学习 Node.js - 第 4 天
    今天,借助ChatGPT继续学习Node.js,重点是异步编程。这是 Node.js 中最重要的概念之一,我很高兴能够开始掌握它。 理论 在 Node.js 中,异步编程因其非阻塞、事件驱动的架构而至关重要。这意味着文件读取、数据库查询或网络请求等操作在等待结果时不会阻塞其他代码的执行。 我们探索了...
    编程 发布于2024-11-05
  • Java 可以定义带有嵌入引号的字符串而不转义吗?
    Java 可以定义带有嵌入引号的字符串而不转义吗?
    揭开 Java 使用嵌入式引号定义字符串的替代方法在 Java 中处理字符串时,您常常会在文字中遇到大量引号,导致繁琐的转义和可读性挑战。虽然其他语言提供了处理这种情况的语法,但 Java 缺乏类似的选项。问题: Java 是否提供了另一种方法来定义带有嵌入引号的字符串而不诉诸转义?答案: 虽然 J...
    编程 发布于2024-11-05

免责声明: 提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发到邮箱:[email protected] 我们会第一时间内为您处理。

Copyright© 2022 湘ICP备2022001581号-3