浅谈React

forwardRef和useImperativeHandle的联动使用

复制代码
import React, { useImperativeHandle, useRef } from "react"
import { forwardRef } from "react"

const CustomInput = forwardRef((props, ref) => {
    const inputRef = useRef<HTMLInputElement>(null)

    useImperativeHandle(ref, () => ({
        focus: () => {
            inputRef.current?.focus()
        }
    }))
    return <div>
        <input ref={inputRef} />
    </div>
})

export default CustomInput

巧用children

  • 一般用法

    父组件:

    import React from "react"
    import Child from './Child'

    const CustomInput = () => {
    return <Child>

    hello 靓仔

    </Child>
    }

    export default CustomInput

    子组件:

    import React from "react"

    const Child = ({
    children
    }) => {
    return

    {children}

    }

    export default Child

  • 函数用法

    父组件:

    import React from "react"
    import Child from './Child'

    const CustomInput = () => {
    return <Child>
    {(arr)=>


    {arr.map((v,idx)=>{
    return
    {v}

    })}
    }
    </Child>
    }

    export default CustomInput

    子组件:

    import React from "react"

    const Child = ({
    children
    }) => {
    const arr = [1,2,4,5]
    return

    {children(arr)}

    }

    export default Child

useEffect

  • 没有依赖,类似于componentDidMount和componentDidUpdate

    import React, { useEffect, useState } from "react"

    const Detail = () => {
    const [count, setCount] = useState(1)
    const [num,setNum] = useState(2)
    useEffect(()=>{
    console.log(count,'count',num,'num')
    })
    return


    <div onClick={() => setCount(count + 1)}>add count

    <div onClick={() => setNum(num + 1)}>add num

count: {count}

num: {num}


}

export default Detail

  • 依赖是个空数组,相当于componentDidMount

    import React, { useEffect, useState } from "react"

    const Detail = () => {
    const [count, setCount] = useState(1)
    const [num,setNum] = useState(2)
    useEffect(()=>{
    console.log(count,'count',num,'num')
    },[])
    return


    <div onClick={() => setCount(count + 1)}>add count

    <div onClick={() => setNum(num + 1)}>add num

count: {count}

num: {num}


}

export default Detail

  • 有依赖,componentDidMount和对应依赖的componentDidUpdate

    import React, { useEffect, useState } from "react"

    const Detail = () => {
    const [count, setCount] = useState(1)
    const [num,setNum] = useState(2)
    useEffect(()=>{
    console.log(count,'count',num,'num')
    },[num])
    return


    <div onClick={() => setCount(count + 1)}>add count

    <div onClick={() => setNum(num + 1)}>add num

count: {count}

num: {num}


}

export default Detail

useEffect和useLayoutEffect

useEffect在渲染后执行,而useLayouEffect是在渲染之前执行

最典型的例子就是实现一个tooltip组件,在性能比较差的情况下,useEffect会先渲染初始状态再更新,而useLayoutEffect会阻塞UI的更新即不会出现组件闪烁的情况~

阻塞代码:

复制代码
let now = performance.now();
while (performance.now() - now < 100) {
}

useEffect在性能差的情况下会出现以下效果

useContext

依赖注入

复制代码
父级:
export const ThemeContext = createContext({});
const App = ()=>{

return  <ThemeContext.Provider value={{name:"real hot"}}>
        ......
    </ThemeContext.Provider>
}

 子级:
  const context = useContext(ThemeContext)
  console.log(context)

useState的变动

在react管辖下(react17.x.x)

  • 函数形式

状态能够更改多次,只渲染一次

复制代码
const handleCount= ()=>{
  setCount(count=>count+1)
  setCount(count=>count+1)
}
  • 对象形式

多次更改状态会被合并成一次更改,即一次生效其他无效,只渲染一次

复制代码
const handleCount= ()=>{
  setCount(count+1)
  setCount(count+1)
}

在异步任务/原生事件下(react17.x.x)

复制代码
const handleCount= ()=>{
setTimeout(()=>{
  setCount(count=>count+1)
  setCount(count=>count+1)
})
}

版本的演变(React18)

react18之后在异步操作或者react事件中都是批量更新,即多个状态更新合成一次渲染,若需要多次渲染可使用flushsync

复制代码
const handleCount= ()=>{
  flushSync(()=>{
    setCount(count=>count+1)
  })
  flushSync(()=>{
    setCount(count=>count+1)
  })

}

探索生命周期函数

父子组件生命周期执行顺序

挂载:

更新:

卸载:

错误处理

生命周期方法:

getDerivedStateFromError

componentDidCatch

缺点(没办法捕获):

  • 异步操作

  • 事件处理函数报错

  • 错误边界自己报错

    class ErrorBoundary extends React.Component {
    constructor(props) {
    super(props);
    this.state = { hasError: false };
    }

    复制代码
    static getDerivedStateFromError(error) {
      // Update state so the next render will show the fallback UI.
      return { hasError: true };
    }
    
    componentDidCatch(error, errorInfo) {
      // You can also log the error to an error reporting service
      logErrorToMyService(error, errorInfo);
    }
    
    render() {
      if (this.state.hasError) {
        // You can render any custom fallback UI
        return <h1>Something went wrong.</h1>;
      }
    
      return this.props.children; 
    }

    }

实现一个简单的Message

复制代码
import ReactDOM from 'react-dom';
import { Modal } from 'antd';
const ToastFn = () => {
let parent = null;
return {
open: function ({ el, container = document.body }) {
// this.destroy();
parent = document.createElement('div');
document.body.appendChild(parent);
ReactDOM.render(<Modal open onCancel={this.destroy}>{el}</Modal>, parent)
},
destroy: function () {
ReactDOM.unmountComponentAtNode(parent);
},
};
};
const Toast = ToastFn();
export default Toast;
上一篇:软件开发C#(Sharp)总结(续)
下一篇:【北京迅为】《i.MX8MM嵌入式Linux开发指南》-第一篇 嵌入式Linux入门篇-第十九章 Linux 工具之make 工具和 makefile 文件
相关推荐
We་ct1 天前
LeetCode 79. 单词搜索:DFS回溯解法详解
前端·算法·leetcode·typescript·深度优先·个人开发·回溯
ujainu1 天前
Electron 实战:将用户输入保存到本地文件 —— 基于 `fs.writeFileSync` 与 IPC 的安全写入方案
javascript·安全·electron
小奶包他干奶奶1 天前
将svg对象化,并动态修改svg图标的颜色、尺寸等
前端·html
Lee川1 天前
React 快速入门:Vue 开发者指南
前端·vue.js·react.js
进击的尘埃1 天前
基于 LangChain.js 的前端 Agent 工作流编排:Tool 注册、思维链可视化与多步推理的实时 DAG 渲染
javascript
用户6158139695161 天前
Elpis: 基于vue3+webpack5+nodejs搭建一个完整项目
前端
90后的晨仔1 天前
S C:\WINDOWS\system32> pnpm i -g openclaw@latest pnpm : 无法加载文件 C:\xx\A
前端
颜酱1 天前
最小生成树(MST)核心原理 + Kruskal & Prim 算法
javascript·后端·算法
蜡台1 天前
Node 版本管理器NVM 安装配置和使用
前端·javascript·vue.js·node·nvm
狂奔蜗牛飙车1 天前
Day3:HTML5 基础标签:h1-h6、p、hr、br、a、img
前端·html·html5·html常用标签的使用方法
热门推荐
01GitHub 镜像站点02Qwen3.5 开源全解析:从 0.8B 到 397B,代际升级 + 全场景选型指南03OpenClaw 使用和管理 MCP 完全指南04UV安装并设置国内源05本地部署 OpenClaw + DeepSeek-R1 完全指南06OpenClaw Control UI安全上下文访问配置07OpenClaw macOS 完整安装与本地模型配置教程(实战版)08Openclow安装保姆级教程09在 Windows 上配置 Claude Code从安装到解决网络问题10OpenClaw 飞书机器人不回复消息?3 小时踩坑总结