mobx使用教程

一、 简介

mobx:

Simple,scalable state management简单可扩展的状态管理

核心思想: 状态变化引起的副作用应该被自动触发

二、前奏知识

1、es5实现类

function Animal(){

}
function Dog(){

}
Object.defineProperties(Animal.prototype,{
    name:{
        value(){
            return 'Animal';
        }
    },
    say:{
        value(){
            return `I'm ${this.name()}`
        }
    }
})
//业务需求,实现Dog继承Animal dog instanceof Animal => true
// dog.__proto__.__proto__=== Animal.prototype
Dog.prototype = Object.create(Animal.prototype,{
    constructor:{
        value:Dog,
        enumberable:false
    },
    name:{
        value(){
            return 'Dog'
        }
    }
})
console.log(new Dog().say())

2、es6实现类

class Animal{
    nam(){
        return 'Animal'
    }
}
class Dog extends Animal{
    food="pisa"
    name(){
        return 'Dog'
    }
    
}
let dog = new Dog()
console.log(dog.food)

3、异常处理

报错:语法错误 food="pisa"是因为确实babel插件导致的,
npm install babel-plugin-transform-class-properties -D
加入webpack.config.js配置

options:{
      presets:['@babel/preset-env'],
      plugins:['transform-class-properties']
}

报错:Cannot read property 'bindings' of null
解决办法:
npm install @babel/preset-env -S,然后将webpack.config.js中的presets:["env"]替换成presets:["@babel/preset-env"]

4、装饰器decorator:在声明阶段实现类与类成员注解的一种语法
babel支持:

npm install -D @babel/plugin-proposal-decorators

配置:

                options:{
                    presets:['@babel/preset-env'],
                    plugins:[['@babel/plugin-proposal-decorators',{"legacy":true}]]
                }

(1)类修饰器

function log(target){
    const desc = Object.getOwnPropertyDescriptors(target.prototype)
    console.log(desc,'---')
    for(const key of Object.keys(desc)){
        if(key === 'constructor'){
            continue;
        }
        const func = desc[key].value;
        if('function' === typeof func){
            Object.defineProperty(target.prototype,key,{
                value(...args){
                    console.log('before ' + key);
                    const ret = func.apply(this,args)
                    console.log('after ' + key)
                    return ret
                }
            })
        }
    }
}

@log
class Numberic{
    PI = 3.1415926;
    add(...nums){
        return nums.reduce((p,n)=>(p+n),0)
    }
}

new Numberic().add(1,2)

成员修饰器
(2)、值属性修饰器

function readonly(target,key,descriptor){
    descriptor.writable = true;
  // 返回这个新的描述符
  return descriptor
}
class Numberic{
    @readonly PI = 3.1415926;
    add(...nums){
        return nums.reduce((p,n)=>(p+n),0)
    }
}
let numb = new Numberic()
numb.PI = 100
console.log(numb.PI)

(3)、函数属性修饰器

function validate(target,key,descriptor){
    const func = descriptor.value;
    descriptor.value = function(...args){
        for(let num of args){
            if('number' !== typeof num){
                throw new Error(`${num} is not a number`)
            }
        }
        return func.apply(this,args)
    }
}

class Numberic{
    PI = 3.1415926;
    @validate
    add(...nums){
        return nums.reduce((p,n)=>(p+n),0)
    }
}
let numb = new Numberic()
console.log(numb.add(1,'2'))

三、mobx常用api

安装

npm install mobx -S

1、observable

是一种让数据的变化可以被观察的方法

(1)、数组类型

import { observable,isObservableArray } from 'mobx'

const arr = observable([1,2,3])
console.log(arr[0],Array.isArray(arr),isObservableArray(arr))

(2)、对象类型

import { observable,isObservableObject,extendObservable } from 'mobx'

const obj = observable({username:'zs'})
extendObservable(obj,{age:18})//给obj添加可观察属性age
console.log(obj['username'],isObservableObject(obj),obj['age'])

(3)、基本类型

import { observable } from 'mobx'

let number = observable.box(5)
let str = observable.box('hello')
let flag = observable.box(true)

number.set(10)
console.log(number.get())

(4)、声明仓库

import { observable } from 'mobx'

class Store{
    @observable username='zs'
    @observable information={message:'hello'}
}
console.log(new Store())

2、对可观察数据做出反应

(1)、computed
第一种用法

在类外返回一个根据其他属性计算后所得的值,该值会因为依赖属性的变化而变化

import { observable,computed } from 'mobx'

class Store{
    @observable username='zs'
    @observable age = 18
    @observable res={message:'hello'}
}
var store = new Store()
var information = computed(()=>{
    console.log('-----')
    return {name:store.username,age:store.age}
})
console.log(information.get())
store.username = 'lisi'
console.log(information.get())

或者

import { observable,computed,makeObservable } from 'mobx'

class Store{
    constructor(){
        makeObservable(this)
    }
    @observable username='zs'
    @observable age = 18
    @observable res={message:'hello'}
}
var store = new Store()
var information = computed(()=>{
    return {name:store.username,age:store.age}
})
information.observe_((change)=>{
    console.log(change,'++++')
})

store.username = 'lisi'
store.age = 19

第二种用法

const number = observable(10);
const plus = computed(() => number.get() > 0);

autorun(() => {
  console.log(plus.get());
});

number.set(-19);
number.set(-1);
number.set(1);

依次输出了true,false,true。
第一个true是number初始化值的时候,10>0为true没有问题。
第二个false将number改变为-19,输出false,也没有问题。
但是当-19改变为-1的时候,虽然number变了,但是number的改变实际上并没有改变plus的值,所以没有其它地方收到通知,因此也就并没有输出任何值。
直到number重新变为1时才输出true。

(2)、autorun

import { observable, autorun } from 'mobx';

const value = observable(0);
const number = observable(100);

autorun(() => {
  console.log(value.get());
});
value.set(1);
value.set(2);
number.set(101);
import {observable, autorun} from 'mobx';

var todoStore = observable({
    /* 一些观察的状态 */
    todos: [],

    /* 推导值 */
    get completedCount() {
        return this.todos.filter(todo => todo.completed).length;
    }
});

/* 观察状态改变的函数 */
autorun(function() {
    console.log("Completed %d of %d items",
        todoStore.completedCount,
        todoStore.todos.length
    );
});

/* ..以及一些改变状态的动作 */
todoStore.todos[0] = {
    title: "Take a walk",
    completed: false
};
// -> 同步打印 'Completed 0 of 1 items'

todoStore.todos[0].completed = true;
// -> 同步打印 'Completed 1 of 1 items'

(3)、when

import { observable,when } from 'mobx'
const flag = observable(false);


when(()=>flag.get(),()=>console.log("it's true"))
flag.set(true);

(4)、Reaction

三、mobx-react

安装依赖

npm install react react-dom prop-types mobx-react -S
npm install @babel/preset-react -D

报错: Plugin/Preset files are not allowed to export objects是因为babel的版本对应不上,解决方式 webpack报错/babel报错的解决方法

mobx6+版本如果需要数据响应式得做些修改和调整

import React,{Component} from 'react'
import ReactDOM from 'react-dom'
import { action,observable,makeObservable } from 'mobx'
import { observer } from 'mobx-react'


class Store {
    constructor(){
        makeObservable(this)
    }
    @observable 
    cache = {queue:[1,2]}
    @action.bound refresh(){
        console.log(this.cache.queue)
        this.cache.queue.push(1)
    }
}
const store = new Store();

@observer
class Bar extends Component {
    // static propTypes = {
    //     queue:observableTypes.observableArray
    // };
    render(){
        const queue = this.props.queue;
        return <span>{queue.length}</span>
    }
}

class Foo extends Component {
    // static propTypes = {
    //     cache:observableTypes.observableObject
    // }
    render(){
        const cache = this.props.cache;
        return <div><button onClick={()=>{
            this.props.refresh()
        }}>refresh</button><Bar queue={cache.queue} /></div>
    }
}

ReactDOM.render(<Foo cache = {store.cache} refresh={store.refresh} />,document.querySelector('#root'))

四、todolist案例

import { observable,action,computed,makeObservable } from 'mobx'
import React,{Component,Fragment} from 'react'
import ReactDOM from 'react-dom'
import { observer,PropTypes as ObservablePropTypes } from 'mobx-react'
import PropTypes from 'prop-types'

class Todo{
    id = Math.random();
    @observable title = '';
    @observable finished = false;
    @action.bound toggle(flag){
        this.finished = flag;
    }
    constructor(title){
        makeObservable(this)
        this.title = title;
    }
}
class Store{
    constructor(){
        makeObservable(this)
    }
    @observable todos = [];
    @action.bound createTodo(title){
        this.todos.unshift(new Todo(title))
        console.log(this.todos)
    }
    @action.bound removeTodo(todo){
        this.todos.remove(todo);
    }
    @computed get left(){
        return this.todos.filter(todo => !todo.finished).length;
    }
}
var store = new Store()

@observer
class TodoItem extends Component{
    static propTypes = {
        todo:PropTypes.shape({
            id:PropTypes.number.isRequired,
            title:PropTypes.string.isRequired,
            finished:PropTypes.bool.isRequired
        }).isRequired
    }
    render(){
        const todo = this.props.todo;
        return <>
        <input 
        type="checkbox" 
        className="toggle" 
        checked={todo.finished}
        onChange={(e)=>{
            console.log(e.target.checked)
            todo.toggle(e.target.checked)
        }}
        ></input>
        <span className={["title",todo.finished && 'finished'].join(' ')}>{todo.title}</span></>
    }
}

@observer
class TodoList extends Component{
    static propTypes = {
        store:PropTypes.shape({
            createTodo:PropTypes.func,
            todos:ObservablePropTypes.observableArrayOf(ObservablePropTypes.observableObject).isRequired
        }).isRequired
    }
    state = { inputValue:'' }
    handleSubmit = (e)=>{
        e.preventDefault();
        var store = this.props.store;
        var inputValue = this.state.inputValue;
        store.createTodo(inputValue);
        this.setState({
            inputValue:''
        })
    }
    handleChange = (e)=>{
        var inputValue = e.target.value;
        this.setState({
            inputValue
        })
    }
    render(){
        const store = this.props.store;
        const todos = store.todos;
        return <div className="todo-list">
            <header>
                <form onSubmit={this.handleSubmit}>
                    <input 
                    type="text" 
                    onChange = {this.handleChange} 
                    value={this.state.inputValue}
                    className="input"
                    placeholder="What needs to finished?" />
                </form>
            </header>
            <ul>
                {
                    todos.map(todo=>{
                        return <li key={todo.id} className = "todo-item">
                            <TodoItem todo={todo} />
                            <span 
                            className="delete"
                            onClick={()=>{
                                store.removeTodo(todo)
                            }}>delete</span>
                        </li>
                    })
                }
            </ul>
            <footer>{store.left} item(s) unfinished</footer>
        </div>
    }
}
ReactDOM.render(<TodoList store={store}/>,document.querySelector('#root'))

package.json内容

{
  "name": "mobxdemo",
  "version": "1.0.0",
  "description": "",
  "main": "webpack.config.js",
  "scripts": {
    "start": "webpack -w",
    "test": "echo \"Error: no test specified\" && exit 1"
  },
  "author": "",
  "license": "ISC",
  "devDependencies": {
    "@babel/core": "^7.15.0",
    "@babel/plugin-proposal-decorators": "^7.14.5",
    "babel-loader": "^8.2.2",
    "babel-plugin-transform-decorators-legacy": "^1.3.5",
    "babel-preset-env": "^1.7.0",
    "@babel/preset-react": "^7.0.0",
    "webpack": "^5.51.1",
    "webpack-cli": "^4.8.0"
  },
  "dependencies": {
    "@babel/preset-env": "^7.15.0",
    "mobx": "^6.3.2",
    "mobx-react": "^7.2.0",
    "prop-types": "^15.7.2",
    "react": "^17.0.2",
    "react-dom": "^17.0.2"
  }
}

webpack.config.js

const path = require('path')
const config = {
    mode:'development',
    entry:path.resolve(__dirname,'src/index.jsx'),
    output:{
        path:path.resolve(__dirname,'dist'),
        filename:'main.js'
    },
    module:{
        rules:[{
            test:/\.jsx$/,
            exclude:/ndoe_modules/,
            use:{
                loader:'babel-loader',
                options:{
                    presets:['@babel/preset-env','@babel/react'],
                    plugins:[['@babel/plugin-proposal-decorators',{"legacy":true}]]
                }
            }
        }]
    },
    devtool:'inline-source-map'
}
module.exports = config

五、工具函数

1、observe监查

class Todo{
    id = Math.random();
    @observable title = '';
    @observable finished = false;
    @action.bound toggle(flag){
        this.finished = flag;
    }
    constructor(title){
        makeObservable(this)
        this.title = title;
    }
}
class Store{
    constructor(){
        makeObservable(this)
        observe(this.todos,(change)=>{
            this.disposers.forEach(disposer => disposer())
            this.disposers = [];
            for(let todo of change.object){
                var disposer = observe(todo,changex=>{
                    console.log(changex);
                })
                this.disposers.push(disposer);
            }
        })
    }
    disposers = [];
    @observable todos = [];
    @action.bound createTodo(title){
        this.todos.unshift(new Todo(title))
        console.log(this.todos)
    }
}
var store = new Store()
store.createTodo('11111')
store.createTodo('2222')
store.todos[0].toggle(true)
//即可监测到todos的变化,也可监测到todos列表中某个元素的变化

2、spy监测
该函数可监测到数据所有层级的变化,该函数在全局调用

//可监测到所有action,update,render等导致数据变化的行为,性能损耗太大,不建议在生产环境使用
syp(event=>console.log(event))

3、"toJS"将observeable对象转换为js对象。
4、trace追踪,该函数需要在副作用中调用,比如render,传入true参数时自带断点调试功能。

六 优化法则

1、细粒度拆分试图组件
2、使用专用组件处理列表
3、尽可能晚地解构可观察数据
package.json

{
  "name": "mobxdemo",
  "version": "1.0.0",
  "description": "",
  "main": "webpack.config.js",
  "scripts": {
    "start": "webpack -w",
    "test": "echo \"Error: no test specified\" && exit 1"
  },
  "author": "",
  "license": "ISC",
  "devDependencies": {
    "@babel/core": "^7.15.0",
    "@babel/plugin-proposal-decorators": "^7.14.5",
    "babel-loader": "^8.2.2",
    "babel-plugin-transform-decorators-legacy": "^1.3.5",
    "babel-preset-env": "^1.7.0",
    "@babel/preset-react": "^7.0.0",
    "webpack": "^5.51.1",
    "webpack-cli": "^4.8.0"
  },
  "dependencies": {
    "@babel/preset-env": "^7.15.0",
    "mobx": "^6.3.2",
    "mobx-react": "^7.2.0",
    "prop-types": "^15.7.2",
    "react": "^17.0.2",
    "react-dom": "^17.0.2"
  }
}

webpack.config.json

const path = require('path')
const config = {
    mode:'development',
    entry:path.resolve(__dirname,'src/index.jsx'),
    output:{
        path:path.resolve(__dirname,'dist'),
        filename:'main.js'
    },
    module:{
        rules:[{
            test:/\.jsx$/,
            exclude:/ndoe_modules/,
            use:{
                loader:'babel-loader',
                options:{
                    presets:['@babel/preset-env','@babel/react'],
                    plugins:[['@babel/plugin-proposal-decorators',{"legacy":true}]]
                }
            }
        }]
    },
    devtool:'inline-source-map'
}
module.exports = config

index.jsx

import { spy,observe,observable,action,computed,makeObservable, trace } from 'mobx'
import React,{Component,Fragment} from 'react'
import ReactDOM from 'react-dom'
import { observer,PropTypes as ObservablePropTypes } from 'mobx-react'
import PropTypes from 'prop-types'



class Todo{
    id = Math.random();
    @observable title = '';
    @observable finished = false;
    @action.bound toggle(flag){
        this.finished = flag;
    }
    constructor(title){
        makeObservable(this)
        this.title = title;
    }
}
class Store{
    constructor(){
        makeObservable(this)
        observe(this.todos,(change)=>{
            this.disposers.forEach(disposer => disposer())
            this.disposers = [];
            for(let todo of change.object){
                var disposer = observe(todo,changex=>{
                    console.log(changex);
                })
                this.disposers.push(disposer);
            }
        })
    }
    disposers = [];
    @observable todos = [];
    @action.bound createTodo(title){
        this.todos.unshift(new Todo(title))
        console.log(this.todos)
    }
    @action.bound removeTodo(todo){
        this.todos.remove(todo);
    }
    @computed get left(){
        return this.todos.filter(todo => !todo.finished).length;
    }
}
var store = new Store()

@observer
class TodoItem extends Component{
    static propTypes = {
        todo:PropTypes.shape({
            id:PropTypes.number.isRequired,
            title:PropTypes.string.isRequired,
            finished:PropTypes.bool.isRequired
        }).isRequired
    }
    render(){
        trace()
        const todo = this.props.todo;
        return <>
        <input 
        type="checkbox" 
        className="toggle" 
        checked={todo.finished}
        onChange={(e)=>{
            console.log(e.target.checked)
            todo.toggle(e.target.checked)
        }}
        ></input>
        <span className={["title",todo.finished && 'finished'].join(' ')}>{todo.title}</span></>
    }
}

@observer
class TodoFooter extends Component{
    render(){
        trace()
        let store = this.props.store;
        return <footer>{store.left} item(s) unfinished</footer>
    }
}
@observer
class TodoView extends Component{
    render(){
        trace()
        let todos = this.props.todos
        return todos.map(todo=>{
            return <li key={todo.id} className = "todo-item">
                <TodoItem todo={todo} />
                <span 
                className="delete"
                onClick={()=>{
                    store.removeTodo(todo)
                }}>delete</span>
            </li>
        })
    }
}
@observer
class TodoHeader extends Component{
    state = { inputValue:'' }
    handleSubmit = (e)=>{
        e.preventDefault();
        var store = this.props.store;
        var inputValue = this.state.inputValue;
        store.createTodo(inputValue);
        this.setState({
            inputValue:''
        })
    }
    handleChange = (e)=>{
        var inputValue = e.target.value;
        this.setState({
            inputValue
        })
    }
    render(){
        trace()
        let store = this.props.store
        return <header>
                    <form onSubmit={this.handleSubmit}>
                        <input 
                        type="text" 
                        onChange = {this.handleChange} 
                        value={this.state.inputValue}
                        className="input"
                        placeholder="What needs to finished?" />
                    </form>
                </header>
    }
}

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

推荐阅读更多精彩内容