成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

手挽手帶你學(xué)React:四檔(下篇)一步一步學(xué)會(huì)react-redux

FullStackDeveloper / 2054人閱讀

摘要:手挽手帶你學(xué)入門四檔用人話教你,理解架構(gòu),以及運(yùn)用在中。學(xué)完這一章,你就可以開始自己的項(xiàng)目了。結(jié)合搭建基礎(chǔ)環(huán)境我們上一章講過了的原理,內(nèi)部是有一個(gè)的,只有才可以控制它變化。

手挽手帶你學(xué)React入門四檔,用人話教你react-redux,理解redux架構(gòu),以及運(yùn)用在react中。學(xué)完這一章,你就可以開始自己的react項(xiàng)目了。

視頻教程

上一篇我們自己實(shí)現(xiàn)了Redux,這一篇我們來看看如何去實(shí)現(xiàn)一個(gè)react-redux

開始之前

本文需要用到的知識(shí)

首先你要會(huì)React的基礎(chǔ)(這是廢話)
高階組件
React context
滿足這三項(xiàng)我們開始往下看。
react結(jié)合redux

搭建基礎(chǔ)環(huán)境

我們上一章講過了redux的原理,內(nèi)部是有一個(gè)store的,store只有dispatch才可以控制它變化。還有在文章一開始我們講了React context 可以跨組件傳遞數(shù)據(jù),那么到現(xiàn)在你想到把我們的store掛到最外層一個(gè)組件的context上了嗎?話不多說 開始!

我們先修改一下我們的react文件(注意不是redux.html這個(gè)文件了)

    // App.js
import React,{Component} from "react"
import PropTypes from "prop-types"  //引入

export default class App extends Component {
    constructor(){
        super()
        this.state={
          
        }
    }
    componentWillMount(){
        // console.log(hashHistory)
    }
    render() {
        return (
            
) } } // 為了展示效果定義子組件一 class Children extends Component{ constructor(){ super() this.state={ } } render(){ return(

我是腦袋

我是身體

) } } // 為了展示效果定義子組件二 ChildrenTwo 是 Children的子組件 但是卻通過context拿到了App組件拿過來的值 (越級(jí)傳遞) class ChildrenTwo extends Component{ constructor(){ super() this.state={ } } render(){ return(
) } }
創(chuàng)建基本store

現(xiàn)在我們做好了示例文件的基礎(chǔ)模板了,然后我們需要?jiǎng)?chuàng)建一個(gè)store.js

// store.js

// 這是我們的 reducer
const changeDom = (state,action) => {
    if(!state)return{
        text : "我是實(shí)例文字",
        color : "red"
    }
    switch(action.type){
        case "CHANGE_TEXT":
        return{
            ...state,
            text:action.text
        }
        case "CHANGE_COLOR":
        return{
            ...state,
            color:action.color
        }
        default:
        return state
    }
}


const creatStore = (reducer)=>{
    let state = null
    const listeners = []
    const subscribe = (liestner)=>listeners.push(liestner)
    const getState = ()=>state
    const dispatch=(action)=>{
        state = reducer(state,action)
        listeners.map(item=>item())
    }
    dispatch({})
    return { getState, dispatch, subscribe }
}
export {creatStore,changeDom}
結(jié)合context使用store

我們現(xiàn)在把我們的子組件通過context公用App的store,并且把渲染方法和dispatch應(yīng)用進(jìn)去

    // App.js
    import React,{Component} from "react"
    import PropTypes from "prop-types"  //引入
    
    // 引入 store
    import {changeDom,creatStore} from "./store"

    const store = creatStore(changeDom)

    export default class App extends Component {
        // 我們?cè)谶@里把store掛到context上
        static childContextTypes = {
            store: PropTypes.object
        }
        getChildContext(){
            return{store}
        }
        constructor(){
            super()
            this.state={
              
            }
        }
        componentWillMount(){
            // console.log(hashHistory)
        }
        render() {
            return (
                
) } } // 這里我們?cè)偃バ薷奈覀兊淖訉O組建 讓他們可以拿到 store // 為了展示效果定義子組件一 class Children extends Component{ static contextTypes = { store: PropTypes.object } constructor(){ super() this.state={ color:"", text:"" } } // 這里我們定義兩個(gè)渲染方法 getColor(){ let store = this.context.store.getState() this.setState({color:store.color}) } // 這里我們定義兩個(gè)渲染方法 getText(){ let store = this.context.store.getState() this.setState({text:store.text}) } // 這里組件加載之前渲染 componentWillMount(){ this.getColor() this.getText() } render(){ return(

{this.state.text}

) } } // 為了展示效果定義子組件二 ChildrenTwo 是 Children的子組件 但是卻通過context拿到了App組件拿過來的值 (越級(jí)傳遞) class ChildrenTwo extends Component{ static contextTypes = { store: PropTypes.object } constructor(){ super() this.state={ } } // 這里我們定義 兩個(gè) action changeColor=()=>{ this.context.store.dispatch({type:"CHANGE_COLOR",color:"green"}) console.log(this.context.store.getState()) } changeText=()=>{ this.context.store.dispatch({type:"CHANGE_TEXT",text:"我變了"}) console.log(this.context.store.getState()) //這里方便大家看到數(shù)據(jù)變了 } render(){ return(
) } }

顯然 現(xiàn)在我們點(diǎn)擊按鈕的時(shí)候,并沒有發(fā)生任何事情,可是我們看console可以看到,store的數(shù)據(jù)確實(shí)變化了,這是為什么呢?很簡單 我們沒有把dom監(jiān)聽放進(jìn)來,下一步我們要設(shè)置監(jiān)聽。

設(shè)置dom監(jiān)聽
    // Children組件 我們?cè)赾omponentWillMount 中添加監(jiān)聽
    class Children extends Component{
        static contextTypes = {
            store: PropTypes.object
          }
        constructor(){
            super()
            this.state={
                color:"",
                text:""
            }
        }

        // 這里我們定義兩個(gè)渲染方法
        getColor(){
            let store = this.context.store.getState()
            this.setState({color:store.color})
        }

        // 這里我們定義兩個(gè)渲染方法

        getText(){
            let store = this.context.store.getState()
            this.setState({text:store.text})
        }

        // 這里組件加載之前渲染
        componentWillMount(){
            this.getColor()
            this.getText()
            this.context.store.subscribe(()=>{this.getColor()})
            this.context.store.subscribe(()=>{this.getText()})// 使用箭頭函數(shù) 保證this指向

        }
        render(){
            return(
                

{this.state.text}

) } }

到這里我們已經(jīng)簡單地實(shí)現(xiàn)了react-redux,可是大家有沒有覺得怪怪的?

開始創(chuàng)建connect

沒錯(cuò) 到這里我們頻繁使用context,并且每個(gè)組件都要去手動(dòng)掛context 這是相當(dāng)麻煩的,現(xiàn)在我們想要讓這些東西都自動(dòng)包裹 自動(dòng)生成,我們?cè)僭趺醋瞿亍?br>我們創(chuàng)建一個(gè)高階組件就可以搞定這個(gè)問題了(一個(gè)函數(shù),接收一個(gè)組件,生成另外一個(gè)包裝好的新組件)

import React, { Component } from "react"
import PropTypes from "prop-types"

export const connect = (WrappedComponent)=>{
    class Connect extends Component{
        static contextTypes = {
            store: PropTypes.object
        }
        render(){
            return
        }
    }
    return Connect
}

我們這里通過 connect函數(shù)來接收一個(gè)組件 經(jīng)過一層封裝 返回一個(gè)包裝了context的組件 但是現(xiàn)在這樣的話 我們每次使用都還需要大量的書寫 this.context 是不是相當(dāng)惡心呢? 并且每個(gè)組件都拿到了store 有很多事它并不需要的,這時(shí)候我們就需要告訴這個(gè)高階組件,我這里就只需要這些東西。這就是 mapStateToProps

import React, { Component } from "react"
import PropTypes from "prop-types"

// 示例 這里不用
// const mapStateToProps=(state)=>{
//     return{
//         color:state.color,
//         text:state.text,
//     }
// }  就是我們需要從store拿出來的東西

const connect= (mapStateToProps) => (WrappedComponent)=>{
    class Connect extends Component{
        static contextTypes = {
            store: PropTypes.object
        }
        render(){
            const store = this.context.stote
            let stateProps = mapStateToProps(store.getState())
            然后我們把stateProps用...展開
            return
        }
    }
    return Connect
}

現(xiàn)在我們利用Connect改造我們的組件

    // App.js
    import React,{Component} from "react"
    import PropTypes from "prop-types"  //引入
    
    // 引入 store
    import {changeDom,creatStore} from "./store"

    const store = creatStore(changeDom)

// ________________________________把connect_____________________________寫在app.js 為的是不引入了 實(shí)際上它是多帶帶拆分在react-redux中的
    const connect = (mapStateToProps) => (WrappedComponent)=>{
        class Connect extends Component{
            static contextTypes = {
                store: PropTypes.object
            }
            render(){
                const store = this.context.store
                console.log(store)
                let stateProps = mapStateToProps(store.getState())
                // 然后我們把stateProps用...展開
                return
            }
        }
        return Connect
    }
// ________________________________把connect_____________________________寫在app.js 為的是不引入了 實(shí)際上它是多帶帶拆分在react-redux中的

// app.js 的其他內(nèi)容...

// ________________________________對(duì)組件一進(jìn)行修改_____________________________
    
    class Children extends Component{
        constructor(){
            super()
            this.state={
                color:"",
                text:""
            }
        }

        render(){
            return(
                

{this.props.text}

) } } const mapStateToProps = (state) => { return { color: state.color, text:state.text } } Children = connect(mapStateToProps)(Children) // ________________________________對(duì)組件一進(jìn)行修改_____________________________

現(xiàn)在我們成功把store里面的所有東西都掛到了props上面,我們不需要去依賴context,只需要調(diào)用props即可。剩下的就是我們的數(shù)據(jù)變更,渲染還有監(jiān)聽了!

對(duì) connect 進(jìn)一步改造 其實(shí)就是像我們上面的組件一樣 掛載監(jiān)聽

    const connect = (mapStateToProps) => (WrappedComponent)=>{
        class Connect extends Component{
            static contextTypes = {
                store: PropTypes.object
            }
            componentWillMount(){
                const{store} = this.context
                this.updata()
                store.subscribe(()=>this.updata())
            }
            updata=()=>{
                 const { store } = this.context
                 let stateProps = mapStateToProps(store.getState(), this.props)
                  this.setState({
                        allProps: { // 整合普通的 props 和從 state 生成的 props
                        ...stateProps,
                        ...this.props
                        }
                    })
            }
            render(){
                // 然后我們把a(bǔ)llProps用...展開
                return
            }
        }
        return Connect
    }
mapDispatchToProps

state我們改造完了,dispatch 是不是也要一起改造呢?答案是肯定的,mapDispatchToProps就是做這個(gè)的。
我們看看 mapDispatchToProps 是怎么寫的 我們倒著推

//const mapDispatchToProps = (dispatch) => {  // 傳入的是 dispatch
//  return {  //返回一個(gè)函數(shù)
//    changeColor: (color) => {
//      dispatch({ type: "CHANGE_COLOR", color: color })
//    }
//  }
//}


const connect = (mapStateToProps,mapDispatchToProps) => (WrappedComponent)=>{
    class Connect extends Component{
        static contextTypes = {
            store: PropTypes.object
        }
        componentWillMount(){
            const{store} = this.context
            this.updata()
            console.log(store)
            store.subscribe(()=>this.updata())
        }
        updata=()=>{
             const { store } = this.context
            let stateProps = mapStateToProps?mapStateToProps(store.getState(), this.props):{}
             let dispatchProps = mapDispatchToProps?mapDispatchToProps(store.dispatch,this.props):{}
             // 我們要考慮到空值處理
              this.setState({
                    allProps: { // 整合普通的 props 和從 state 生成的 props
                    ...stateProps,
                    ...this.props
                    ...dispatchProps,
                    }
                })
        }
        render(){
            // 然后我們把a(bǔ)llProps用...展開
            return
        }
    }
    return Connect
}

到這里我們可以把ChildrenTwo的代碼也改造了

    
    class ChildrenTwo extends Component{
        constructor(){
            super()
            this.state={
            
            }
        }
        render(){
            console.log(this.props)
            return(
                
) } } const mapDispatchToProps = (dispatch)=>{ return { //返回一個(gè)函數(shù) changeColor: (color) => { dispatch({ type: "CHANGE_COLOR", color: color }) }, changeText:(text)=>{ dispatch({ type: "CHANGE_TEXT", text: text }) } } } ChildrenTwo = connect(null,mapDispatchToProps)(ChildrenTwo)

到現(xiàn)在 一個(gè)簡單的 react-redux已經(jīng)差不多了,但是react-redux里面還有一個(gè) 但是我們還沒有這東西 這是什么呢?實(shí)際上它做的事跟我們App這個(gè)組件做的事一毛一樣,主要就是把store掛到context上。

    export class Provider extends Component{
        static childContextTypes = {
            store: PropTypes.object
        }
        getChildContext(){
            return{store}
        }
        render() {
            return (
                
{this.props.children}
) } }

于是 我們現(xiàn)在的代碼變成了這樣

好了 道理講完了 我們現(xiàn)在開始拆分代碼啦。

// store.js
export const creatStore = (reducer)=>{
    let state = null
    const listeners = []
    const subscribe = (liestner)=>listeners.push(liestner)
    const getState = ()=>state
    const dispatch=(action)=>{
        state = reducer(state,action)
        listeners.map(item=>item())
    }
    dispatch({})
    return { getState, dispatch, subscribe }
}
// reducer.js

// 這是我們的 reducer  可以多帶帶拆分成一個(gè)js文件 自己拆吧
export const changeDom = (state,action) => {
    if(!state)return{
        text : "我是實(shí)例文字",
        color : "red"
    }
    switch(action.type){
        case "CHANGE_TEXT":
        return{
            ...state,
            text:action.text
        }
        case "CHANGE_COLOR":
        return{
            ...state,
            color:action.color
        }
        default:
        return state
    }
}
// react-redux.js
import React,{Component} from "react"
import PropTypes from "prop-types"  //引入

export const connect = (mapStateToProps,mapDispatchToProps) => (WrappedComponent)=>{
    class Connect extends Component{
        static contextTypes = {
            store: PropTypes.object
        }
        componentWillMount(){
            const{store} = this.context
            this.updata()
            store.subscribe(()=>this.updata())
        }
        updata=()=>{
             const { store } = this.context
             let stateProps = mapStateToProps?mapStateToProps(store.getState(), this.props):{}
             let dispatchProps = mapDispatchToProps?mapDispatchToProps(store.dispatch,this.props):{}

             // 做一下空值處理
              this.setState({
                    allProps: { // 整合普通的 props 和從 state 生成的 props
                    ...stateProps,
                    ...this.props,
                    ...dispatchProps,
                    }
                })
        }
        render(){
            // 然后我們把a(bǔ)llProps用...展開
            return
        }
    }
    return Connect
}


export class Provider extends Component{
        static childContextTypes = {
            store: PropTypes.object
        }
        getChildContext(){
            return {store:this.props.store}
        }
        render() {
            return (
                
{this.props.children}
) } }
// App.js
import React,{Component} from "react"
import Children from "./Children"
import ChildrenTwo from "./ChildrenTwo"
export default class App extends Component {
    constructor(){
        super()
        this.state={
          
        }
    }

    render() {
        return (
            
) } }
// Children.js
import React,{Component} from "react"
import{connect} from "./react-redux.js"
class Children extends Component{
    constructor(){
        super()
        this.state={
            color:"",
            text:""
        }
    }

    render(){
        return(
            

{this.props.text}

) } } const mapStateToProps = (state) => { return { color: state.color, text:state.text } } Children = connect(mapStateToProps)(Children) export default Children
// ChildrenTwo.js

import React,{Component} from "react"
import{connect} from "./react-redux.js"
class ChildrenTwo extends Component{
        constructor(){
            super()
            this.state={
            
            }
        }
        render(){
            return(
                
) } } const mapDispatchToProps = (dispatch)=>{ return { //返回一個(gè)函數(shù) changeColor: (color) => { dispatch({ type: "CHANGE_COLOR", color: color }) }, changeText:(text)=>{ dispatch({ type: "CHANGE_TEXT", text: text }) } } } ChildrenTwo = connect(null,mapDispatchToProps)(ChildrenTwo) export default ChildrenTwo

拆完了的代碼是不是簡單明了
真正工作里面 我們需要多做兩步

npm i redux --save
npm i react-redux --save

然后 我們
對(duì)照著修改這幾個(gè)位置

// creatStore 在 redux 插件中
// connect,Provider 在 react-redux 插件中
// 也就是用到哪里了 對(duì)應(yīng)修改哪里 改完了你就發(fā)現(xiàn)了新大陸了
import { createStore } from "redux"
import { connect,Provider } from "react-redux"

不知不覺發(fā)現(xiàn)自己不僅僅會(huì)用了react-redux 并且還自己實(shí)現(xiàn)了一個(gè)react-redux 很舒坦的呢

總結(jié)

在我們四檔下篇到這里結(jié)束了,這就是react-redux的實(shí)現(xiàn)和寫法,大家自己去實(shí)現(xiàn)真正的寫法,這里不做演示相當(dāng)于給大家留個(gè)作業(yè),有錯(cuò)誤或者是有建議 或者有不懂的地方 掃碼加我微信給大家解答。

視頻制作中

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/101641.html

相關(guān)文章

  • 手挽手帶你學(xué)React四檔(上)步一學(xué)會(huì)react-redux (自己寫個(gè)Redux)

    手挽手帶你學(xué)React入門四檔,用人話教你react-redux,理解redux架構(gòu),以及運(yùn)用在react中。學(xué)完這一章,你就可以開始自己的react項(xiàng)目了。 之前在思否看到過某個(gè)大神的redux搭建 忘記了大神的名字 這里只記得內(nèi)容了 憑借記憶和當(dāng)時(shí)的學(xué)習(xí)路線寫下本文 隔空感謝 本人學(xué)習(xí)react-redux的時(shí)候遇到了很多坎,特別是不理解為什么這么用,這是什么東西,用來做什么。加上各種名詞讓人...

    sixgo 評(píng)論0 收藏0
  • 手挽手帶你學(xué)React:一檔 React環(huán)境搭建,語法規(guī)則,基礎(chǔ)使用

    摘要:當(dāng)屬性是一個(gè)回調(diào)函數(shù)時(shí),函數(shù)接收底層元素或類實(shí)例取決于元素的類型作為參數(shù)。 手挽手帶你學(xué)React入門第一期,帶你熟悉React的語法規(guī)則,消除對(duì)JSX的恐懼感,由于現(xiàn)在開發(fā)中都是使用ES6語法開發(fā)React,所以這次也使用ES6的模式進(jìn)行教學(xué),如果大家對(duì)ES6不熟悉的話,先去看看class相關(guān)內(nèi)容吧,這里我也慢慢帶大家一步一步學(xué)會(huì)React。 視頻教程 視頻教程可移步我的個(gè)人博客:h...

    BicycleWarrior 評(píng)論0 收藏0
  • 手挽手帶你學(xué)VUE:一檔 VUE簡介以及常用內(nèi)部指令

    摘要:這樣,我們用寫的就寫好了。真的假的大家可以看到,這些在插值表達(dá)式內(nèi)的表達(dá)式直接返回了運(yùn)行完成的結(jié)果,值得一提的是,差值表達(dá)式內(nèi)的規(guī)則和標(biāo)簽內(nèi)的規(guī)則是類似的。 視頻教程 由于思否不能插入視頻,視頻請(qǐng)大家移步,http://www.henrongyi.top 什么是VUE VUE是一套用于構(gòu)建用戶界面的漸進(jìn)式框架,VUE并不是一個(gè)真正意義上的mvvm框架,它更傾向是一種數(shù)據(jù)驅(qū)動(dòng)框架.所以我...

    不知名網(wǎng)友 評(píng)論0 收藏0
  • 手挽手帶你學(xué)VUE:一檔 VUE簡介以及常用內(nèi)部指令

    摘要:這樣,我們用寫的就寫好了。真的假的大家可以看到,這些在插值表達(dá)式內(nèi)的表達(dá)式直接返回了運(yùn)行完成的結(jié)果,值得一提的是,差值表達(dá)式內(nèi)的規(guī)則和標(biāo)簽內(nèi)的規(guī)則是類似的。 視頻教程 由于思否不能插入視頻,視頻請(qǐng)大家移步,http://www.henrongyi.top 什么是VUE VUE是一套用于構(gòu)建用戶界面的漸進(jìn)式框架,VUE并不是一個(gè)真正意義上的mvvm框架,它更傾向是一種數(shù)據(jù)驅(qū)動(dòng)框架.所以我...

    go4it 評(píng)論0 收藏0
  • 手挽手帶你學(xué)VUE:四檔 Vue-cli3 Vuex Vue-router

    摘要:安裝過后到命令行執(zhí)行檢查版本,如果彈出版本的話恭喜你安裝成功,我們開始進(jìn)行下面的步驟了。全局安裝的包名稱由改成了。如果你已經(jīng)全局安裝了舊版本的或,你需要先通過卸載它。中的非常類似于事件每個(gè)都有一個(gè)字符串的事件類型和一個(gè)回調(diào)函數(shù)。 視頻教程 由于思否不支持視頻外鏈,視頻請(qǐng)移步http://www.henrongyi.top 你能學(xué)到什么 在這一期的學(xué)習(xí)進(jìn)度中,我們會(huì)開始學(xué)習(xí)在我們工作開...

    Sourcelink 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

FullStackDeveloper

|高級(jí)講師

TA的文章

閱讀更多
最新活動(dòng)
閱讀需要支付1元查看
<