ReactNative开发-Redux学习记录

前言

Redux这东西……真是令人又爱又恨。前前后后看了不下10篇文章+官方文档来理解这东西。差点就从入门到放弃。不过框架这东西,百变不离其中,懂得分工合作的思想后,什么MVCMVPMVVC都不在话下。借助这篇记录,来巩固下自己学习和使用Redux的过程。

本文章配套Demo点我下载

什么是Redux?

  • 官方说法Redux是一个状态容器管理工具。将所有状态进行统一管理,适用于多交互,多数据的场景。

  • 我的理解:我们可以把采用了Redux后的应用看作是一个component,整个应用只有一个state,所有页面都可以拿到这个state中的内容。当这个state中的内容发生改变的时候,与之关联的页面也会跟着刷新状态。

基本概念

首先,在未使用Redux的项目中,组件(父组件、子组件、孙子组件)间都各自含有的stateprops属性。而且props是从父级组件上分发下来的属性,只能是从上往下走;而state是组件内部自行管理的状态属性。因此整个应用并没有数据向上回溯的能力。要么从上面单向得到并向下级分发,要么自行内部消化刷新页面状态。正如下图所示:

因此,一但应用场景的页面较多,交互复杂的时候,每次都去通过state重新刷新页面引起页面变化,就会出现卡卡卡卡卡的情况。

那么,采用了Redux后又会怎样呢?先来看一下App结构图:

最后,从上图可以看到采用Redux后的App结构比没采用Redux时在最外包裹了一层Provider。接下来我们就从这个Prodiver开始,逐个解析StoreActionReducer

  • Provider:整个App的容器,位于App的最外层。能够使得原有组件变得接受Redux的store作为props。可以理解为所有的页面都是它的子组件,这样就能够像父组件向子组件传递props。

    例如想通过改变孙子组件3的同时刷新孙子组件2的页面状态,这时后就可以先把孙子组件3要改变的状态通知告诉Provider,然后再由Provider刷新孙子组件2的页面状态。这样就能使得整个App内的组件都拥有数据往上回溯的能力。

  • Store存放和管理App内所有组件State的地方。

    整个App只允许有一个Store对象,改变了Store中的state就可以实现我们改变UI的操作。

  • Action:用户触发或程序触发的一个普通对象。

State 的变化,会导致View的变化。但是,用户接触不到State,只能接触到View。所以,State的变化必须是View导致的。Action就是View 发出的通知,表示State应该要发生变化了。例如要实现登录操作,就要发起登录action,当登录成功的就返回View显示登录成功的状态和信息,登录失败就返回View显示登录失败的状态和信息。

  • Reducer:根据action操作来做出不同的数据响应,返回一个新的state

Redux状态管理流程

梳理一下流程就是:用户触发action→部署触发的action后通知reducer→reducer→新store→反馈到UI上更新页面

实战案例:App登录操作

说了一大堆原理和废话后,还是得看实战案例。不然脑袋记不住。

实例需求

我们要做一个App登录操作,能让用户输入账号密码登录。登录成功就弹出提醒登录成功,跳转到主页并显示出当前登录用户的信息。登录失败就弹出登录失败提醒。如果用户不登录也可以选择跳过登录,直接跳转到主页并弹出欢迎游客信息。

需求分析

通过需求,我们提炼出其中重要的点:

  • 登录成功状态
  • 登录失败状态
  • 跳过登录

具体操作

步骤一:创建

创建一个名为ReduxDemo的空项目。

打开终端,输入

react-native init ReduxDemo --version 0.48.1

步骤二:安装

进入项目根目录,在终端输入命令安装redux、react-navigation(用作跳转页面)、redux-thunk(稍后讲解)和redux-logger(稍后讲解)

npm install --save redux
npm install --save react-redux
npm install --save react-navigation
npm install --save redux-thunk
npm install --save redux-logger
npm install --save react-native-root-toast

步骤三:构建目录结构

在根目录下新建一个app文件夹,然后在这个文件夹下再分别创建actionscontantspagesreducersstore文件夹。如下图所示:

步骤四:统一入口,利用react-navigation实现页面跳转

修改index.android.jsindex.ios.js文件,两者都改成如下内容:

import React, { Component } from 'react';
import {
    AppRegistry,
} from 'react-native';
import App from './app/app';
AppRegistry.registerComponent('ReduxDemo', () => App);

在app文件下创建app.js文件,并添加一下内容:

import React from 'react';
import {
    StackNavigator,
    TabNavigator
} from 'react-navigation';
import Login from './pages/Login';
import Home from './pages/Home';

const App = StackNavigator({
    Login:{screen:Login},
    Home:{screen:Home}
},{
});

export default App;

接着在pages文件夹下添加登陆页和首页:

接着继续在Login文件夹下的index.js文件里添加如下内容:


import React, { Component } from 'react';
import {
    AppRegistry,
    StyleSheet,
    Text,
    View,
    Dimensions,
    TextInput,
    TouchableOpacity
} from 'react-native';

import {NavigationActions} from 'react-navigation'
const screenWidth = Dimensions.get('window').width;  //屏幕的宽度
const screenHeight = Dimensions.get('window').height;  //屏幕的高度

export default class Login extends Component {

    static navigationOptions = {
        header: null
    }

    constructor(props) {
        super(props);
        this.state = {
            uid:'',
            pwd:'',
        };
    }

    render() {
        return (
            <View style={styles.container}>
                <TextInput
                    style = {styles.textInput}
                    blurOnSubmit={true}
                    returnKeyType="done"
                    placeholder = '请输入账号'
                    selectionColor = "#bac3d4"
                    placeholderTextColor = '#bac3d4'
                    underlineColorAndroid = "transparent"
                    onChangeText={(text)=>{this.setState({uid:text})}}
                    value={this.state.uid}/>

                <TextInput
                    style = {styles.textInput}
                    blurOnSubmit={true}
                    returnKeyType="done"
                    placeholder = '请输入密码'
                    selectionColor = "#bac3d4"
                    placeholderTextColor = '#bac3d4'
                    underlineColorAndroid = "transparent"
                    onChangeText={(text)=>{this.setState({pwd:text})}}
                    value={this.state.pwd}/>

                <View style={styles.btn}>
                    <TouchableOpacity onPress={this.login}
                                      style={styles.radiusBtn}>
                        <Text style={styles.text}>登 录</Text>
                    </TouchableOpacity>
                    <TouchableOpacity onPress={this.skip}
                                      style={styles.radiusBtn}>
                        <Text style={styles.text}>跳 过</Text>
                    </TouchableOpacity>
                </View>

            </View>
        );
    }

    login = () => {
        this.props.navigation.navigate('Home')
    }

    skip = () => {
        this.props.navigation.navigate('Home')
    }
}

const styles = StyleSheet.create({
    container: {
        flex: 1,
        justifyContent: 'center',
        alignItems: 'center',
        backgroundColor: '#F5FCFF',
    },
    textInput:{
        width:0.8*screenWidth,
        height:0.08*screenHeight,
        color:'#1b6cec',
        fontSize:16,
        borderWidth: 1,
        borderRadius:8,
        borderColor:'#1b6cec',
        marginBottom: 15
    },
    btn:{
        height:0.25*screenHeight,
        justifyContent:'center',
        alignItems:'center'
    },
    radiusBtn:{
        backgroundColor:'#1b6cec',
        width:0.8*screenWidth,
        height:0.08*screenHeight,
        borderRadius:25,
        justifyContent:'center',
        alignItems:'center',
        marginBottom:8
    },
    text:{
        color:'#fff',
        fontSize:16
    }
});

接着继续在Home文件夹下的index.js文件里添加如下内容:

import React, { Component } from 'react';
import {
    AppRegistry,
    StyleSheet,
    Text,
    View
} from 'react-native';

export default class Login extends Component {

    static navigationOptions = {
        header: null
    }

    render() {
        return (
            <View style={styles.container}>
                <Text style={styles.welcome}>
                    Welcome to Home!
                </Text>
                <Text style={styles.instructions}>
                    To get started, edit index.android.js
                </Text>
                <Text style={styles.instructions}>
                    Double tap R on your keyboard to reload,{'\n'}
                    Shake or press menu button for dev menu
                </Text>
            </View>
        );
    }
}

const styles = StyleSheet.create({
    container: {
        flex: 1,
        justifyContent: 'center',
        alignItems: 'center',
        backgroundColor: '#F5FCFF',
    },
    welcome: {
        fontSize: 20,
        textAlign: 'center',
        margin: 10,
    },
    instructions: {
        textAlign: 'center',
        color: '#333333',
        marginBottom: 5,
    },
    textInput:{
        width:0.8*screenWidth,
        height:0.08*screenHeight,
        color:'#1b6cec',
        fontSize:16,
        borderWidth: 1,
        borderRadius:8,
        borderColor:'#1b6cec',
        marginBottom: 15
    }
});

到这里为止,App已经可以进行页面跳转啦。

步骤五:配置Redux

1.正如前面所说的,我们需要为使用了Redux的App项目最外层包裹一层Provider。因此在app文件夹下添加一个index.js文件,并使用Provider包裹整个程序的入口,并将store传递进去。代码如下:

import { AppRegistry,View,Text ,} from 'react-native';
import React, { Component } from 'react';
import {Provider} from 'react-redux';
import configureStore from './store/ConfigureStore';
// 调用 store 文件中的rootReducer常量中保存的方法
const store = configureStore();
import App from './app';

// 项目中使用了react-navigation,推荐的做法是将初始文件写在一个文件中,
// 所以app.js也可以看做是页面的初始化入口
export default class Root extends Component {
 render() {
     return (
         //包裹app
         <Provider store={store}>
             <App />
         </Provider>
     );
 }
};

AppRegistry.registerComponent('ReduxDemo', () => Root);

2.然后为了能让程序运行时首先索引到app文件夹下的index.js文件,需要把index.android.jsindex.ios.js文件修改成以下内容:

require('./app/index');

3.紧接着为store配置参数设置。在store文件夹下创建ConfigureStore文件,并添加以下内容:

// redux库里面提供的方法,创建store和middleware中间件
import {createStore , applyMiddleware} from 'redux';  
import thunk from 'redux-thunk';
import logger from 'redux-logger';
import RootReducer from '../reducers/RootReducer';

let middlewares = [];

middlewares.push(thunk);
if (__DEV__) {
 middlewares.push(logger);
}
// 通过applyMiddleware将中间件添加
const createStoreWithMiddleware = applyMiddleware(...middlewares)(createStore);
// 导出configureStore,里面携带着reducer,中间件,初始值
export default function configureStore(initialState){
 return createStoreWithMiddleware(RootReducer,initialState);
}

这里涉及到了新的知识:中间件,redux-thunk和redux-logger 。这里简单说下中间件的概念和代码中的两个是干什么用的:

中间件概念:用于为使用了Redux的项目添加额外的功能,如日志打印(redux-logger),异步请求(redux-thunk)等。

因为使用的中间件都是现成的,我们直接引入就好,这里就不再展开讲解了。

4.然后继续reducers文件夹下创建一个RootReducer.js文件,(由于Redux中只允许有一个store,当业务越来越庞大的时候,我们就需要拆分出N个reducer。这时候,就需要把这N个reducer组合起来,因此我们需要一个根reducer。),并添加以下代码:

import { combineReducers } from 'redux';
import LoginReducer from './LoginReducer';

//取决于这里你加入了多少 reducer
export default RootReducer = combineReducers({
 LoginReducer:LoginReducer
});

5.RootReducer.js中我们引入了一个LoginReducer(处理由用户或程序触发LoginAction后,分派过来的任务并根据对应的types,修改更新store中的某些值),因此我们也需要在reducers文件夹下创建一个LoginReducer.js并添加以下代码:

// ActionTypes里面存放着App中可能触发的事件
import * as types from './../contants/ActionTypes';
// 初始化值
const initialState = {
    showMsg:'',
    userName:'',
    tel:''
};
// 导出LoginReducer。
export default function LoginReducer(state = initialState, action){
    // 通过switch来判断types的值,在action中实现功能。
    switch (action.type) {
        // 当type=LOGIN_SUCCESS时,会将action中的值,
        // 赋值给showMsg、userName和tel。在Login文件夹下的index.js中就能拿到
        // showMsg、userName和tel的值。
        case types.LOGIN_SUCCESS:
            return Object.assign({}, state, {
                showMsg:action.showMsg,
                userName:action.userName,
                tel:action.tel
            });
        case types.LOGIN_FAIL:
            return Object.assign({}, state, {
                showMsg:action.showMsg,
                userName:action.userName,
                tel:action.tel
            });
        case types.SKIP_LOGIN:
            return Object.assign({}, state, {
                showMsg:action.showMsg,
                userName:action.userName,
                tel:action.tel
            });
        default:
            return state;
    }
}

6.继续,我们需要在contants文件夹下创建ActionTypes.js,为我们的登录事件添加几种状态:

// 登录成功
export const LOGIN_SUCCESS = 'LOGIN_SUCCESS';
// 登录失败
export const LOGIN_FAIL = 'LOGIN_FAIL';
//跳过登录
export const SKIP_LOGIN = 'SKIP_LOGIN';

7.然后跟着需要创建action事件,简单说就是用户或程序触发了A事件,而这个A事件就需要去处理对应的操作。像用户触发登录操作,那么这个就是LoginAction,具体操作就是拿到用户输入的账号密码后发起请求去验证密码正确与否。文章中用的就是登录案例,所以就在actions文件夹下创建LoginAction.js,并添加以下代码:

// 获取actionType中的全部状态,需要哪个就用哪个
import * as types from './../contants/ActionTypes';

export function LoginAction(uid,pwd,flag) {
    return dispatch => {
        if(uid == 'lyichao' && pwd == '123456' && flag == 0){
            dispatch(loginSuccess({
                showMsg:'LoginSuccess',
                userName:'lyichao',
                tel:'13888888888'
            }));
        }else if(uid == '' && pwd == '' && flag == 1){
            dispatch(skipLogin({
                showMsg:'SkipLogin',
                userName:'',
                tel:''
            }));
        }else {
            dispatch(loginFail({
                showMsg:'LoginFail',
                userName:'',
                tel:''
            }));
        }
    }
};

function loginSuccess(data) {
    return {
        // type是必要参数,通过type值判断
        type: types.LOGIN_SUCCESS,
        ...data
    };
}
function loginFail(data) {
    return {
        // type是必要参数,通过type值判断
        type: types.LOGIN_FAIL,
        ...data
    };
}
function skipLogin(data) {
    return {
        type: types.SKIP_LOGIN,
        ...data
    };
}

8.最后,我们还需要为页面和各个模块间进行关联。在Login文件夹下的index.js文件新增以下内容:

import React, { Component } from 'react';
import {
    AppRegistry,
    StyleSheet,
    Text,
    View,
    TextInput,
    Dimensions,
    TouchableOpacity
} from 'react-native';
import Toast from 'react-native-root-toast';
import { connect } from 'react-redux';
import { LoginAction } from '../../actions/LoginAction';
import {NavigationActions} from 'react-navigation';
const screenWidth = Dimensions.get('window').width;  //屏幕的宽度
const screenHeight = Dimensions.get('window').height;  //屏幕的高度
class Login extends Component {

    static navigationOptions = {
        header: null
    }

    constructor(props) {
        super(props);
        this.state = {
            uid:'',
            pwd:'',
            skip:0, //0正常登录 1跳过登录
        };
    }

    componentWillReceiveProps(nextProps) {
        console.log('componentWillReceiveProps');
        console.log(nextProps);
        // 每次值更新的时候,都会走这个方法,所以可以在这个方法里面添加判断,用来更新页面
        const { userName,tel,showMsg } = nextProps.LoginReducer;
        console.log("showMsg=",showMsg)
        if(showMsg === 'LoginSuccess' ){
            console.log(1);
            Toast.show('登录成功', {duration: Toast.durations.SHORT});
            this.props.navigation.navigate('Home',{userName:userName,tel:tel})
        }else if(showMsg === 'LoginFail' ){
            console.log(2);
            Toast.show('登录失败', {duration: Toast.durations.SHORT});
        }else{
            console.log(3);
            Toast.show('欢迎游客!', {duration: Toast.durations.SHORT});
            this.props.navigation.navigate('Home',{userName:userName,tel:tel})
        }
    }

    render() {
        return (
            <View style={styles.container}>
                <TextInput
                    style = {styles.textInput}
                    blurOnSubmit={true}
                    returnKeyType="done"
                    placeholder = '请输入账号'
                    selectionColor = "#bac3d4"
                    placeholderTextColor = '#bac3d4'
                    underlineColorAndroid = "transparent"
                    onChangeText={(text)=>{this.setState({uid:text})}}
                    value={this.state.uid}/>

                <TextInput
                    style = {styles.textInput}
                    blurOnSubmit={true}
                    returnKeyType="done"
                    placeholder = '请输入密码'
                    selectionColor = "#bac3d4"
                    placeholderTextColor = '#bac3d4'
                    underlineColorAndroid = "transparent"
                    onChangeText={(text)=>{this.setState({pwd:text})}}
                    value={this.state.pwd}/>

                <View style={styles.btn}>
                    <TouchableOpacity onPress={this.login}
                                      style={styles.radiusBtn}>
                        <Text style={styles.text}>登 录</Text>
                    </TouchableOpacity>
                    <TouchableOpacity onPress={this.skip}
                                      style={styles.radiusBtn}>
                        <Text style={styles.text}>跳 过</Text>
                    </TouchableOpacity>
                </View>

            </View>
        );
    }

    login = () => {
        console.log("login")
        let {uid,pwd} = this.state;
        this.props.LoginAction(uid,pwd,0);
    }

    skip = () => {
        console.log("skip")
        this.props.LoginAction('','',1);
    }


}

const styles = StyleSheet.create({
    container: {
        flex: 1,
        justifyContent: 'center',
        alignItems: 'center',
        backgroundColor: '#F5FCFF',
    },
    textInput:{
        width:0.8*screenWidth,
        height:0.08*screenHeight,
        color:'#1b6cec',
        fontSize:16,
        borderWidth: 1,
        borderRadius:8,
        borderColor:'#1b6cec',
        marginBottom: 15
    },
    btn:{
        height:0.25*screenHeight,
        justifyContent:'center',
        alignItems:'center'
    },
    radiusBtn:{
        backgroundColor:'#1b6cec',
        width:0.8*screenWidth,
        height:0.08*screenHeight,
        borderRadius:25,
        justifyContent:'center',
        alignItems:'center',
        marginBottom:8
    },
    text:{
        color:'#fff',
        fontSize:16
    }
});

export default connect((state) => {
    const { LoginReducer } = state;
    return {
        LoginReducer,
    };
},{ LoginAction })(Login)

Home文件夹下的index.js也要相应修改如下:


import React, { Component } from 'react';
import {
    AppRegistry,
    StyleSheet,
    Text,
    View
} from 'react-native';

export default class Login extends Component {

    static navigationOptions = {
        header: null
    }

    constructor(props) {
        super(props);
        this.state = {
            userName:this.props.navigation.state.params.userName,
            tel:this.props.navigation.state.params.tel,
        };

    }

    render() {
        let {userName,tel} = this.state;
        return (
            <View style={styles.container}>
                <Text style={styles.welcome}>
                    Welcome to Home!
                </Text>
                <Text style={styles.instructions}>
                    {userName}
                </Text>
                <Text style={styles.instructions}>
                    {tel}
                </Text>
            </View>
        );
    }
}

const styles = StyleSheet.create({
    container: {
        flex: 1,
        justifyContent: 'center',
        alignItems: 'center',
        backgroundColor: '#F5FCFF',
    },
    welcome: {
        fontSize: 20,
        textAlign: 'center',
        margin: 10,
    },
    instructions: {
        textAlign: 'center',
        color: '#333333',
        marginBottom: 5,
    },
});

大功告成,现在已经集成好Redux了!来看看效果~

Demo效果演示

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,658评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,482评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,213评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,395评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,487评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,523评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,525评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,300评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,753评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,048评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,223评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,905评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,541评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,168评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,417评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,094评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,088评论 2 352