我的编程空间,编程开发者的网络收藏夹
学习永远不晚

React错误的习惯用法分析详解

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

React错误的习惯用法分析详解

过多的声明state

在我们React的日常开发中一些常用的写法,看似运行的很好,实际可能并不优雅。学习React并不是如何如何使用它,而是如何写出优雅,干净的代码。下面举一些例子,总结了一些React开发中不好的写法及相应更好的写法。(仅代表个人观点)

问题

一个组件中声明了过多的state,过多的setState方法。例如下面的这样:

import { useState } from "react";
export default function MoreState() {
  const [username, setUsername] = useState("");
  const [age, setAge] = useState("");
  const [gender, setGender] = useState("");
  const [email, setEmail] = useState("");
  const [password, setPassword] = useState("");
  const [address, setAddress] = useState("");
  const [city, setCity] = useState("");
  const onSubmit = () => {
    // ...
  };
  return (
    <form onSubmit={onSubmit}>
      <input
        type="text"
        name="username"
        placeholder="username"
        value={username}
        onChange={(e) => setUsername(e.target.value)}
      />
      <br />
      <input
        type="text"
        name="age"
        placeholder="age"
        value={age}
        onChange={(e) => setAge(e.target.value)}
      />
      <br />
      <input
        type="text"
        name="gender"
        placeholder="gender"
        value={gender}
        onChange={(e) => setGender(e.target.value)}
      />
      <br />
      <input
        type="text"
        name="email"
        placeholder="email"
        value={email}
        onChange={(e) => setEmail(e.target.value)}
      />
      <br />
      <input
        type="text"
        name="password"
        placeholder="password"
        value={password}
        onChange={(e) => setPassword(e.target.value)}
      />
      <br />
      <input
        type="text"
        name="address"
        placeholder="address"
        value={address}
        onChange={(e) => setAddress(e.target.value)}
      />
      <br />
      <input
        type="text"
        name="city"
        placeholder="city"
        value={city}
        onChange={(e) => setCity(e.target.value)}
      />
      <br />
      <button type="submit">提交</button>
    </form>
  );
}

实际上这样并不好维护,接手项目的人都疯了?。还有这样的:

解决方法

把能合并的state,合并成一个对象表示。当然也可以使用useReducer。当属性中出现嵌套结构时,例如属性中有对象和数组时,使用useReducer更好一些。

import { useState } from "react";
export default function MoreState() {
  const [userInfo, setUserInfo] = useState({
    username: "",
    age: "",
    gender: "",
    email: "",
    password: "",
    address: "",
    city: ""
  });
  const onChange = (e) => {
    setUserInfo((pre) => ({ ...pre, [e.target.name]: e.target.value }));
  };
  const onSubmit = (e) => {
    e.preventDefault();
    console.log(111, userInfo);
  };
  return (
    <form onSubmit={onSubmit}>
      <input
        type="text"
        name="username"
        placeholder="username"
        onChange={onChange}
      />
      <br />
      <input type="text" name="age" placeholder="age" onChange={onChange} />
      <br />
      <input
        type="text"
        name="gender"
        placeholder="gender"
        onChange={onChange}
      />
      <br />
      <input type="text" name="email" placeholder="email" onChange={onChange} />
      <br />
      <input
        type="text"
        name="password"
        placeholder="password"
        onChange={onChange}
      />
      <br />
      <input
        type="text"
        name="address"
        placeholder="address"
        onChange={onChange}
      />
      <br />
      <input type="text" name="city" placeholder="city" onChange={onChange} />
      <br />
      <button type="submit">提交</button>
    </form>
  );
}

不必要的state

问题

我们在开发React表单时,通常会使用state来记录表单的值,例如:

import { useState } from "react";
export default function NoState() {
  const [username, setUsername] = useState("");
  const [password, setPassword] = useState("");
  const onSubmit = (e) => {
    e.preventDefault();
    console.log("需要提交的数据", username, password);
  };
  console.log("组件重新渲染了");
  return (
    <form onSubmit={onSubmit}>
      <label htmlFor="name">名字</label>
      <input
        type="text"
        value={username}
        onChange={(e) => setUsername(e.target.value)}
      />
      <br />
      <label htmlFor="name">密码</label>
      <input
        type="text"
        value={password}
        onChange={(e) => setPassword(e.target.value)}
      />
      <br />
      <button type="submit">提交</button>
    </form>
  );
}

上面的代码看似并没有什么问题,但是我们只是在提交的时候用到了state,并没有在其他地方使用过这些state。这个例子中我们并不关心这些state值的变化,我们只关心我们提交的数据是否正确。而且我们每次输入的时候组件都是重新渲染。这并不友好,这个时候我们需要非受控组件。

解决方法

当表单元素不多时,使用ref来处理,并且每次输入都不会引起组件的重新渲染,因为这个时候我们只关心提交的数据,没有在其他地方使用过这些state。

import { useRef } from "react";
export default function NoState() {
  const usernameRef = useRef();
  const posswordRef = useRef();
  const onSubmit = (e) => {
    e.preventDefault();
    console.log(
      "需要提交的数据",
      usernameRef.current.value,
      posswordRef.current.value
    );
  };
  console.log("组件重新渲染了");
  return (
    <form onSubmit={onSubmit}>
      <label htmlFor="name">名字</label>
      <input type="text" ref={usernameRef} />
      <br />
      <label htmlFor="name">密码</label>
      <input type="text" ref={posswordRef} />
      <br />
      <button type="submit">提交</button>
    </form>
  );
}

过多的useEffect

问题

有时当页面第一次挂载时,我们需要进行网络请求,我们经常会这样写:

import { useEffect, useState } from "react";
export default function MoreUseEffect() {
  const [data, setData] = useState();
  useEffect(() => {
    fetch("/ss/ss").then((res) => {
      setData(res.data);
    });
  }, []);
  useEffect(() => {
    // 进行其他逻辑处理...
    console.log(data);
  }, [data]);
  return <>页面第一次加载时请求</>;
}

引入了过多的useEfffect,实际上我们只是需要使用请求到的数据来进行其他逻辑的处理,并不需要数据变化时做一些事情。

解决方法

把数据的处理逻辑放入第一个useEffect中直接处理。

import { useEffect } from "react";
export default function MoreUseEffect() {
  useEffect(() => {
    fetch("/ss/ss").then((res) => {
    	// setData(res.data);
      // 在这里直接进行数据处理...
      console.log('')
    });
  }, []);
  return <>页面第一次加载时请求</>;
}

请求竞争问题

问题

下面是对fetch请求进行了封装,这种写法有一个问题:当同时有多个请求时,由于请求返回的时间不一样,会出现竞争关系,不会按照请求的顺序返回结果,这样就造成返回的结果不知道是哪次的。

import { useEffect, useState } from "react";
export default function useFetch(url) {
  const [loading, setLoading] = useState(true);
  const [data, setData] = useState();
  const [error, setError] = useState();
  useEffect(() => {
    setLoading(true);
    fetch(url)
      .then((res) => {
        setData(res.data);
      })
      .catch((e) => {
        setError(e);
      })
      .finally(() => setLoading(false));
  }, [url]);
  return {
    loading,
    data,
    error
  };
}

解决方法

需要在请求URL变化之后取消前一次的请求。

import { useEffect, useState } from "react";
export default function useFetch(url) {
  const [loading, setLoading] = useState(true);
  const [data, setData] = useState();
  const [error, setError] = useState();
  useEffect(() => {
    const controller = new AbortController();
    setLoading(true);
    fetch(url, { signal: controller.signal })
      .then((res) => {
        setData(res.data);
      })
      .catch((e) => {
        setError(e);
      })
      .finally(() => setLoading(false));
    return () => {
      controller.abort();
    };
  }, [url]);
  return {
    loading,
    data,
    error
  };
}

使用三元表达式代替&&

使用 && 常见的错误

1.当状态值不是Boolean,而是数字0时,数字0会在UI中显示。

import { useState } from "react";
export default function MoreUseEffect() {
	const [arr] = useState([])
  return <>
    {
      arr.length && <div>11111</div>
    }
    </>;
}

解决方法

  • 转成Boolean
  • 使用三元表达式代替 && (推荐)

传递特殊属性ref

问题

ref属性是React的特殊属性,不能直接传递使用。

import {useRef} from 'react'
function InputCom({ref}) {
  return (
    <input type='text' ref={ref} />
  )
}
function App() {
  const inpRef = useRef(null)
  const focus = () => {
    inpRef.current?.focus()
  }
  return (
    <>
      <InputCom ref={inpRef} />
    </>
  )
}

如果想传递ref需要借助forwardRef函数。

解决方法

借助forwardRef转发ref属性

import { forwardRef, useRef } from "react";
// function InputCom({ ref }) {
//   return <input type="text" ref={ref} />;
// }
const InputCom = forwardRef((props, ref) => {
  return <input type="text" ref={ref} />;
});
export default function ProRef() {
  const inpRef = useRef(null);
  const focus = () => {
    inpRef.current?.focus();
  };
  return (
    <>
      <InputCom ref={inpRef} />
      <br />
      <button onClick={focus}>focus</button>
    </>
  );
}

以上就是React错误用法习惯分析详解的详细内容,更多关于React错误用法习惯的资料请关注编程网其它相关文章!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

React错误的习惯用法分析详解

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

下载Word文档

猜你喜欢

React错误的习惯用法分析详解

这篇文章主要为大家介绍了React错误用法习惯分析详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
2023-01-09

ReactuseState的错误用法避坑详解

这篇文章主要为大家介绍了ReactuseState的错误用法避坑详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
2023-01-12

前后端项目分离解决cors错误的方法详解

随着前后端分离技术的越来越盛行,跨域问题也逐渐凸显了出来,下面这篇文章主要给大家介绍了关于前后端项目分离解决cors错误的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
2023-02-15

MySQL学习之分组查询的用法详解

目录为什么要分组逐级分组逐级分组对 SELECT 子句的要求对分组结果集再次做汇总计算GROUP_CONCAT 函数GROUP BY 子句的执行顺序该章节来开始学习分组查询,上一章节我们学习了聚合函数,默认统计的是全表范围内的数据,配合上
2022-07-28

总结分析Git pull 常见的错误及其解决方法

在使用 Git 进行项目协作时,我们经常会使用 git pull 命令来同步远程仓库中的代码到本地。尽管这个命令非常简单易用,但有时候可能会遇到各种各样的错误,甚至导致项目无法正常运行。本文将分析 Git pull 常见的错误及其解决方法。
2023-10-22

XP系统弹出Ravmond.exe应用程序错误的原因分析及解决方法

Ravmond.exe是XP系统中的一个进程,通常与Realtek音频驱动程序相关。当出现Ravmond.exe应用程序错误时,可能是由于以下几个原因导致:1. 音频驱动程序问题:Ravmond.exe可能与Realtek音频驱动程序有关,
2023-08-30

Win8系统蓝屏且提示错误代码0x000000ED的故障分析及解决方法

win8电脑如果遇到了蓝屏代码0x000000ED的蓝屏故障,具体现象如下所示:故障分析:可能是硬盘出现了坏道,修复坏道就能解决蓝屏故障问题。 解决方法:1、电脑不心装上了恶意软件,或上网时产生了恶意程序。解决方法:建议用360 卫士 、金
2022-06-04

Win7系统出现蓝屏提示错误代码0x00000124的原因分析及解决方法

电脑出现蓝屏想必大家都遇到过吧,最近就有win7系统用户发现他的电脑在操作的时候,突然出现了蓝屏的现象,并提示0x00000124错误代码,这是怎么回事呢,接下来我们的小编以win7 64位纯净版系统为例,给大家分享一下具体的解决方法吧。蓝
2023-05-22

Win8.1系统安装补丁遇到报错提示错误代码0x80070020的故障分析及解决方法

下载Windows 8.1补丁执行安装时可能会遇到报错:0x80070020,具体如下所示:0x80070020错误代码的意思是:进程无法访问文件,因为文件被另一进程使用。 故障分析: 错误造成的可能原因是某些系统文件被第三方反病毒软件锁定
2022-06-04

Win8.1系统安装LOL英雄联盟提示NSIS Error错误的故障分析及解决方法

在Win8.1系统下安装了英雄联盟游戏时出现了错误,提示NSIS Error错误,具体现象如下所示:Win8.1安装LOL英雄联盟提示NSIS Error错误的原因:1、网络不稳定或各种原因导致下载的程序不完整引起的2、电脑中病毒导致程序损
2022-06-04

Win8.1专业版出现蓝屏提示错误代码0x0000001的原因分析及解决方法

Win8系统出现蓝屏,蓝屏的错误代码为0x0000001。这个问题可能是由于外接的设备输入了错误的指令,导致系统的冲突。那么这个问题应该如何解决呢?一起来看一下Win8出现蓝屏错误代码0x0000001的解决方法吧。 故障原因分析: 程序发
2022-06-04

Win8系统提示无效的产品密钥错误代码0x80070424的故障分析及解决方法

Win8系统提示无效的产品密钥错误代码0x80070424,如下图所示:故障分析:可能是Software Protection服务丢失,即在services中无法找到该服务。解决方法:1、按”Win+R”组合键,调出
2022-06-05

Win8.1系统更新Flash后网页经常出现错误提示的故障分析及解决方法

Win8.1系统更新Flash后网页经常出现错误提示,具体错误现象如下所示:故障原因分析: win8.1正式版系统Flash可通过自动更新来升级,有些大概就是更新了不正常的Flash版本导致。 解决方案: 1、打开控制面板—程序
2022-06-04

编程热搜

目录