热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

[Web前端]mobx教程(三)在React中使用Mobx

copyfrom:  https:blog.csdn.netsmk108articledetails85053903Mobx提供了一个mobx-react包帮助开发者方便地在Rea

react

copy from :  https://blog.csdn.net/smk108/article/details/85053903

Mobx提供了一个mobx-react包帮助开发者方便地在React中使用Mobx,mobx-react中有observer、Provider、inject几个常用的api。在《mobx系列(二)-mobx主要概念》中我们已经介绍过observer,本文介绍下inject、Provider,以及Mobx如何与React结合使用。

1、Provider

Provider是一个React组件,使用React的上下文(context)机制,可以用来向下传递stores,即把state传递给其子组件。

例如,有如下形式的一个store:

import {observable, computed, action} from 'mobx';

class userStoreClass {

@observable user = {

name: 'admin',

role: '管理员'

};

count = 0;

@computed get userName(){

return this.user.name;

}

@action changeUser(){

if(this.count % 2 === 1){

this.user = {

name: 'admin',

role: '管理员'

};

}else{

this.user.name = 'guest';

this.user.role = '访客';

this.user.isGuest = 'true';

}

this.count ++;

}

}

const userStore = new userStoreClass();

export default userStore;

使用Provider传递store的方式为:

import React from 'react';

import ReactDOM from 'react-dom';

import {configure} from 'mobx';

import {Provider} from 'mobx-react';

import userStore from './models/userStore';

import App from './components/App';

// 状态始终需要通过动作来更新(实际上还包括创建)

configure({'enforceActions': 'always'});

ReactDOM.render((

), document.getElementById('container'));

如果有多个store,可以使用类似于如下形式:

const stores = {

mainStore, userStore, commonStore

};

ReactDOM.render((

), document.getElementById('container'));

2、@inject

inject是一个高阶组件,作用是将组件连接到提供的stores。具体说是与Provider结合使用,从Provider提供给应用的state数据中选取所需数据,以props的形式传递给目标组件。用法为:

inject(stores)(component)

@inject(stores) class Component...

对应上节的例子,App内使用的组件User使用@inject方式为:

import React, {Component} from 'react';

import {inject, observer} from 'mobx-react';

import {Button} from 'antd';

import './style.css';

@inject( 'userStore')

@observer

export default class User extends Component{

constructor(props){

super(props);

this.state = {};

}

render(){

// 可以以this.props.userStore形式获取store内state

const {user} = this.props.userStore;

// 以.形式使用对象的属性值

return(

name:{user.name}

role:{user.name}

{user.isGuest ? `isGuest:${user.isGuest}` : ''}

);

}

}

3、react-mobx实践示例

我写了一个react-mobx的简单demo,地址为:https://github.com/smk108/mobx_demo ,

demo的结构为:

依据组件的划分使用了3个store,需要说明的是我是以class的形式创建的store,store中export的是store class的instance,例如第一节中userStore的形式:

const userStore = new userStoreClass();

export default userStore;

在React中使用Mobx的方式有很多,Mobx不会强制要求以某种方式使用它,我在demo中使用的方式仅仅是其中一种。事实上,你可以以任何你喜欢并且能生效的方式使用它。在下一篇《Mobx定义数据存储》中会介绍、对比文档推荐使用的数据存储和我的demo中使用的数据结构间的不同。

4、可观察的局部组件状态

Mobx允许在响应式React组件内使用自由地使用状态,意味着我们可以将一些状态像普通React组件一样管理,例如对上面提到的demo中User组件做如下修改:

import React, {Component} from 'react';

import {inject, observer} from 'mobx-react';

import {Button} from 'antd';

import Timer from '../Timer';

import './style.css';

@inject( 'userStore')

@observer

export default class User extends Component{

constructor(props){

super(props);

this.state = {

userChangeTimes: 0

};

}

handleChangeUser(){

this.props.userStore.changeUser();

let {userChangeTimes} = this.state;

userChangeTimes ++ ;

this.setState({userChangeTimes});

}

render(){

const {user} = this.props.userStore;

return(

name:{user.name}

role:{user.name}

{user.isGuest ? `isGuest:${user.isGuest}` : ''}

user change times: {this.state.userChangeTimes}

);

}

}

User组件state的userChangeTimes可以使用setState进行修改,一切都和不使用Mobx时一样。但是,推荐响应式组件应该没有或很少有状态,因为在与其他组件共享的对象中封装(视图)状态通常更方便。针对响应式组件需要维护单独状态的情况,Mobx为我们提供了更加方便的一种方式-可观察的局部组件状态。

Mobx允许使用@observable在React组件内引入可观察属性,意味着我们不需要通过React 的冗长和强制性的 setState 机制也可以在组件中拥有功能同样强大的本地状态(local state)。

用法如下(使用demo中Timer组件举例):

import React, {Component} from 'react';

import {inject, observer} from 'mobx-react';

import {observable, action} from "mobx";

import './style.css';

@inject('commonStore')

@observer

export default class Timer extends Component{

constructor(props){

super(props);

this.state = {};

}

@observable secOndsPassed= 0;

componentWillMount(){

this.props.commonStore.startTime();

this.timer = setInterval(this.handleChangeSecondsPassed,1000);

}

@action.bound handleChangeSecondsPassed(){

this.secondsPassed ++;

}

render(){

const {time} = this.props.commonStore;

return(

{time}

Seconds passed:{this.secondsPassed}

);

}

}

secondsPassed作为组件内可观察的局部状态,不使用setState也触发UI的响应。

需要注意的是:

可观察局部状态会被render提取调用;
可观察局部状态的修改会触发React的componentWillUpdate和componentDidUpdate生命周期,不会触发其它的生命周期;
如果你需要使用React的其它生命周期方法,请使用基于state的常规React API;
5、生命周期钩子

当使用mobx-react时可以定义一个新的生命周期钩子函数componentWillReact,当组件因为它观察的状态发生改变时,组件会重新渲染,这时componentWillReact会触发,可以帮助追溯渲染并找到导致渲染的动作(action)。

修改demo中User组件举例如下:

import React, {Component} from 'react';

import {inject, observer} from 'mobx-react';

import {Button} from 'antd';

import Timer from '../Timer';

import './style.css';

@inject( 'userStore')

@observer

export default class User extends Component{

constructor(props){

super(props);

this.state = {

userChangeTimes: 0

};

}

handleChangeUser(){

this.props.userStore.changeUser();

let {userChangeTimes} = this.state;

userChangeTimes ++ ;

this.setState({userChangeTimes});

}

componentWillReact() {

console.log("I will re-render, since the user has changed!");

}

render(){

const {user} = this.props.userStore;

return(

name:{user.name}

role:{user.name}

{user.isGuest ? `isGuest:${user.isGuest}` : ''}

user change times: {this.state.userChangeTimes}

);

}

}

需要注意的是:

componentWillReact 不接收参数;
componentWillReact 初始化渲染前不会触发 (使用 componentWillMount 替代);
componentWillReact 对于 mobx-react@4+, 当接收新的 props 时并在 setState 调用后会触发此钩子;
像User组件内通过setState修改userChangeTimes也会触发此钩子;
6、React优化

本小节介绍几项基本的React优化策略,有些是基于在React中使用Mobx时特有的策略,有些是会用React通用的策略。

使用大量的小组件
@observer 组件会追踪它们使用的所有值,并且当它们中的任何一个改变时重新渲染。 所以你的组件越小,它们需要重新渲染产生的变化则越小;这意味着用户界面的更多部分具备彼此独立渲染的可能性。

在专用组件中渲染列表(避免多个组件受影响,一起重新渲染)
不要使用数组的索引作为 key(虚拟dom)
不用使用数组索引或者任何将来可能会改变的值作为 key

晚一点使用间接引用值
使用 mobx-react 时,推荐尽可能晚的使用间接引用值。 这是因为当使用 observable 间接引用值时 MobX 会自动重新渲染组件。 如果间接引用值发生在组件树的层级越深,那么需要重新渲染的组件就越少。


推荐阅读
  • Go Cobra命令行工具入门教程
    本文介绍了Go语言实现的命令行工具Cobra的基本概念、安装方法和入门实践。Cobra被广泛应用于各种项目中,如Kubernetes、Hugo和Github CLI等。通过使用Cobra,我们可以快速创建命令行工具,适用于写测试脚本和各种服务的Admin CLI。文章还通过一个简单的demo演示了Cobra的使用方法。 ... [详细]
  • 解决nginx启动报错epoll_wait() reported that client prematurely closed connection的方法
    本文介绍了解决nginx启动报错epoll_wait() reported that client prematurely closed connection的方法,包括检查location配置是否正确、pass_proxy是否需要加“/”等。同时,还介绍了修改nginx的error.log日志级别为debug,以便查看详细日志信息。 ... [详细]
  • uniapp开发H5解决跨域问题的两种代理方法
    本文介绍了uniapp开发H5解决跨域问题的两种代理方法,分别是在manifest.json文件和vue.config.js文件中设置代理。通过设置代理根域名和配置路径别名,可以实现H5页面的跨域访问。同时还介绍了如何开启内网穿透,让外网的人可以访问到本地调试的H5页面。 ... [详细]
  • mobx inject进入后只有observable ... [详细]
  • 本文介绍了RPC框架Thrift的安装环境变量配置与第一个实例,讲解了RPC的概念以及如何解决跨语言、c++客户端、web服务端、远程调用等需求。Thrift开发方便上手快,性能和稳定性也不错,适合初学者学习和使用。 ... [详细]
  • 本文介绍了如何使用C#制作Java+Mysql+Tomcat环境安装程序,实现一键式安装。通过将JDK、Mysql、Tomcat三者制作成一个安装包,解决了客户在安装软件时的复杂配置和繁琐问题,便于管理软件版本和系统集成。具体步骤包括配置JDK环境变量和安装Mysql服务,其中使用了MySQL Server 5.5社区版和my.ini文件。安装方法为通过命令行将目录转到mysql的bin目录下,执行mysqld --install MySQL5命令。 ... [详细]
  • 本文介绍了Python爬虫技术基础篇面向对象高级编程(中)中的多重继承概念。通过继承,子类可以扩展父类的功能。文章以动物类层次的设计为例,讨论了按照不同分类方式设计类层次的复杂性和多重继承的优势。最后给出了哺乳动物和鸟类的设计示例,以及能跑、能飞、宠物类和非宠物类的增加对类数量的影响。 ... [详细]
  • IT方面的论坛太多了,有综合,有专业,有行业,在各个论坛里混了几年,体会颇深,以前是论坛哪里人多 ... [详细]
  • GPT-3发布,动动手指就能自动生成代码的神器来了!
    近日,OpenAI发布了最新的NLP模型GPT-3,该模型在GitHub趋势榜上名列前茅。GPT-3使用的数据集容量达到45TB,参数个数高达1750亿,训练好的模型需要700G的硬盘空间来存储。一位开发者根据GPT-3模型上线了一个名为debuid的网站,用户只需用英语描述需求,前端代码就能自动生成。这个神奇的功能让许多程序员感到惊讶。去年,OpenAI在与世界冠军OG战队的表演赛中展示了他们的强化学习模型,在限定条件下以2:0完胜人类冠军。 ... [详细]
  • PHP反射API的功能和用途详解
    本文详细介绍了PHP反射API的功能和用途,包括动态获取信息和调用对象方法的功能,以及自动加载插件、生成文档、扩充PHP语言等用途。通过反射API,可以获取类的元数据,创建类的实例,调用方法,传递参数,动态调用类的静态方法等。PHP反射API是一种内建的OOP技术扩展,通过使用Reflection、ReflectionClass和ReflectionMethod等类,可以帮助我们分析其他类、接口、方法、属性和扩展。 ... [详细]
  • 本文介绍了在Android Studio中使用命令行build gradle的方法,并解决了一些常见问题,包括手动配置gradle环境变量和解决External Native Build Issues的方法。同时提供了相关参考文章链接。 ... [详细]
  • 一、路由首先需要配置路由,就是点击good组件进入goodDetail组件配置路由如下{path:goodDetail,component:goodDetail}同时在good组件中写入如下点击事件,路由中加入 ... [详细]
  • 其实之前也有下载过完整的android源码,但是从来没有对这个做过一些总结,在加上最近需要经常去看,索性就在从新下载,编译一下,其实这些东西官网上面都有。http:sou ... [详细]
  • 详解react组件通讯方式(多种)
    这篇文章主要介绍了详解react组件通讯方式,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着 ... [详细]
  • 1223  drf引入以及restful规范
    [toc]前后台的数据交互前台安装axios插件,进行与后台的数据交互安装axios,并在main.js中设置params传递拼接参数data携带数据包参数headers中发送头部 ... [详细]
author-avatar
祥仔先森_530
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有