• 前端框架 React 学习总结


    目录

    一、React在HTML里的运用

    二、React框架的常用操作

    项目打包

    1、JSX基础语法规则

    2、state数据的使用

    3、生命周期

    4、数据的双向绑定与Ref

    5、PropTypes验证规则

    6、React里的插槽

    7、错误边界

    8、直接操作refs元素

    9、高阶组件的运用案例

    10、性能优化

    11、Hook生命周期钩子的使用

    12、React里的计算属性

    三、组件之间的传值

    1、父子组件之间传值

    2、子向父传值

    3、context实现跨层级通信

    context hook案例

    四、网络请求框架使用

    五、React路由的使用

    声明式导航

    编程式导航

    编程式跳转

    六、Anti Desgin的使用实现UI界面

    七、Redux中央仓库的使用

    1、新建src/redux/index.js用于存放redux的文件

    2、在src/index.js里引入

    3、可以通过logger查看redux日志

    4、Redux的模块化管理

    5、注意事项

    6、总结Redux使用的完整步骤


    一、React在HTML里的运用

    React在HTML里的使用核心就是导入3个依赖:

    1. <script crossorigin src="https://unpkg.com/react@18/umd/react.development.js">script>
    2. <script crossorigin src="https://unpkg.com/react-dom@18/umd/react-dom.development.js">script>
    3. <script src="https://unpkg.com/babel-standalone@6/babel.min.js">script>

    案例代码如下:

    1. html>
    2. <html lang="en">
    3. <head>
    4. <meta charset="UTF-8">
    5. <title>testtitle>
    6. head>
    7. <body>
    8. <div id="root">div>
    9. body>
    10. <script crossorigin src="https://unpkg.com/react@18/umd/react.development.js">script>
    11. <script crossorigin src="https://unpkg.com/react-dom@18/umd/react-dom.development.js">script>
    12. <script src="https://unpkg.com/babel-standalone@6/babel.min.js">script>
    13. <script type="text/babel">
    14. //1、创建虚拟Dom
    15. // 方式一 JSX
    16. const VDOM = (
    17. <h1 id="title">
    18. <span>Hello,Reactspan>
    19. h1>
    20. )
    21. //方式二 js语法创建DOM
    22. // const VDOM=React.createElement("h1",{id:"title"},React.createElement("span",{},"Hello,React"))
    23. //2、将虚拟DOM转成真实DOM,并插入页面
    24. ReactDOM.render(VDOM, document.getElementById("root"))
    25. script>
    26. html>

    运行效果:

    执行流程如下:

    二、React框架的常用操作

    脚手架创建:

    使用详情查看官网,这里只记录我的笔记:React 官方中文文档 – 用于构建用户界面的 JavaScript 库

    npx create-react-app projectName

    项目打包

     打包时为了路径对应需要到package.json中添加"homepage":"./"这一属性。

    1、JSX基础语法规则

    案例代码:object对象不能执行渲染,arr可以通过map函数进行遍历输出。

    1. import React from 'react';
    2. function App () {
    3. const num=16
    4. const bool=true
    5. const name="string"
    6. const arr=[1,2,3,4,"Jack",true]
    7. const obj={name:"Mary",age:12}
    8. return(
    9. <div style={{marginLeft:"20px"}}>
    10. <div>num:{num}div>
    11. <div>bool:{bool}div>
    12. <div>name:{name}div>
    13. <div>arr:{arr}div>
    14. <div>arr遍历: {
    15. arr.map((item,index)=>(
    16. <p key={index}><span color={"red"}>{item}span>p>
    17. ))
    18. }div>
    19. {/*对象不能直接打印*/}
    20. {/*<div>obj:{obj}div>*/}
    21. <div>obj.name:{obj["name"]}div>
    22. div>
    23. )
    24. }
    25. export default App;

    效果:

    2、state数据的使用

    参考文档:State & 生命周期 – React

    3、生命周期

    发送网络请求一般在componentDidMount里执行,

    销毁组件开销一般在componentWillUnmount里执行。

    4、数据的双向绑定与Ref

    方法一: 

    1. class Test extends Component {
    2. constructor (props, context) {
    3. super(props, context)
    4. this.state={
    5. val: "233"
    6. }
    7. }
    8. changeVal=(e)=>{
    9. console.log(e)
    10. this.setState({
    11. val: e.target.value
    12. })
    13. console.log(e.target.value)
    14. }
    15. render () {
    16. return (
    17. <div>
    18. <input type="text" onChange={this.changeVal} value={this.state.val} />
    19. div>
    20. )
    21. }
    22. }

    方法二:通过ref实现

    1. class Test extends Component {
    2. constructor (props, context) {
    3. super(props, context)
    4. this.state={
    5. val: "233"
    6. }
    7. //1、在构造函数里创建ref的语法
    8. this.myRef=React.createRef()
    9. }
    10. search=()=>{
    11. //3、获取DOM元素
    12. console.log(this.myRef.current.value)
    13. }
    14. render () {
    15. return (
    16. <div>
    17. {/*2、绑定到元素上去*/}
    18. <input type="text" ref={this.myRef} onChange={this.search}/>
    19. div>
    20. )
    21. }
    22. }

    refs的操作参考:

    Refs and the DOM – React

    5、PropTypes验证规则

    参考:

    使用 PropTypes 进行类型检查 – React

    6、React里的插槽

    参考:

    组合 vs 继承 – React

    7、错误边界

    避免一错全不渲染的情况

    参考:错误边界 – React

    8、直接操作refs元素

    函数组件里使用:

    参考:Refs 转发 – React

    9、高阶组件的运用案例

    参考:高阶组件 – React

    实现组件加载时间的复用

    1. import React from "react"
    2. export default function showTime (Comp) {
    3. return class extends React.Component {
    4. constructor (props, context) {
    5. super(props, context)
    6. this.state = {
    7. startTime: new Date().getTime(),
    8. loadingTime: 0
    9. }
    10. }
    11. componentDidMount () {
    12. let endTime = new Date().getTime()
    13. this.setState({
    14. loadingTime: endTime - this.state.startTime
    15. })
    16. }
    17. render () {
    18. return (
    19. <Comp loadingTime={this.state.loadingTime}/>
    20. )
    21. }
    22. }
    23. }

    这样便实现了高阶组件的复用,需要使用这个功能时,只需要调用该函数对象进行功能追加即可。

    10、性能优化

    参考:性能优化 – React

    1、新版本以后Component用PureComponent,

    PureComponent自带state和props的检查,但其中有变化时才重写渲染,否则不重新渲染,提升性能。

    函数组件中的优化:相当于pureComonent

    React.memo()生命周期钩子,相当于pureCompnent会在props和state变化时才发生更新。

    案例代码:

    1. import React from "react"
    2. const componentTwo = React.memo((props) => {
    3. return (
    4. <div>
    5. div>
    6. )
    7. })
    8. export default componentTwo

    2、组件用完后资源记得释放

    11、Hook生命周期钩子的使用

    参考:Hook API 索引 – React

    使用函数组件的效率一般会比类组件效率高一些,但在函数组件(无状态组件)中又没有state等属性,所以这里诞生了Hook为函数组件添加state和生命周期等元素。

    Hook 简介 – React

    案例一: useState  修改状态

    1. import React,{useState} from "react"
    2. function FunComponentOne () {
    3. //定义一个变量,初始值为0,可通过setCount方法修改值
    4. //第一个位置变量名,第二个位置方法名:修改变量的方法
    5. const [count,setCount]=useState(0)
    6. return (
    7. <div>
    8. <h2>{count}h2>
    9. <button onClick={()=>setCount(count+1)}>+button>
    10. div>
    11. )
    12. }
    13. export default FunComponentOne

    案例二:useEffect  生命周期钩子

    1. import React,{useState,useEffect} from "react"
    2. function FunComponentOne () {
    3. //定义一个变量,初始值为0,可通过setCount方法修改值
    4. //第一个位置变量名,第二个位置方法名
    5. const [count,setCount]=useState(0)
    6. //第二个参数为空,相当于生命周期钩子:componentDidMount+componentDidUpdate
    7. //加载完成和数据修改都会走该函数
    8. useEffect(() => {
    9. console.log("=======================")
    10. })
    11. //第二个参数为空数组[],相当于componentDidMount,可以用于网络请求
    12. useEffect(() => {
    13. console.log("++++++++++++++++++++++++++++")
    14. }, [])
    15. //第二个参数可以传入有值的数组,当数组里的变量修改,会调用该函数
    16. useEffect(() => {
    17. console.log("!!!!!!!!!!!!!!!!!!!!!!!!!")
    18. }, [count])
    19. //当第二个参数为空数组[],且有返回函数时相当于componentWillUnMount
    20. //一般用于销毁主键
    21. useEffect(() => {
    22. return function clearUp(){
    23. console.log("clearUp")
    24. }
    25. },[])
    26. return (
    27. <div>
    28. <h2>{count}h2>
    29. <button onClick={()=>setCount(count+1)}>+button>
    30. div>
    31. )
    32. }
    33. export default FunComponentOne

    案例3: Hook reducer类似于升级版的useState

    参考:Hook API 索引 – React

    案例4:自定义Hook 降低耦合度

    参考:自定义 Hook – React

    可以将频繁调用的hook定义到自己的hook里,注意要用use开头

    12、React里的计算属性

    相当于Vue里的Computed

    三、组件之间的传值

    1、父子组件之间传值

    通过组件传参的方式:

    1. <ChildOne num={12}/>
    2. <ChildTwo num={24}/>

    子组件接收参数:

    1. class ChildTwo extends Component {
    2. render () {
    3. return (
    4. <div>
    5. <h2>我是子组件ChildTwo: {this.props.name}h2>
    6. div>
    7. )
    8. }
    9. }
    10. function ChildOne (prop) {
    11. return(
    12. <div>
    13. <h2>ChildOne子组件:{prop.num}h2>
    14. div>
    15. )
    16. }

    2、子向父传值

    通过父组件传递方法对象给子组件,子组件再调用该方法并传入对应参数和处理给父组件。

    父组件:

    1. function ParentOne () {
    2. function getSonData (data) {
    3. console.log(data)
    4. }
    5. return (
    6. <div>
    7. <ChildThree getSonData={getSonData} />
    8. div>
    9. )
    10. }

    子组件:

    1. function ChildThree (prop) {
    2. function sendData(){
    3. prop.getSonData("我是子组件的数据")
    4. }
    5. return(
    6. <div>
    7. <button onClick={sendData}>点击向父组件传值button>
    8. div>
    9. )
    10. }

    这里父组件将getSonData方法对象先传递给子组件,子组件拿到方法对象后可以通过prop进行调用并传入子组件的参数到方法里,此时会调用父组件里的方法以拿到子组件的数据。

    3、context实现跨层级通信

    参考:Context – React

     (1)首先,创建一个MyContext.js用来管理context环境变量

    1. import React from "react"
    2. //创建中间仓库
    3. const MyContext=React.createContext(undefined)
    4. export default MyContext

    (2)案例

    第一层父组件,提供数据

    1. class LayerOne extends Component {
    2. constructor (props, context) {
    3. super(props, context)
    4. this.state={
    5. name: "cute Tom"
    6. }
    7. }
    8. render () {
    9. return (
    10. <div>
    11. <h1>我是oneh1>
    12. <MyContext.Provider value={this.state}>
    13. <LayerTwo/>
    14. MyContext.Provider>
    15. div>
    16. )
    17. }
    18. }

    第二层:包含第三层

    1. class LayerTwo extends Component {
    2. render () {
    3. return (
    4. <div>
    5. <h2>我是twoh2>
    6. <LayerThree/>
    7. div>
    8. )
    9. }
    10. }

    第三层可以直接消费数据,注意这里需要声明一下是哪个MyContext

    1. class LayerThree extends Component {
    2. render () {
    3. return (
    4. <div>
    5. <h3>我是threeh3>
    6. <MyContext.Consumer>
    7. {value => <div><h2>{value.name}h2>div>}
    8. MyContext.Consumer>
    9. div>
    10. )
    11. }
    12. }
    13. LayerThree.contextType=MyContext

    效果:

    context hook案例

    函数组件中的使用:Context Hook

    链接:useContext使用 - 简书 

    四、网络请求框架使用

    见我博客:

    前端框架 网络请求 Fetch Axios_Dragon Wu的博客-CSDN博客

    五、React路由的使用

    官方文档:React Router: Declarative Routing for React.js

    参考:

    React Router 6 (React路由) 最详细教程-阿里云开发者社区

    6.v新特性:

    react-router 6 新特性总结 - 知乎

    首先安装依赖:

    yarn add react-router-dom@6

    声明式导航

    (推荐使用编程式导航效率更高)

    案例代码:一般使用BrowseRouter有history记录

    1. import React, { Component } from "react"
    2. import { BrowserRouter, Routes, Route, Navigate, NavLink, useParams,Link,Outlet } from "react-router-dom"
    3. class Home extends Component {
    4. render () {
    5. return (
    6. <div>
    7. <h1>Homeh1>
    8. div>
    9. )
    10. }
    11. }
    12. class About extends Component {
    13. render () {
    14. return (
    15. <div>
    16. <h1>Abouth1>
    17. div>
    18. )
    19. }
    20. }
    21. function Detail () {
    22. console.log(useParams())
    23. return (
    24. <div>
    25. <h2>详情h2>
    26. <p>id: {useParams().id}p>
    27. div>
    28. )
    29. }
    30. function Main(){
    31. return (
    32. <div>
    33. <h1>文档h1>
    34. <Link to={"/doc/one"}>文档1Link>
    35. <Link to={"/doc/two"}>文档2Link>
    36. <Link to={"/doc/three"}>文档3Link>
    37. {/*嵌套路由时注意书写这个标签*/}
    38. <Outlet/> {/* 指定路由组件呈现的位置 */}
    39. div>
    40. )
    41. }
    42. function App () {
    43. return (
    44. <div>
    45. <BrowserRouter>
    46. <div>
    47. <NavLink to={"/home"}>首页NavLink>|
    48. <NavLink to={"/about"}>关于NavLink>|
    49. <NavLink to={"/detail/123"}>详情NavLink>|
    50. <NavLink to={"/doc"}>文档NavLink>
    51. div>
    52. <Routes>
    53. <Route path={"/home"} element={<Home />} />
    54. <Route path={"/about"} element={<About />} />
    55. {/*嵌套路由*/}
    56. <Route path={"doc"} element={<Main/>}>
    57. <Route path={"one"} element={<h3>oneh3>} />
    58. <Route path={"two"} element={<h3>twoh3>} />
    59. <Route path={"three"} element={<h3>threeh3>} />
    60. Route>
    61. {/*动态路由*/}
    62. <Route path={"/detail/:id"} element={<Detail />} />
    63. {/*路由重定向,也可用于404处理*/}
    64. <Route path="/*" element={<Navigate to="/home" replace />} />
    65. {/*404处理*/}
    66. <Route path="*" element={<NotFound />} />
    67. Routes>
    68. BrowserRouter>
    69. div>
    70. )
    71. }
    72. export default App
    73. // 用来作为 404 页面的组件
    74. const NotFound = () => {
    75. return <div>你来到了没有知识的荒原div>
    76. }

    编程式导航

    (1)新建src/router/index.js文件

    1. import { Navigate } from "react-router-dom"
    2. import Home from "../page/home/Home"
    3. import HomeLeft from "../page/home/HomeLeft"
    4. import HomeRight from "../page/home/HomeRight"
    5. import About from "../page/About"
    6. import Detail from "../page/Detail"
    7. const routes=[
    8. {
    9. path: "/",
    10. element: <Navigate to="/home" />
    11. },
    12. {
    13. path: "home",
    14. element: <Home />,
    15. children: [
    16. {
    17. index: true,
    18. element: <HomeLeft />
    19. },
    20. {
    21. path: "right",
    22. element: <HomeRight />
    23. }]
    24. },
    25. {
    26. path: "/about",
    27. element: <About />
    28. },
    29. {
    30. path: "/detail",
    31. element: <Detail />
    32. },
    33. { path: "*", element: <p>404页面不存在p> }
    34. ]
    35. export default routes

    (2)为src/index.js添加BrowserRouter容器

    注意: BrowserRouter必须在App标签的外层

     (3)App.js如下:

    1. import React from 'react'
    2. import routes from "./router/index.js"
    3. import {useRoutes,NavLink } from "react-router-dom"
    4. function App () {
    5. // useRoutes可以用路由表生成...结构
    6. // 根据路由表生成对应的路由规则
    7. const element = useRoutes(routes)
    8. return(
    9. <div id="App">
    10. <div>
    11. <NavLink to={"/home"} >首页NavLink>|
    12. <NavLink to={"/about"}>关于NavLink>|
    13. <NavLink to={"/detail"}>详情NavLink>
    14. div>
    15. <div>
    16. {element}
    17. div>
    18. div>
    19. )
    20. }
    21. export default App
    (4)若有嵌套路由要使用标签标明子组件插入的位置
    1. class Home extends Component {
    2. render () {
    3. return (
    4. <div>
    5. <h1>Homeh1>
    6. <Link to={"/home/right"}>homeRightLink>
    7. <Link to={"/home"}>homeLeftLink>
    8. {/*嵌套路由时注意书写这个标签*/}
    9. <Outlet/> {/* 指定路由组件呈现的位置 */}
    10. div>
    11. )
    12. }
    13. }

    编程式跳转

    默认是push 模式

    1. export default function HomeNews() {
    2. const navigate = useNavigate();
    3. const jump = ()=>{
    4. navigate('/home')
    5. }
    6. return (
    7. <News>
    8. <button onClick={jump}>点击跳转button>
    9. News>
    10. )
    11. }

    使用{replace:true} 就会变为replace模式

    navigate('/home', { replace: true });
    

    也可以使用 navigate(-1) 传入一个数字来进行跳转

    navigate(1)//传入数字

    六、Anti Desgin的使用实现UI界面

    官方文档:Ant Design - A UI Design Language

    1、添加到项目:

    yarn add antd

    2、样式引入

    全局映入样式:在src/index.js里引入样式,不推荐,会导入很多无用的样式

    import 'antd/dist/antd.css';  // or 'antd/dist/antd.less'

    按需映入,推荐

    下面两种方式都可以只加载用到的组件。

    • 使用 babel-plugin-import(推荐)。

    • yarn add babel-plugin-import
      1. // .babelrc or babel-loader option
      2. {
      3. "plugins": [
      4. ["import", { "libraryName": "antd", "style": "css" }] // `style: true` 会加载 less 文件
      5. ]
      6. }

    • 然后只需从 antd 引入模块即可,无需单独引入样式。等同于下面手动引入的方式。

      1. // babel-plugin-import 会帮助你加载 JS 和 CSS
      2. import { DatePicker } from 'antd';
    • 手动引入

      1. import DatePicker from 'antd/lib/date-picker'; // 加载 JS
      2. import 'antd/lib/date-picker/style/css'; // 加载 CSS
      3. // import 'antd/lib/date-picker/style'; // 加载 LESS

    3、组件里直接调用即可

    七、Redux中央仓库的使用

    当我们的项目稍微复杂一些时,原生的state可能无法高效的管理和操作数据缓存,通过Redux可以将数据统一管理,并且减低代码耦合。

    参考:入门 Redux | Redux 中文官网

     其工作原理与hook reducer类似

    yarn add @reduxjs/toolkit

    安装参考:安装 | Redux 中文官网

    1、新建src/redux/index.js用于存放redux的文件

    1. //1、引入redux
    2. import { configureStore } from "@reduxjs/toolkit"
    3. //2、创建仓库
    4. const store = configureStore({reducer})
    5. //3、reducer为store服务的执行者, action={type:"",data:5}
    6. function reducer (preState = 10, action) {
    7. switch (action.type) {
    8. case "add":
    9. return preState + action.data
    10. case "sub":
    11. return preState - action.data
    12. default:
    13. return preState
    14. }
    15. }
    16. //4、使用store
    17. console.log(store)
    18. //获取仓库的数据
    19. console.log(store.getState())
    20. //触发action
    21. store.dispatch({
    22. type: "add",
    23. data: 5
    24. })
    25. console.log(store.getState())

    2、在src/index.js里引入

    运行结果:可以看到数据已经被操作了

    3、可以通过logger查看redux日志

    yarn add redux-logger

    添加logger后:

    1. //1、引入redux
    2. import { configureStore } from "@reduxjs/toolkit"
    3. //引入日志
    4. import { logger } from "redux-logger/src"
    5. //2、创建仓库
    6. // const store = configureStore({reducer})
    7. const store = configureStore({
    8. reducer,
    9. middleware: (getDefaultMiddleware) => getDefaultMiddleware().concat(logger)
    10. })
    11. //3、reducer为store服务的执行者, action={type:"",data:5}
    12. function reducer (preState = 10, action) {
    13. switch (action.type) {
    14. case "add":
    15. return preState + action.data
    16. case "sub":
    17. return preState - action.data
    18. default:
    19. return preState
    20. }
    21. }
    22. //4、使用store
    23. console.log(store)
    24. //获取仓库的数据
    25. console.log(store.getState())
    26. //触发action
    27. store.dispatch({
    28. type: "add",
    29. data: 5
    30. })
    31. console.log(store.getState())

    可以看到日志的打印:

    合并多个reducer 

    1. //1、引入redux
    2. import { configureStore} from "@reduxjs/toolkit"
    3. //引入日志
    4. import { logger } from "redux-logger/src"
    5. //2、创建仓库
    6. // const store = configureStore({reducer})
    7. const store = configureStore({
    8. //合并多个reducer
    9. reducer:{
    10. reducer,
    11. reducer2
    12. },
    13. middleware: (getDefaultMiddleware) => getDefaultMiddleware().concat(logger)
    14. })
    15. //3、reducer为store服务的执行者, action={type:"",data:5}
    16. function reducer (preState = 10, action) {
    17. switch (action.type) {
    18. case "add":
    19. return preState + action.data
    20. case "sub":
    21. return preState - action.data
    22. default:
    23. return preState
    24. }
    25. }
    26. function reducer2 (preState = { user: "", num: 5 }, action) {
    27. const { type, data } = action
    28. let newState = { ...preState }
    29. switch (type) {
    30. case "addUser":
    31. newState.user = data.user
    32. return newState
    33. case "delUser":
    34. newState.user = ""
    35. return newState
    36. default:
    37. return newState
    38. }
    39. }
    40. //4、使用store
    41. console.log(store)
    42. //获取仓库的数据
    43. console.log(store.getState())
    44. // //触发action
    45. store.dispatch({
    46. type: "add",
    47. data: 5
    48. })
    49. console.log(store.getState())
    50. store.dispatch({
    51. type: "addUser",
    52. data: {
    53. user: "大猫"
    54. }
    55. })
    56. store.dispatch({
    57. type: "delUser"
    58. })

    运行结果:

    4、Redux的模块化管理

    项目书写时我们需要加上命名空间以避免重复:

    另外,我们还需要将常量单独提取到一个文件里管理:

    分类管理reducers:

     reducer/index.js: 引入所有reducers方便调用

    1. import { count } from "./countReducer"
    2. import {user} from "./userReducer"
    3. export const reducers={
    4. count,
    5. user
    6. }

    redux/index.js引入reducer的索引文件即可:

    1. //1、引入redux
    2. import { configureStore} from "@reduxjs/toolkit"
    3. //引入日志
    4. import { logger } from "redux-logger/src"
    5. //引入reducers
    6. import { reducers } from "./reducers"
    7. //2、创建仓库
    8. // const store = configureStore({reducer})
    9. const store = configureStore({
    10. //合并多个reducer
    11. reducer: reducers,
    12. middleware: (getDefaultMiddleware) => getDefaultMiddleware().concat(logger)
    13. })

    5、注意事项

    返回值注意:

    书写时最好按照这种格式的逻辑:

    redux默认不支持异步操作,一般实现思路是等异步任务做完后同步回调时进行redux操作:

    6、总结Redux使用的完整步骤

    安装依赖:

    1. yarn add @reduxjs/toolkit
    2. yarn add react-redux

    日志依赖:

    yarn add redux-logger

    (1)创建store仓库,reducers分开管理

    reducers:  数据开发中type字符串应放入一个常量管理

    1. export function count(preState={num:0},action){
    2. const {type,data}=action
    3. let newState={...preState}
    4. switch (type) {
    5. case "num/add":
    6. newState.num+=data.num
    7. return newState
    8. case "num/sub":
    9. newState.num-=data.num
    10. return newState
    11. default:
    12. return newState
    13. }
    14. }
    1. export function user(preState={user:{name:"",age:1}},action){
    2. const {type,data}=action
    3. let newState={...preState}
    4. switch (type) {
    5. case "user/update":
    6. newState.user=data.user
    7. return newState
    8. case "user/delete":
    9. newState.user={name:"",age:1}
    10. return newState
    11. default:
    12. return newState
    13. }
    14. }

    索引所有reducers:

    1. import { count } from "./countReducer"
    2. import { user } from "./userReducer"
    3. export const reducers = {
    4. count,
    5. user
    6. }

    创建store仓库并开启日志:

    1. //1、引入redux
    2. import { configureStore } from "@reduxjs/toolkit"
    3. //引入日志
    4. import { logger } from "redux-logger/src"
    5. //导入reducers
    6. import { reducers } from "./reducers"
    7. //2、创建仓库
    8. // const store = configureStore({reducer})
    9. const store = configureStore({
    10. //合并多个reducer
    11. reducer: reducers,
    12. middleware: (getDefaultMiddleware) => getDefaultMiddleware().concat(logger)
    13. })
    14. export default store

    (2)导入到src/index.js文件,全局配置

    (3)组件中的使用

    1. import React from "react"
    2. import { connect } from "react-redux"
    3. function ReduxTestComp (props) {
    4. return (
    5. <div>
    6. <h1>count: {props.count.num}h1>
    7. <h1>user:{props.user.user.age},{props.user.user.name}h1>
    8. <button onClick={()=>props.updateUser({user:{ name: "Jack", age: 12 }})}>修改用户button>
    9. <button onClick={()=> props.deleteUser()}>删除用户button>
    10. <button onClick={()=>props.addNum()}>+button>
    11. div>
    12. )
    13. }
    14. export default connect((state) => {//读取仓库中所有state
    15. console.log(state)
    16. return {
    17. count: state.count,
    18. user: state.user
    19. }
    20. }, (dispatch) => {//action操作
    21. console.log(dispatch)
    22. return {
    23. updateUser: (data) => {
    24. return dispatch({ type: "user/update", data: data })
    25. },
    26. deleteUser: ()=>{
    27. return dispatch({type:"user/delete"})
    28. },
    29. addNum:()=>{
    30. return dispatch({type:"num/add",data:{num:12}})
    31. }
    32. }
    33. })(ReduxTestComp)
    1. import React from "react"
    2. import ReduxTestComp from "./component/reduxTest/ReduxTestComp"
    3. function App () {
    4. return (
    5. <div id="App">
    6. <ReduxTestComp />
    7. div>
    8. )
    9. }
    10. export default App

    至此,实现redux的全程应用。

  • 相关阅读:
    java毕业生设计助农脱贫系统计算机源码+系统+mysql+调试部署+lw
    Nginx——下载,安装,启动,关闭,配置——负载均衡——静态代理——动静分离
    linux内核调试工具之kprobe(二)
    一款GUI跨平台自动化测试工具分享——Squish,支持Qt框架
    萤石网络发布家用及商用清洁机器人 积极布局具身智能
    基于android的车辆违章停放执法移动APP-计算机毕业设计
    springboot和vue:九、v-for中的key+vue组件化开发
    JAVA通过JNA 调用c++动态链接库
    arcgis制作采样线及采样点
    Delft3D水动力-富营养化模型实践技术高级应用
  • 原文地址:https://blog.csdn.net/qq_50909707/article/details/127847618