首页 > 其他分享 >React笔记-基础

React笔记-基础

时间:2022-12-08 15:01:47浏览次数:50  
标签:react 基础 笔记 React action 路由 组件 redux store

Redirect的使用
  1. 一般写在所有路由注册的最下方,当所有路由都无法匹配的时候,跳转到Redirect指定的路由

  2. 具体编码:

    <Switch>
    	<Route path='./xxx' component={xxx}/>
        <Route path='./xxx' component={xxx}/>
        <Redirect to="/xxx"/>
    </Switch>
    
嵌套路由
  1. 注册子路由时要写上父路由的path值

  2. 路由的匹配是按照注册路由的顺序进行的

向路由组件传递参数
  1. params参数

    路由链接(携带参数):

    <Link to='demo/test/tom/18'>详情</Link>
    

    注册路由(声明接收):

    <Route path="/demo/test/:name/:age" component={Test}></Route>
    

    接收参数:

    const {name,age} = this.props.match.params
    
  2. search参数

    urlencoded格式编码:key=value&key=value

    react自带一个库,处理这种转换

    import qs from 'querystring'
    

    search参数无需声明接收,正常注册路由即可

    路由链接(携带参数):

    <Link to='/demo/test?name=tom&age=18'>详情</Link>
    

    注册路由(无需声明,正常注册即可):

    <Route path="/demo/test" component={Test} />
    

    接收参数:

    const { search } = this.props.location
    

    备注:获取到的search是urlencoded编码字符串,需要借助querystring解析

  3. state参数

    location是history的属性

    路由链接(携带参数):

    <Link to={{path:'/demo/test',state:{name:'tom',age:18}}}>详情</Link>
    

    注册路由(无需声明,正常注册即可):

    <Route path="/demo/test" component={Test}/>
    

    接收参数:

    this.props.location.state
    

    备注:刷新也是可以

    默认是push模式,开启replace需要在Link标签里面加replace={true}

编程式路由导航

不用用户自己点,可以自己控制跳转

一般组件没有history,只有路由组件有

借助this.props.history对象上的API操作对路由跳转、前进和后退

this.props.history.push()

this.props.history.replace()

this.props.history.goBack()

this.props.history.goForward()

this.props.history.go()

路由组件:接收三个固定的属性

  • history:

    go:f go(n)

    goBack:f goBack()

    goForward: f goForward()

    push:f push(path,state)

    replace:f replace(path,state)

  • location:

    pathname:"/about"

    search:""

    state:undefined

  • match:

    params:{}

    path:"/about"

    url:"/about"

withRouter

一般组件在withRouter处理之后,也可以有路由组件的API

withRouter返回值是一个新组件

import {withRouter} from 'react-router-dom'
export default withRouter(XXX)   
BrowserRouter与HashRouter的区别
  1. 底层原理不一样:

    BrowserRouter使用的是H5的history API,不兼容IE9及以下版本。HashRouter使用的是URL的哈希值

  2. url表现形式一样

    BrowserRouter的路径中没有#,例如:localhost:3000/demo/test

    HashRouter的路径包含#,例如:localhost:3000/#/demo/test

  3. 刷新后对路由state参数的影响

    • BrowserRouter没有任何影响,因为state保存在history对象中
    • HashRouter刷新后会导致路由state参数的丢失
  4. 备注:HashRouter可以用于解决一些路径错误相关的问题

antd使用

antd4.x文档不详细,可以去看3.x

yarn eject会把react里面webpack的核心配置都给暴露出来

react里面的插件都需要下载、引入、实例化

按需引入:

去官网文档查看,需要对create-react-app的默认配置进行自定义,引入react-app-rewired并修改package.json里的启动配置

在项目的根目录创建一个config-overrides.js用于修改默认配置

yarn add react-app-rewired customize-cra
antd自定义主题
  1. 安装依赖

    先安装less和less-loader react-app-rewired customize-cra babel-plugin-import less less-loader

  2. 修改package.json

    "scripts":{
        "start":"react-app-rewired start",
        "build":"react-app-rewired build",
        "test":"react-app-rewired test",
        "eject":"react-scripts eject",
    }
    
  3. 根目录下创建config-overrides.js

module.exports = override(
    fixBabelImports('import',{
        libraryName:'antd',
        libraryDirectory:'es',
        style:true
    })
    addLessLoader({
    	lessOptions:{
            javascriptEnable: true,
            modifyVars:{'@primary-color':'orange'}    
    	}
    })
)

备注:不需要在组件里面亲自引入样式了,即import 'antd/dist/antd.css'应该删掉

redux
  1. redux是专门用于做状态管理的JS库
  2. 作用:集中式管理react应用中多个组件的共享状态

什么情况下使用redux:

  1. 某个组件状态,需要让其它组件可以随时拿到(共享)
  2. 一个组件需要改变另一个组件状态(通信)
  3. 总体原则:能不用刘不用,如果不用比较吃力才考虑使用

在这里插入图片描述

store相当于一个指挥者

reducers是执行者,加工状态,需要传入之前的状态

reducer可以做两件事,初始化状态、加工状态

react components相当于客人

action creators相当于服务员

store相当于老板

reducers相当于后厨

redux的使用

在src目录下,新建一个redux文件夹,里面新建一个count_reducer.js和store.js

store.js

// 该文件专门用于暴露一个store对象,整个应用只有一个store对象
// 引入createStore,专门用于创建redux中最核心的store对象
import { createStore } form 'redux'
// 引入为Count组件服务的reducer
import countReducer from './count_reducer'

export default createStore(countReducer)

count_reducer.js

// 该文件时用于创建一个为count组件服务的reducer,reducer的本质就是一个函数
// reducer会接到两个参数,分别为之前的状态(preState)动作对象(action)
export default function countReducer(preState,action){
    const {type,data} = action
    // 根据type决定如何加工数据
    switch(type) {
        case 'increment':
            return preState + data
        case 'decrement':
            return preState - data
        default:
            return 0
    }
}

在需要的组件里面引用

import { state } form 'xxx'       //redux的store
// 一开始初始化的时候没有type
// 在函数里面调用方法
export default class Count extends Component {
    componentDidMount(){  // redux中任意状态变化都会调用这个回调
        // 检测redux中任意状态的变化,只要变化就调用render
        store.subscribe(()=>{
            this.setState({})
        })
    }
    increment = ()=>{
        const {value} = this.selectNumber
        store.dispatch({type:'increment',data:value*1})
    }    
}

this.setState可以帮忙改状态,还会重新render一下

// redux更新可以放在index.js里面
import store from './redux/store'
store.subscribe(()=>{
    ReactDOM.render(<App/>,document.getElementById('root'))
})
redux简单总结
  1. 去除Count组件自身状态

  2. src下建立:

    -redux

    ​ -store.js

    ​ -count_reducer.js

  3. store.js

    • 引入redux中的createStore函数,创建一个store
    • createStore调用时要传入一个为其服务的reducer
    • 记得暴露store对象
  4. count_reducer.js

    • reducer的本质是一个函数,接收:preState,action 返回加工后的状态
    • reducer有两个作用:初始化状态,加工状态
    • reducer被第一次调用时,是store自动触发的,传递的preState是undefined
  5. 在index.js中检测store中状态的改变,一旦发生改变重新渲染<App/>

    备注:redux只负责管理状态,至于状态的改变驱动着页面的展示,要靠我们自己写

count_action.js

// 专门为Count组件生成action对象
export const createIncrementAction = data => ({type:'increment',data})
export const createDecrementAction = data => ({type:'decrement',data})

在组件中使用的时候,直接

// 引入这个count_action.js

constant.js

// redux下面定义常量的地方,防止在redux的其他文件里面写错
export const INCREMENT = 'increment'
异步action

action:

  1. Object{} 同步
  2. function 异步

同步action,action的值为Object类型的一般对象

异步action,就是action的值为函数

异步action,就是action的值为函数

// 在count_action里面添加异步,就是action的值为函数,异步action中一般都会调用同步action,异步action不是必须要用的
export const createIncrementAsyncAction = (data,time) => {
    return (dispatch)=>{
        setTimeout(()=>{
            dispatch(createIncrementAction(data))
        },time)
    }
}

传给store会报错,需要一个中间件

npm i redux-thunk -S

在store.js里面引入

import thunk from 'redux-thunk'
import {createStore, applyMiddleware} from 'redux'
// 暴露store
export default createStore(countReducer,applyMiddleware(thunk))
react-redux

react官方出的

  1. 所以的UI组件都应该包裹一个容器组件,他们是父子关系
  2. 容器组件是真正和redux打交道的,里面可以随意的使用redux的api
  3. UI组件中不能使用任何redux的api
  4. 容器组件会传给UI组件
    • redux中所保存的状态
    • 用于操作状态的方法
  5. 备注:容器给UI传递:状态、操作状态的方法,均通过props传递

img

Count(UI组件)放到components里面,Count(容器组件)放到containers里面

react-redux基本使用

  • 明确两个概念

    • UI组件:不能使用任何redux和api,只负责页面的呈现、交互等
    • 容器组件:负责和redux通信,将结果交给UI组件
  • 如何创建一个容器组件-靠react-redux的connect函数

    connect(mapStateToProps, mapDispatchToProps)(UI组件)

    • mapStateToProps:映射状态,返回值是一个对象
    • mapDispatchToProps:映射操作状态的方法
  • 备注1:容器组件中的store是靠props传进去的,而不是在容器组件中直接引入

  • 备注2:mapDispatchToProps,也可以是一个对象,api自动调用dispatch

react-redux优化
  1. 容器组件和UI组件整合成一个文件

  2. 无需自己给容器组件传递store,给<App/>包裹一个<Provider store={store}>即可

  3. 使用了react-redux后也不用再自己检测redux中状态的改变了,容器组件可以自动完成这个工作

  4. mapDispatchToProps也可以简单的写成一个对象

  5. 一个组件要和redux"打交道"需要经过几步:

    • 定义好UI组件---不暴露

    • 引入connect生成一个容器组件,并暴露,写法如下:

      connect(
      	state => ({key:value})
          {key:xxxxAction}
      )(UI组件)
      
    • 在UI组件中通过this.props.xxxx读取和操作状态

react-redux练习

containers文件夹里面,通常actions里面放一堆文件、reducers里面放一些文件,constant.js是一个单独的文件store.js也是一个单独的文件

案例react-redux数据共享版

当有多个reducer的时候,需要在reducer里面引入

import {combineReducers} from 'redux'
const allReducer = combineReducers({
    he:countReducer,
    rens:personReducer
})
// 暴露store
export default createStore(allReducer,applyMiddleware(thunk))
  1. 定义一个person组件和count组件通过redux共享数据
  2. 为person组件编写,reducer、action配置constant常量
  3. 重点Person的reducer和Count的Reducer要使用combineReducers进行合并,合并后的总状态是一个对象
  4. 交给store的总是reducer,最后注意在组件中取出状态的时候,记得取到位
纯函数
  1. 同样的输入(实参),必定得到同样的输出(返回)
  2. 必须遵守以下约束
    • 不得改写参数数据
    • 不会产生任何副作用,例如网络请求,输入和输出设备
    • 不能调用Date.now()或者Math.random()等不纯的方法
  3. redux的reducer函数必须是一个纯函数
redux开发者工具

安装chrome插件redux devtools

项目里面安装redux-devtools-extension

import {composeWithDevTools} from 'redux-devtools-extension'

export default createStore(allReducer,composeWithDevTools(applyMiddleware(thunk)))

react扩展

标签:react,基础,笔记,React,action,路由,组件,redux,store
From: https://www.cnblogs.com/oaoa/p/16966096.html

相关文章

  • 每天一点基础K8S--K8S中的控制器replicaset
    1、replicaset基础前面都是通过yaml文件创建自主式pod,如果pod中的服务都是通过单纯的重启策略restartPolicy(always、OnFailure、never)来重启pod,但是如果pod运行的node异......
  • react 项目初建 保存时报错 Uncaught ReferenceError: process is not defined
    新创建了一个react项目,想做平时学习使用。发现保存代码重新编译后,控制台会报错,并且页面出现个iframe标签,页面动弹不得    解决方案:删除 node_modules文件......
  • 用NetCore + ReactJS 实现一个前后端分离的网站 (6) 缓存机制
    1.前言对于实时性要求不高的资源,我们一般可以利用缓存机制来降低数据库的请求压力。轻量级的应用可以用自带的MemoryCache,如果对缓存的高并发、持久化有要求的可以用Red......
  • Angular 复习与进阶系列 – 谈谈 ASP.NET Core & Angular & React 在业务开发上各自的
    前言日常,我的开发都围绕着ASP.NETCore和Angular.这篇想聊聊它们各自的特点和解决问题的方式. 以及最重要的,我们该在什么时候使用何种方案更为妥当.  浅谈......
  • 一. TypeScript基础
    1.TS运行环境方式一:通过webpack,配置本地的TypeScript编译环境和开启一个本地服务,可以直接运行在浏览器上方式二:通过ts-node库,为TypeScript的运行提供执行环境安装ts......
  • React实战篇——四、在React中使用Redux
    四、在React中使用Redux1,Redux可以和很多库一起用,为了方便在React中使用Redux,我们需要单独安装react-redux2,展示组件和容器组件展示组件负责应用的UI展示,也就是组件如何......
  • 区块链技术基础
    个人微博:https://weibo.com/u/2434823180 区块链通识区块链技术基础区块链原理挖矿原理软分叉和硬分叉: https://www.cnblogs.com/......
  • (转)Guava 是个风火轮之基础工具(1)
    前言Guava是Java开发者的好朋友。虽然我在开发中使用Guava很长时间了,GuavaAPI的身影遍及我写的生产代码的每个角落,但是我用到的功能只是Guav......
  • Python入门笔记(一)
    文章目录​​第一章初识python​​​​1.1python介绍​​​​1.2Python、pycharm、anaconda安装​​​​1.3print()和input()​​​​1.4保留字和标识符​​​​1.5变......
  • Python入门笔记(三)
    文章目录​​第十二章异常处理​​​​12.1常见异常​​​​12.2处理异常:try--except​​​​12.3创建异常类型:raise语句​​​​12.4断言:assert​​​​12.5存储数......