学习react的笔记
生命周期 - 概述
组件的生命周期是指组件从被创建到挂载到页面中运行起来,再到组件不用时卸载的过程,注意,只有类组件才有生命周期(类组件 实例化 函数组件 不需要实例化)
http://projects.wojtekmaj.pl/react-lifecycle-methods-diagram/
生命周期 - 挂载阶段
生命周期 - 更新阶段
生命周期 - 卸载阶段
钩子函数 componentWillUnmount
触发时机 (组件卸载)(从页面中消失)
作用(执行清理工作(比如:清理定时器等))
Hooks基础
Hooks概念理解
1. 什么是hooks
Hooks的本质:一套能够使函数组件更强大,更灵活的“钩子”
React体系里组件分为 类组件 和 函数组件
经过多年的实战,函数组件是一个更加匹配React的设计理念 UI = f(data),也更有利于逻辑拆分与重用的组件表达形式,而先前的函数组件是不可以有自己的状态的,为了能让函数组件可以拥有自己的状态,所以从react v16.8开始,Hooks应运而生
注意点:
- 有了hooks之后,为了兼容老版本,class类组件并没有被移除,俩者都可以使用
- 有了hooks之后,不能在把函数成为无状态组件了,因为hooks为函数组件提供了状态
hooks只能在函数组件中使用
2. Hooks解决了什么问题
Hooks的出现解决了俩个问题 1. 组件的状态逻辑复用 2.class组件自身的问题
- 组件的逻辑复用
在hooks出现之前,react先后尝试了 mixins混入,HOC高阶组件,render-props等模式
但是都有各自的问题,比如mixin的数据来源不清晰,高阶组件的嵌套问题等等
- class组件自身的问题
class组件就像一个厚重的‘战舰’ 一样,大而全,提供了很多东西,有不可忽视的学习成本,比如各种生命周期,this指向问题等等,而我们更多时候需要的是一个轻快灵活的’快艇’
useState
1. 基础使用
作用 :useState为函数组件提供状态(state)
使用步骤
- 导入 useState 函数
- 调用 useState 函数,并传入状态的初始值
- 从useState函数的返回值中,拿到状态和修改状态的方法
- 在JSX中展示状态
- 调用修改状态的方法更新状态
代码实现
1 2 3 4 5 6 7 8 9 10 11
| import { useState } from 'react'
function App() { const [count, setCount] = useState(0) return ( <button onClick={() => { setCount(count + 1) }}>{count}</button> ) } export default App
|
2. 状态的读取和修改
读取状态
该方式提供的状态,是函数内部的局部变量,可以在函数内的任意位置使用
修改状态
- setCount是一个函数,参数表示最新的状态值
- 调用该函数后,将使用新值替换旧值
- 修改状态后,由于状态发生变化,会引起视图变化
注意事项
修改状态的时候,一定要使用新的状态替换旧的状态,不能直接修改旧的状态,尤其是引用类型
3. 组件的更新过程
函数组件使用 useState hook 后的执行过程,以及状态值的变化
- 从头开始执行该组件中的代码逻辑
- 调用 useState(0) 将传入的参数作为状态初始值,即:0
- 渲染组件,此时,获取到的状态 count 值为: 0
- 点击按钮,调用 setCount(count + 1) 修改状态,因为状态发生改变,所以,该组件会重新渲染
- 组件重新渲染时,会再次执行该组件中的代码逻辑
- 再次调用 useState(0),此时 React 内部会拿到最新的状态值而非初始值,比如,该案例中最新的状态值为 1
- 再次渲染组件,此时,获取到的状态 count 值为:1
注意:useState 的初始值(参数)只会在组件第一次渲染时生效。也就是说,以后的每次渲染,useState 获取到都是最新的状态值,React 组件会记住每次最新的状态值
1 2 3 4 5 6 7 8 9 10 11
| import { useState } from 'react'
function App() { const [count, setCount] = useState(0) console.log(count) return ( <button onClick={() => { setCount(count + 1) }}>{count}</button> ) } export default App
|
4. 使用规则
- useState 函数可以执行多次,每次执行互相独立,每调用一次为函数组件提供一个状态
1 2 3 4 5 6
| function List(){ const [name, setName] = useState('cp') const [list,setList] = useState([]) }
|
- useState 注意事项
只能出现在函数组件中
不能嵌套在if/for/其它函数中(react按照hooks的调用顺序识别每一个hook)
1 2 3 4 5 6 7 8 9 10
| let num = 1 function List(){ num++ if(num / 2 === 0){ const [name, setName] = useState('cp') } const [list,setList] = useState([]) }
|
- 可以通过开发者工具查看hooks状态
useEffect
1. 理解函数副作用
什么是副作用
副作用是相对于主作用来说的,一个函数除了主作用,其他的作用就是副作用。对于 React 组件来说,主作用就是根据数据(state/props)渲染 UI,除此之外都是副作用(比如,手动修改 DOM)
常见的副作用
数据请求 ajax发送
手动修改dom
localstorage操作
useEffect函数的作用就是为react函数组件提供副作用处理的!
2. 基础使用
作用: 为react函数组件提供副作用处理
使用步骤
- 导入 useEffect 函数
- 调用 useEffect 函数,并传入回调函数
- 在回调函数中编写副作用处理(dom操作)
- 修改数据状态
- 检测副作用是否生效
代码实现
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| import { useEffect, useState } from 'react'
function App() { const [count, setCount] = useState(0)
useEffect(()=>{ document.title = `当前已点击了${count}次` }) return ( <button onClick={() => { setCount(count + 1) }}>{count}</button> ) } export default App
|
3. 依赖项控制执行时机
- 不添加依赖项
组件首次渲染执行一次,以及不管是哪个状态更改引起组件更新时都会重新执行
- 组件初始渲染
- 组件更新 (不管是哪个状态引起的更新)
1 2 3
| useEffect(()=>{ console.log('副作用执行了') })
|
- 添加空数组
组件只在首次渲染时执行一次
1 2 3
| useEffect(()=>{ console.log('副作用执行了') },[])
|
- 添加特定依赖项
副作用函数在首次渲染时执行,在依赖项发生变化时重新执行
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| function App() { const [count, setCount] = useState(0) const [name, setName] = useState('zs')
useEffect(() => { console.log('副作用执行了') }, [count]) return ( <> <button onClick={() => { setCount(count + 1) }}>{count}</button> <button onClick={() => { setName('cp') }}>{name}</button> </> ) }
|
注意事项
useEffect 回调函数中用到的数据(比如,count)就是依赖数据,就应该出现在依赖项数组中,如果不添加依赖项就会有bug出现
自定义hook
需求描述:自定义一个hook函数,实现获取滚动距离Y
const [y] = useWindowScroll()
1 2 3 4 5 6 7 8 9 10
| import { useState } from "react"
export function useWindowScroll () { const [y, setY] = useState(0) window.addEventListener('scroll', () => { const h = document.documentElement.scrollTop setY(h) }) return [y] }
|
需求描述: 自定义hook函数,可以自动同步到本地LocalStorage
const [message, setMessage] = useLocalStorage(key,defaultValue)
message可以通过自定义传入默认初始值
每次修改message数据的时候 都会自动往本地同步一份
1 2 3 4 5 6 7 8 9 10
| import { useEffect, useState } from 'react'
export function useLocalStorage (key, defaultValue) { const [message, setMessage] = useState(defaultValue) // 每次只要message变化 就会自动同步到本地ls useEffect(() => { window.localStorage.setItem(key, message) }, [message, key]) return [message, setMessage] }
|
Hooks进阶
useState - 回调函数的参数
能够理解useState回调函数作为参数的使用场景
使用场景
参数只会在组件的初始渲染中起作用,后续渲染时会被忽略。如果初始 state 需要通过计算才能获得,则可以传入一个函数,在函数中计算并返回初始的 state,此函数只在初始渲染时被调用
语法
1
| const [name, setName] = useState(()=>{ // 编写计算逻辑 return '计算之后的初始值'})
|
语法规则
- 回调函数return出去的值将作为 name 的初始值
- 回调函数中的逻辑只会在组件初始化的时候执行一次
语法选择
- 如果就是初始化一个普通的数据 直接使用 useState(普通数据) 即可
- 如果要初始化的数据无法直接得到需要通过计算才能获取到,使用useState(()=>{})
来个需求
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
| import { useState } from 'react'
function Counter(props) { const [count, setCount] = useState(() => { return props.count }) return ( <div> <button onClick={() => setCount(count + 1)}>{count}</button> </div> ) }
function App() { return ( <> <Counter count={10} /> <Counter count={20} /> </> ) }
export default App
|
useEffect - 清理副作用
- 使用场景
在组件被销毁时,如果有些副作用操作需要被清理,就可以使用此语法,比如常见的定时器
语法及规则
1 2 3 4 5 6 7 8
| useEffect(() => { console.log('副作用函数执行了') return () => { console.log('清理副作用的函数执行了') } })
|
- 定时器小案例
添加副作用函数前:组件虽然已经不显示了,但是定时器依旧在运行
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
| import { useEffect, useState } from 'react' function Foo() { useEffect(() => { setInterval(() => { console.log('副作用函数执行了') }, 1000) }) return <div>Foo</div> }
function App() { const [flag, setFlag] = useState(true) return ( <> <button onClick={() => setFlag(false)}>click</button> {flag ? <Foo/> : null} </> ) } export default App
|
添加清理副作用函数后:一旦组件被销毁,定时器也被清理
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24
| import { useEffect, useState } from 'react'
function Foo() { useEffect(() => { const timerId = setInterval(() => { console.log('副作用函数执行了') }, 1000) return () => { clearInterval(timerId) } }) return <div>Foo</div> } function App() { const [flag, setFlag] = useState(true) return ( <> <button onClick={() => setFlag(false)}>click</button> {flag ? <Foo/> : null} </> ) } export default App
|
useEffect - 发送网络请求
能够掌握使用useEffect hook发送网络请求
- 使用场景
如何在useEffect中发送网络请求,并且封装同步 async await操作
- 语法要求
不可以直接在useEffect的回调函数外层直接包裹 await ,因为异步会导致清理函数无法立即返回
useEffect(async ()=>{
const res = await axios.get(‘http://geek.itheima.net/v1_0/channels')
console.log(res)
},[])
3. 正确写法
在内部单独定义一个函数,然后把这个函数包装成同步
1 2 3 4 5
| useEffect(()=>{ async function fetchData(){ const res = await axios.get('http://geek.itheima.net/v1_0/channels') console.log(res) } },[])
|
useRef
能够掌握使用useRef获取真实dom或组件实例的方法
1. 使用场景
在函数组件中获取真实的dom元素对象或者是组件对象
2. 使用步骤
- 导入 useRef 函数
- 执行 useRef 函数并传入null,返回值为一个对象 内部有一个current属性存放拿到的dom对象(组件实例)
- 通过ref 绑定 要获取的元素或者组件
获取dom
1 2 3 4 5 6 7 8 9 10 11 12 13
| import { useEffect, useRef } from 'react' function App() { const h1Ref = useRef(null) useEffect(() => { console.log(h1Ref) },[]) return ( <div> <h1 ref={ h1Ref }>this is h1</h1> </div> ) } export default App
|
获取组件实例
函数组件由于没有实例,不能使用ref获取,如果想获取组件实例,必须是类组件
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
| Foo.js
class Foo extends React.Component { sayHi = () => { console.log('say hi') } render(){ return <div>Foo</div> } } export default Foo
App.js
import { useEffect, useRef } from 'react' import Foo from './Foo' function App() { const h1Foo = useRef(null) useEffect(() => { console.log(h1Foo) }, []) return ( <div> <Foo ref={ h1Foo } /></div> ) } export default App
|
useContext
能够掌握hooks下的context使用方式
实现步骤
- 使用createContext 创建Context对象
- 在顶层组件通过Provider 提供数据
- 在底层组件通过useContext函数获取数据
代码实现
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24
| import { createContext, useContext } from 'react'
const Context = createContext()
function Foo() { return <div>Foo <Bar/></div> }
function Bar() { const name = useContext(Context) return <div>Bar {name}</div> }
function App() { return ( <Context.Provider value={'this is name'}> <div><Foo/></div> </Context.Provider> ) }
export default App
|
参考文档
gitee
官方文档