vuex (一个组件修改了state的数据、其他组件获取到的值是同步更新修改后的值、有的时候数据更新了页面没有更新可以通过this.$forceUpdate强制刷新)

# 优点:
	js 原生的数据对象写法, 比起 localStorage 不需要做转换, 使用方便
	属于 vue 生态一环, 能够触发响应式的渲染页面更新 (localStorage 就不会)
	限定了一种可预测的方式改变数据, 避免大项目中, 数据不小心的污染

# 缺点:
	刷新浏览器,vuex中的state会重新变为初始状态
 	解决方案-插件 vuex-persistedstate
安装:npm i vuex -S
src下新建store文件夹index.js文件、或者直接vue ui方式创建项目时添加vuex
1.store文件夹下index.js
import Vue from 'vue'
import Vuex from 'vuex'

Vue.use(Vuex);
//然后在main.js引入并挂载

export default new Vuex.Store({
    /1. state存放公共数据
    state: { name: '张三',age: 18 },
    /2. getters相当于 vue中的computed属性,方法第一个参数是state、返回将state中数据计算的值
    getters: { val(state) { return state.name + '-' + state.age } },
    /3. mutations相当于 vue中的methods、只不过是写一些同步的逻辑代码修改state中的数据、其中写的方法第一个参数为state,第二个参数是携带的参数、在页面中通过this.$store.commit('addnum', { number: 5 })调用 addnum()方法
    mutations: { addnum(state, num) { state.age + num.number } },
    /4. actions类似于mutations、里面写异步逻辑代码,actions一般操作mutations、而不直接操作satate中的数据
	//getUserInfo()为异步的请求方法,返回一个对象,然后在组件methods中通过 this.$store.dispatch('getUserInfo')调用这个方法
    actions: { 
        async getUserInfo(context) { 
            //context相当于整个$Store对象、类似this.$store、包含state、getters、mutations、actions
           var res = await axios.get('url')
           //下面相当于this.$store.commit、第一个参数是mutations中的方法名、第二个参数是要传入的数据
           context.commit('setUserInfo', res)
            //下面调用 actions自己的 getToken()方法
           context.dispatch('getToken')
        } 
    }
});
2.main.js全局引入vuex并挂载
//引入vuex
import store from '@/store/index'

new Vue({
  //全局挂载vuex
  store,
  render: h => h(App),
}).$mount('#app')
3.所有组件都能直接使用vuex中的值、例如:App.vue
<template>
   <div>
    {{name}}
    {{age}}
       <div @click='addAge'></div>
   </div>
</template>
<script>
    //通过 mapState辅助函数简化 computed中的代码
    import { mapState } from 'vuex'
    //引入 mapGetesers辅助函数
    import { mapGetters } from 'vuex'
    //引入 mapMutations辅助函数 简化methods中的代码
    import { mapMutations } from 'vuex'
    //引入 mapAcsions辅助函数 简化methods中的代码
    import { mapActions } from 'vuex'
    export default {
        computed: {
            //computed 监听获取到 state中公共的值
            ...mapState(['name', 'age']),
            //computed 监听获取到 getters中计算的得到的值
            ...mapGetters('[val]')
        },
        methods() {
        	//通过 mapMutations辅助函数简化 页面调用的addAge()方法
        	...mapMutations(['addAge']),
            //通过 mapActions辅助函数简化,调用 getUserInfo()请求方法
            ...mapActions(['getUserInfo'])
    	},
    }
</script>

//1. computed: mapState(['name', 'age']) 相当于下面
computed: {
	name() { return this.$store.state.name},
	age() { return this.$store.state.age }
}


//2. 如果computed中有另外自定义计算属性、用es6的扩展运算符、
例如:
computed: {
	val() {},
	...mapState(['name', 'age'])
}

//3. mapGetters('[val]')相当于下面
val() {
	this.$store.getters.val
}


//4. mapMutations(['addAge']) 相当于下面
addAge() {
     //这里调用 mutations中定义的 addnum()方法,调用时最好写成对象形式可以传递更多信息
     this.$store.commit('addnum', { number: 5 })
}


//5. mapActions(['getUserInfo']) 相当于下面
getUserInfo() {
	this.$store.dispatch('getUserInfo')
}
在实际开发当中,state里面的属性值是空的,当登录以后,再进行获取对应的信息。
登录以后,需要得到用户信息,那如何得到呢?
首先进入页面的时候调用actions中的getUserInfo方法
例如:
created(){ this.getUserInfo() }
methods:{ ...mapActions(['getUserInfo']) }

总结:

​ ① 依赖state得到新的数据,用getters(跟computed一样,只读)

​ ② 修改state的属性值,就用mutations(同步操作)

mixins (区别于vuex、mixins中一个组件修改了数据,其他组件获取到这个数据还是最开始没修改之前的值)

场景:解决项目中多出代码复用的情况
#局部引入mixins
1.src下新建文件夹mixins下创建index.js
里面写组件script中定义的data、computed、watch、methods、created等复用的代码
export const showMixin = {
    data() { return { num: 2 } },
    methods: { console.log('123') }
}

2.在其他组件引入
import { showMixin } from '@/mixins/index'
export default {
    mixins: [showMixin]
}

注意:
	1.如果组件 data中定义了和 mixins相同的属性、会以组件数据优先
	2.如果有同名的方法名、两个都会被调用、混入的方法先调用、组件自身的方法后调用

#全局引入
main.js中
Vue.mixin({
    created() { 
        const options = this.$option.myoption
        if () { console.log(myoption) }
    }
})

new Vue({
    option: { myoption: 'test' },
    render: h => h(App),
}).$mount('#app')