iis服务器助手广告广告
返回顶部
首页 > 资讯 > 前端开发 > JavaScript >详解React中key的作用
  • 919
分享到

详解React中key的作用

2024-04-02 19:04:59 919人浏览 薄情痞子
摘要

目录问题: 点击按钮的时候,span的颜色会变成红色吗?分析上述问题:第二种情况:key的取值为索引值第三种情况:key的取值确定且唯一:如果想要达到预期效果,我们要设置唯一且确定的

要了解React中key的作用,可以从key的取值入手,key的取值可以分为三种,不定值、索引值、确定且唯一值

在下面的代码中,key的取值是不定值(Math.random())

问题: 点击按钮的时候,span的颜色会变成红色吗?


import React, { useState } from 'react';

function App() {
  const [initMap, setInitMap] = useState([1,2,3,4]);
  const handleClick = () => {
    setInitMap([1,2,3,4])
    var spanEle = document.getElementsByTagName('span');
    Array.from(spanEle).map(it => it.style.color = 'red')
  }
  
  return (
    <div className="App" id="app">
      {
        initMap.map((it,index) => <div key={Math.random()}><span>color</span></div>)
      }
      <button onClick={() => handleClick()}></button>
    </div>
  );
}

export default App;

答案是:不会

这个问题涉及react渲染机制和diff算法

官网中对于diff有如下规则:

  • 当元素类型变化时,会销毁重建
  • 当元素类型不变时,对比属性
  • 当组件元素类型不变时,通过props递归判断子节点
  • 递归对比子节点,当子节点是列表时,通过key和props来判断。若key一致,则进行更新,若key不一致,就销毁重建

分析上述问题:

当点击按钮时,setInitMap([1,2,3,4])会造成渲染,渲染时会生成新的虚拟dom,但此时获取到的span元素是之前的元素(因为setInitMap是异步执行的),所以新旧dom会做对比

在initMap.map((it,index) => <div key={Math.random()}><span>color</span></div>)这段代码中

这里的div是列表,对比第四条diff规则,react会根据key来判断是否更新真实dom。key= {Math.random()},新旧dom的值不一致,就会重新生成div。而我们是给更新之前的元素加了红色的样式,所以重新创建的元素上不会有这个样式,效果如下

第二种情况:key的取值为索引值

上面我们分析的结果是,因为key的变化,导致div元素在render的时候会重新生成。那如果key在render前后保持不变呢?例如,将key改为index

问题: 这段代码在button点击之后,span的颜色会变吗?


return (
    <div className="App" id="app">
      <Spin spinning={spin}></Spin>
      {
        initMap.map((it,index) => <div key={index}><span>color</span></div>)
      }
      <button onClick={() => handleClick()}></button>
    </div>
  );

答案:会

分析: 因为在render前后,index不变,所以div不会重新生成,接着对比span元素,span元素在render前后,属性变化,因此react只会为span元素应用新的属性,但是他们指向的还是之前的元素

第三种情况:key的取值确定且唯一:

在这个例子中,通过将key设置成index,span的颜色有了变化,但是在使用key时,React官网不推荐使用index

改造一下上面的代码


  const [initMap, setInitMap] = useState([1,2,3,4]);
  const handleClick = () => {
    setInitMap([3,2,1,4])
  }
  return (
    <div className="App" id="app">
      {
        initMap.map((it,index) => <div key={index}><input type="radio" />{it}</div>)
      }
      <button onClick={() => handleClick()}>点击</button>
    </div>
  );
}

在初始化的时候选中值为3的按钮

点击按钮

我们预期的效果是,选中的依旧是值为3的按钮,但此时变成了值为1的按钮

分析:

  1. setState之后会导致render
  2. div的index不变,所以div不会重新生成,input不受state和props控制,因此元素的状态不变
  3. 所以变化的只有受state影响的it

如果想要达到预期效果,我们要设置唯一且确定的key

测试一:


{
   initMap.map((it) => <div key={it}><input type="radio" />{it}</div>)
}

初始化的时候选中第三个按钮

点击按钮

这才是符合预期的效果

思考一下,将key设置为Math.random(),会有什么效果?按钮的状态会保留吗?

点击前:

点击后:

radio的状态不会被保留

通过上面的例子,我们大概可以理解React中key的作用了,下面的内容是对React知识点的一些扩展

扩展内容: 文章开始的代码还涉及React两个其他知识点,一个是提到过的React渲染条件,一个是对真实dom的操作;

扩展一: React渲染条件


import './App.CSS';
import React, { useState } from 'react';

function App() {
  const [initMap, setInitMap] = useState([1,2,3,4]);
  const [spin, setSpin] = useState(false);
  const handleClick = () => {
    setSpin(true); //变化部分
    var spanEle = document.getElementsByTagName('span');
    Array.from(spanEle).map(it => it.style.color = 'red')
    setSpin(false); //变化部分
  }
  
  return (
    <div className="App" id="app">
      <Spin spinning={spin}></Spin>
      {
        initMap.map((it,index) => <div key={Math.random()}><span>{it}</span></div>)
      }
      <button onClick={() => handleClick()}></button>
    </div>
  );
}

export default App;

测试结果如下 点击前:

点击后:

在这段代码中,div的key仍然使用的是Math.random(),但initMap的state并没有改变,所以没有重新渲染,此时div不会销毁重建

扩展二:是否可以对真实dom操作

在React中,虚拟dom的出现是为了减少对真实dom的操作,因为真实的dom元素是一个较复杂的对象,操作的话计算量比较大。我们上面的代码中,都是直接操作dom节点,更改样式,这样并不可取。由于React是根据state和props的变化来渲染页面,因此通过state来控制页面渲染比较好

修改后的代码如下:


function App() {
  const [initMap, setInitMap] = useState([1,2,3,4]);
  const [spin, setSpin] = useState(false);
  const [showColor, setShowColor] = useState(false);
  const handleClick = () => {
    setInitMap([3,2,1,4]);
    setShowColor(true);
  }
  
  return (
    <div className="App" id="app">
      <Spin spinning={spin}>
      {
        initMap.map((it,index) => <div key={Math.random()}><span className={showColor && 'span-color'}>color</span></div>)
      }
      </Spin>
      <button onClick={() => handleClick()}>点击</button>
    </div>
  );
}

此时span是受控组件,可以通过showColor的状态控制元素的渲染

点击前:

点击后:

使用state控制渲染后,代码量会变少,同时结果符合预期

总结

  1. 在使用key时,要保证key的唯一和确定性,如果key的值为Math.random(),可能造成组件重新构建,使之前对元素的操作失效
  2. 在渲染页面时,尽量不要操作真实的dom,使用state来更新页面

以上就是详解React中key的作用的详细内容,更多关于React key的作用的资料请关注编程网其它相关文章!

--结束END--

本文标题: 详解React中key的作用

本文链接: https://www.lsjlt.com/news/123327.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 详解React中key的作用
    目录问题: 点击按钮的时候,span的颜色会变成红色吗?分析上述问题:第二种情况:key的取值为索引值第三种情况:key的取值确定且唯一:如果想要达到预期效果,我们要设置唯一且确定的...
    99+
    2024-04-02
  • react中key的作用是什么
    本文小编为大家详细介绍“react中key的作用是什么”,内容详细,步骤清晰,细节处理妥当,希望这篇“react中key的作用是什么”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。...
    99+
    2024-04-02
  • React中key的作用有哪些
    这篇文章将为大家详细讲解有关React中key的作用有哪些,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。要了解React中key的作用,可以从key的取值入手,key的取值可以分为三种,不定值、索引值、确...
    99+
    2023-06-14
  • Vue中key的作用及原理详解
    目录1. 先说结论2. key的作用2.1 举一个例子2.2 修改一下上述示例2.3 再修改一下示例3. key的实现原理1. key为index的情况。2. key为id的情况。总...
    99+
    2024-04-02
  • react中的key怎么使用
    这篇“react中的key怎么使用”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“react...
    99+
    2024-04-02
  • React中key怎么使用
    这篇文章给大家分享的是有关React中key怎么使用的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。在渲染列表时,React的差异比较算法需要一个在列表范围内的唯一key来提高性能...
    99+
    2024-04-02
  • Spring@Cacheable注解中key的使用详解
    目录Spring@Cacheable注解中key使用下面是几个使用参数作为key的示例condition属性指定发生的条件@CachePut@CacheEvictallEntries...
    99+
    2024-04-02
  • Vue开发实例探究key的作用详解
    目录前言为什么不推荐使用 index 作为 key?如果 key 重复会导致什么样的错误?使用 key 和不使用 key 有什么差别?key的实际应用上述结论在Vue3中也成立吗?总...
    99+
    2023-01-10
    Vue key作用 Vue key
  • VUE v-for中的:key详解
    不在v-for的标签中加入key时。 <!DOCTYPE html> <html lang="en"> <head> <meta c...
    99+
    2024-04-02
  • mysql中primary key的作用
    mysql 中 primary key(主键)是唯一标识表中每一行的列或列组合。主要作用包括:1. 唯一性约束:确保每行拥有唯一值;2. 优化查询性能:作为索引,提高查询和排序速度;3....
    99+
    2024-04-26
    mysql
  • React 中 setState 的异步操作案例详解
    目录前言React 中的 setState 为什么需要异步操作?什么时候setState会进行同步操作?前言 在使用state的时候, 如果我们企图直接...
    99+
    2024-04-02
  • React使用useEffect解决setState副作用详解
    目录介绍一下APIfetch()方法访问APIsetState的副作用使用useEffect解决这个问题使用useEffect操控函数运行介绍一下API 本文主要内容:描述了setS...
    99+
    2022-11-13
    React useEffect解决setState React useEffect setState
  • Vue 列表渲染 key的原理和作用详解
    目录列表渲染 key 的原理和作用key的原理分析key的作用总结列表渲染 key 的原理和作用 key就是为该节点做身份标识,如果对key绑定index的值,那么很容易出现问题: ...
    99+
    2024-04-02
  • 详解React Fiber的工作原理
    目录啥是React Fiber? 为什么会有React Fiber? React Fiber到底怎么工作的? React Fiber的实现原理 React Fiber对我们日常开发有...
    99+
    2024-04-02
  • React使用redux基础操作详解
    目录一,什么是redux二,安装redux谷歌调试工具三,操作store 改变四,写redux的小技巧一,什么是redux Redux是一个用来管理管理数据状态和UI状态的JavaS...
    99+
    2023-01-13
    React使用redux React redux
  • react为什么不推荐使用index作为key
    1.旧的虚拟dom和新的虚拟dom对比,首先看他们的key是否相同 2.相同继续对比他们的内容,不同生成新的真实dom进行替换 3.如果内容和key都相同,复用旧的真实dom 不做改...
    99+
    2024-04-02
  • 详解Flutter中key的正确使用方式
    目录1、什么是key2、key的更新原理3、key的分类GlobalKeyLocalKey总结1、什么是key Widget中有个可选属性key,顾名思义,它是组件的标识符,当设置...
    99+
    2023-01-28
    Flutter key使用方式 Flutter key
  • 详解React Hooks是如何工作的
    目录1. React Hooks VS 纯函数2. 简单 myUseState3. 改进 myUseState4. 实现原理引发的 Hooks 规则1. React Hooks VS...
    99+
    2024-04-02
  • react 中 mobx的使用案例详解
    1.新建一个mobx.jsx文件 import { useContext } from "react" import MyContext from '../../utils/Cont...
    99+
    2023-05-15
    react mobx使用 react mobx
  • 详解React中Fragment的简单使用
    目录react 中的 Fragment标签渲染Fragment 标签Fragment 标签和 <></> 区别react 中的 ...
    99+
    2022-11-13
    React Fragment使用 React Fragment
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作