【react 全家桶】列表 Key
创始人
2025-05-30 08:46:56
0

文章目录

  • 06 【列表 & Key】
    • 1.列表
      • 1.1 渲染多个组件
      • 1.2 基础列表组件
    • 2.key
      • 2.1 基本使用
      • 2.2 用 key 提取组件
      • 2.3 key 值在兄弟节点之间必须唯一
      • 2.4 在 JSX 中嵌入 map()
    • 3.diff算法
      • 3.1 什么是虚拟 DOM ?
      • 3.2 diff 算法
      • 3.3 用index作为key可能会引发的问题

06 【列表 & Key】

首先,让我们看下在 Javascript 中如何转化列表。

如下代码,我们使用 map() 函数让数组中的每一项变双倍,然后我们得到了一个新的列表 doubled 并打印出来:

const numbers = [1, 2, 3, 4, 5];
const doubled = numbers.map((number) => number * 2);console.log(doubled);

代码打印出 [2, 4, 6, 8, 10]

在 React 中,把数组转化为元素列表的过程是相似的。

1.列表

1.1 渲染多个组件

你可以通过使用 {} 在 JSX 内构建一个元素集合。

下面,我们使用 Javascript 中的 map() 方法来遍历 numbers 数组。将数组中的每个元素变成

  • 标签,最后我们将得到的数组赋值给 listItems

    const numbers = [1, 2, 3, 4, 5];
    const listItems = numbers.map((number) =>  
  • {number}
  • );

    然后,我们可以将整个 listItems 插入到

      元素中:

        {listItems}

      在 CodePen 上尝试

      const numbers = [1, 2, 3, 4, 5];
      const listItems = numbers.map((numbers) =>
    • {numbers}
    • );const root = ReactDOM.createRoot(document.getElementById('root')); root.render(
        {listItems}
      );

      这段代码生成了一个 1 到 5 的项目符号列表。

      image-20221024211657792

      1.2 基础列表组件

      通常你需要在一个组件中渲染列表。

      我们可以把前面的例子重构成一个组件,这个组件接收 numbers 数组作为参数并输出一个元素列表。

      function NumberList(props) {const numbers = props.numbers;const listItems = numbers.map((number) =>
    • {number}
    • );return (
        {listItems}
      ); }const numbers = [1, 2, 3, 4, 5]; const root = ReactDOM.createRoot(document.getElementById('root')); root.render(numbers} />);

      当我们运行这段代码,将会看到一个警告 a key should be provided for list items,意思是当你创建一个元素时,必须包括一个特殊的 key 属性。我们将在下一节讨论这是为什么。

      让我们来给每个列表元素分配一个 key 属性来解决上面的那个警告:

      function NumberList(props) {const numbers = props.numbers;const listItems = numbers.map((number) =>
    • number.toString()}>{number}
    • );return (
        {listItems}
      ); }const numbers = [1, 2, 3, 4, 5];const root = ReactDOM.createRoot(document.getElementById('root')); root.render( numbers} />);

      在 CodePen 上尝试

      image-20221024211835947

      2.key

      2.1 基本使用

      key 帮助 React 识别哪些元素改变了,比如被添加或删除。因此你应当给数组中的每一个元素赋予一个确定的标识。

      const numbers = [1, 2, 3, 4, 5];
      const listItems = numbers.map((number) =>
    • number.toString()}>{number}
    • );

      一个元素的 key 最好是这个元素在列表中拥有的一个独一无二的字符串。通常,我们使用数据中的 id 来作为元素的 key:

      const todoItems = todos.map((todo) =>
    • todo.id}>{todo.text}
    • );

      当元素没有确定 id 的时候,万不得已你可以使用元素索引 index 作为 key:

      const todoItems = todos.map((todo, index) =>// Only do this if items have no stable IDs
    • index}>{todo.text}
    • );

      如果列表项目的顺序可能会变化,我们不建议使用索引来用作 key 值,因为这样做会导致性能变差,还可能引起组件状态的问题。可以看看 Robin Pokorny 的深度解析使用索引作为 key 的负面影响这一篇文章。如果你选择不指定显式的 key 值,那么 React 将默认使用索引用作为列表项目的 key 值。

      要是你有兴趣了解更多的话,这里有一篇文章深入解析为什么 key 是必须的可以参考。

      2.2 用 key 提取组件

      元素的 key 只有放在就近的数组上下文中才有意义。

      比方说,如果你提取出一个 ListItem 组件,你应该把 key 保留在数组中的这个 元素上,而不是放在 ListItem 组件中的

    • 元素上。

      例子:不正确的使用 key 的方式

      function ListItem(props) {const value = props.value;return (// 错误!你不需要在这里指定 key:
    • value.toString()}>{value}
    • ); }function NumberList(props) {const numbers = props.numbers;const listItems = numbers.map((number) =>// 错误!元素的 key 应该在这里指定:number} />);return (
        {listItems}
      ); }

      例子:正确的使用 key 的方式

      function ListItem(props) {// 正确!这里不需要指定 key:return 
    • {props.value}
    • ; }function NumberList(props) {const numbers = props.numbers;const listItems = numbers.map((number) =>// 正确!key 应该在数组的上下文中被指定number.toString()} value={number} />);return (
        {listItems}
      ); }

      在 CodePen 上尝试

      一个好的经验法则是:在 map() 方法中的元素需要设置 key 属性。

      2.3 key 值在兄弟节点之间必须唯一

      数组元素中使用的 key 在其兄弟节点之间应该是独一无二的。然而,它们不需要是全局唯一的。当我们生成两个不同的数组时,我们可以使用相同的 key 值:

      function Blog(props) {const sidebar = (
        {props.posts.map((post) =>
      • post.id}>{post.title}
      • )}
      );const content = props.posts.map((post) =>
      post.id}>

      {post.title}

      {post.content}

      );return (
      {sidebar}
      {content}
      ); }const posts = [{id: 1, title: 'Hello World', content: 'Welcome to learning React!'},{id: 2, title: 'Installation', content: 'You can install React from npm.'} ];const root = ReactDOM.createRoot(document.getElementById('root')); root.render(posts} />);

      在 CodePen 上尝试

      key 会传递信息给 React ,但不会传递给你的组件。如果你的组件中需要使用 key 属性的值,请用其他属性名显式传递这个值:

      const content = posts.map((post) =>post.id}id={post.id}title={post.title} />
      );
      

      上面例子中,Post 组件可以读出 props.id,但是不能读出 props.key

      2.4 在 JSX 中嵌入 map()

      在上面的例子中,我们声明了一个单独的 listItems 变量并将其包含在 JSX 中:

      function NumberList(props) {const numbers = props.numbers;const listItems = numbers.map((number) =>number.toString()}value={number} />);return (
        {listItems}
      ); }

      JSX 允许在大括号中嵌入任何表达式,所以我们可以内联 map() 返回的结果:

      function NumberList(props) {const numbers = props.numbers;return (
        {numbers.map((number) =>number.toString()}value={number} />)}
      ); }

      在 CodePen 上尝试

      这么做有时可以使你的代码更清晰,但有时这种风格也会被滥用。就像在 JavaScript 中一样,何时需要为了可读性提取出一个变量,这完全取决于你。但请记住,如果一个 map() 嵌套了太多层级,那可能就是你提取组件的一个好时机。

      3.diff算法

      3.1 什么是虚拟 DOM ?

      在谈 diff 算法之前,我们需要先了解虚拟 DOM 。它是一种编程概念,在这个概念里,以一种虚拟的表现形式被保存在内存中。在 React 中,render 执行的结果得到的并不是真正的 DOM 节点,而是 JavaScript 对象

      虚拟 DOM 只保留了真实 DOM 节点的一些基本属性,和节点之间的层次关系,它相当于建立在 JavaScript 和 DOM 之间的一层“缓存”

      hello world!

      上面的这段代码会转化可以转化为虚拟 DOM 结构

      {tag: "div",props: {class: "hello"},children: [{tag: "span",props: {},children: ["hello world!"]}]
      }
      

      其中对于一个节点必备的三个属性 tag,props,children

      • tag 指定元素的标签类型,如“lidiv
      • props 指定元素身上的属性,如 classstyle,自定义属性
      • children 指定元素是否有子节点,参数以数组形式传入

      而我们在 render 中编写的 JSX 代码就是一种虚拟 DOM 结构。

      3.2 diff 算法

      每个组件中的每个标签都会有一个key,不过有的必须显示的指定,有的可以隐藏。

      如果生成的render出来后就不会改变里面的内容,那么你不需要指定key(不指定key时,React也会生成一个默认的标识),或者将index作为key,只要key不重复即可。

      但是如果你的标签是动态的,是有可能刷新的,就必须显示的指定key。使用map进行遍历的时候就必须指定Key:

      this.state.num.map((n,index)=>{return 
      index} >新闻{n}
      })

      这个地方虽然显示的指定了key,但是官网并不推荐使用Index作为Key去使用

      这样会很有可能会有效率上的问题

      举个例子:

      在一个组件中,我们先创建了两个对象,通过循环的方式放入< li>标签中,此时key使用的是index。

      person:[{id:1,name:"张三",age:18},{id:2,name:"李四",age:19}
      ]this.state.person.map((preson,index)=>{return  
    • index}>{preson.name}
    • })

      如下图展现在页面中:

      image-20221024225054061

      此时,我们想在点击按钮之后动态的添加一个对象,并且放入到li标签中,在重新渲染到页面中。

      我们通过修改State来控制对象的添加。

      
      addObject = () =>{let {person} = this.state;const p = {id:(person.length+1),name:"王五",age:20};this.setState({person:[p,...person]});
      }
      

      如下动图所示:

      addObject

      这样看,虽然完成了功能。但是其实存在效率上的问题, 我们先来看一下两个前后组件状态的变化:

      image-20221024225208300

      我们发现,组件第一个变成了王五,张三和李四都移下去了。因为我们使用Index作为Key,这三个标签的key也就发生了改变【张三原本的key是0,现在变成了1,李四的key原本是1,现在变成了2,王五变成了0】

      在组件更新状态重新渲染的时候,就出现了问题:

      因为react是通过key来比较组件标签是否一致的,拿这个案例来说:

      首先,状态更新导致组件标签更新,react根据Key,判断旧的虚拟DOM和新的虚拟DOM是否一致

      key = 0 的时候 旧的虚拟DOM 内容是张三 新的虚拟DOM为王五 ,react认为内容改变,从而重新创建新的真实DOM.

      key = 1 的时候 旧的虚拟DOM 内容是李四,新的虚拟DOM为张三,react认为内容改变,从而重新创建新的真实DOM

      key = 2 的时候 旧的虚拟DOM没有,创建新的真实DOM

      这样原本有两个虚拟DOM可以复用,但都没有进行复用,完完全全的都是新创建的;这就导致效率极大的降低。

      其实这是因为我们将新创建的对象放在了首位,如果放在最后其实是没有问题的,但是因为官方并不推荐使用Index作为key值,我们推荐使用id作为key值。从而完全避免这样的情况。

      3.3 用index作为key可能会引发的问题

      key不需要全局唯一,只需在当前列表中唯一即可。元素的key最好是固定的,这里直接举个反例,有些场景我们会使用元素的索引为key像这种:

      const students = ['孙悟空', '猪八戒', '沙和尚'];
      const ele = 
        {students.map((item, index) =>
      • {item}
      • )}

      上例中,我使用了元素的索引(index)作为key来使用,但这有什么用吗?没用!因为index是根据元素位置的改变而改变的,当我们在前边插入一个新元素时,所有元素的顺序都会一起改变,那么它和React中按顺序比较有什么区别吗?没有区别!而且还麻烦了,唯一的作用就是去除了警告。所以我们开发的时候偶尔也会使用索引作为key,但前提是元素的顺序不会发生变化,除此之外不要用索引做key。

      1. 若对数据进行:逆序添加、逆序删除等破坏
        顺序操作:会产生没有必要的真实DOM更新 界面效果没问题,但效率低。

      2. 如果结构中还包含输入类的DOM:会产生错误DOM更新 界面有问题。

      3. 注意! 如果不存在对数据的逆序添加、逆序删除等破坏顺序操作,仅用于渲染列表用于展示,使用index作为key是没有问题的。

      开发如何选择key?

      最好使用每一条数据的唯一标识作为key 比如id,手机号,身份证号

      如果确定只是简单的展示数据,用Index也是可以的

      而这个判断key的比较规则就是Diff算法

      Diff算法其实就是react生成的新虚拟DOM和以前的旧虚拟DOM的比较规则:

      • 如果旧的虚拟DOM中找到了与新虚拟DOM相同的key:
        • 如果内容没有变化,就直接只用之前旧的真实DOM
        • 如果内容发生了变化,就生成新的真实DOM
      • 如果旧的虚拟DOM中没有找到了与新虚拟DOM相同的key:
        • 根据数据创建新的真实的DOM,随后渲染到页面上
  • 相关内容

    热门资讯

    中证A500ETF摩根(560... 8月22日,截止午间收盘,中证A500ETF摩根(560530)涨1.19%,报1.106元,成交额...
    A500ETF易方达(1593... 8月22日,截止午间收盘,A500ETF易方达(159361)涨1.28%,报1.104元,成交额1...
    何小鹏斥资约2.5亿港元增持小... 每经记者|孙磊    每经编辑|裴健如 8月21日晚间,小鹏汽车发布公告称,公司联...
    中证500ETF基金(1593... 8月22日,截止午间收盘,中证500ETF基金(159337)涨0.94%,报1.509元,成交额2...
    中证A500ETF华安(159... 8月22日,截止午间收盘,中证A500ETF华安(159359)涨1.15%,报1.139元,成交额...
    科创AIETF(588790)... 8月22日,截止午间收盘,科创AIETF(588790)涨4.83%,报0.760元,成交额6.98...
    创业板50ETF嘉实(1593... 8月22日,截止午间收盘,创业板50ETF嘉实(159373)涨2.61%,报1.296元,成交额1...
    港股异动丨航空股大幅走低 中国... 港股航空股大幅下跌,其中,中国国航跌近7%表现最弱,中国东方航空跌近5%,中国南方航空跌超3%,美兰...
    电网设备ETF(159326)... 8月22日,截止午间收盘,电网设备ETF(159326)跌0.25%,报1.198元,成交额409....
    红利ETF国企(530880)... 8月22日,截止午间收盘,红利ETF国企(530880)跌0.67%,报1.034元,成交额29.0...