vue非父子组件通信问题及解决方法

问题描述:

最近在做登录部分时遇到一个场景,当点击 “用户”按钮时,首先渲染login组件,在用户登录后直接跳转到用户信息界面(user组件)。这里遇到了需要将login组件通过异步请求获得的用户信息传向 user组件,但是login和login组件不是父子组件,就暂称为兄弟组件吧。 简而言之,我要解决的就是兄弟组件之间的信息传递问题。

//位于login.vue 中
export default {
 methods:{
 login(){
  this.$axios({
  method: 'post',
  url: '/student/login/',
  data: {
  username: this.username,
  password: this.password
  }
  })
  .then(function (response) {
  this.$router.replace({path: 'user'});
  }.bind(this))
  .catch(function (error) {
  console.log(error);
  alert('用户名或密码错误');
  });
 }
 }
} 

在异步请求完成后,就跳转到user组件中,但是response的请求返回信息就无法传到非父子组件的user组件中。

.............................................................................................................................................................................................................

解决问题:

一、首先我想到了将login组件 和 user组件放到一起(两个组件合2为1),加避免了组件间的信息传递,但是这种解决方案太low,破坏了项目原有的结构。

二、查文档得知了

非父子关系的两个组件之间也需要通信。在简单的场景下,可以使用一个空的 Vue 实例作为事件总线:

var bus = new Vue()
// 触发组件 A 中的事件
bus.$emit('id-selected', 1)
// 在组件 B 创建的钩子中监听事件
bus.$on('id-selected', function (id) {
 // ...
}) 

以自己遇到的问题为例,写出具体的实现:(warning:  我自己遇到的情况不能用这种简单的方式解决)

最好先新建一个js文件,来创建出我们的eventBus,我们把它命名为bus.js

import Vue from 'vue';
export default new Vue(); 

然后在 login.vue 文件中:

//位于login.vue 中
import Bus from '../bus.js';
export default {
 methods:{
 login(){
  this.$axios({
  method: 'post',
  url: '/student/login/',
  data: {
  username: this.username,
  password: this.password
  }
  })
  .then(function (response) {
  /*
  * 这里的 'login-on'是在Bus中自己构造的一个事件(不需要在别处申明),通过emit()函数
  * 手动触发这个事件,然后在user.vue 中可以监听这个事件,一旦发现事件触发,就可以利用
  * 回调函数接收 response 这个对象,达到信息传递的目的
  * */
  Bus.$emit('login-on',response);
  this.$router.replace({path: 'user'});
  }.bind(this))
  .catch(function (error) {
  console.log(error);
  alert('用户名或密码错误');
  });
 }
 }
} 

在user.vue文件中:

//位于user.vue 中
import Bus from '../bus.js';
export default {
 mouted : {
  Bus.$on('login-on', message => { // 这里的message就是从login.vue中传来的数据
  console.log(message);
 });
 }
}

这样就完成了数据传递。

但是在测试时发现在user.vue组件的监听 没有被触发

分析:在我遇到的情境中,login.vue组件和user.vue 组件是互相替代的关系,即需要从login组件通过vue-router跳转到user组件,在login.vue销毁后才开始渲染user.vue组件,所以在user.vue中的

Bus.$on('login-on', message => 

监听是在事件以及触发之后才开始,自然不会产生回调函数。

由此可以看出,官方推荐的eventbus 解决方案的缺陷在于, 在数据传递过程中,两个组件必须都已经被渲染过。我的这种情景下,这种方法不适用。

 三、vuex解决方案:

熟悉vuex的大佬们请直接跳过吧。

最好在src的根目录下新建一个文件,叫 store.js

//位于store.js 中
import Vue from 'vue'
import Vuex from 'vuex'
Vue.use(Vuex);
export default new Vuex.Store({
 /*
 * state指的就是储存的数据,
 * 下面的数据是我在项目中需要用的数据字段
 * */
 state: {
 has_login: false,
 id: 1,
 mobile_num: '',
 name: ''
 },
 /*
 * mutations里面规定的就是想要改变state(数据)的动作函数,
 * 下面的user_message 就是我将传入的message赋值给仓库中的
 * state数据字段,达到更新数据的目的
 * */
 mutations: {
 user_message (state, message) {
 state.has_login = true;
 state.id = message.data.id;
 state.mobile_num = message.data.mobile_num;
 state.name = message.data.name;
 }
 }
}) 

然后在login.vue组件中,提交收到的用户信息

//位于login.vue 中
import userMessage from '../store';
export default new Vuex.Store({
 methods: {
 loginSubmit () {
 this.$axios({
  method: 'post',
  url: '/student/login/',
  data: {
  username: this.username,
  password: this.password
  }
 })
  .then(function (response) {
  //这里调用 store.js中 mutations里面的user_message函数,从而改变仓库中的state数据
  userMessage.commit('user_message', response);
  this.$router.replace({path: 'user'});
  }.bind(this))
  .catch(function (error) {
  console.log(error);
  alert('用户名或密码错误');
  });
 }
}) 

最后在user.vue组件中接收vuex仓库中存储的信息,即先引入仓库

//位于user.vue 中
import userMessage from '../store'; 

然后可以直接将 userMessage.state 赋值给user.vue作用域中的数据字段,同时,vuex 的state有热更新的属性,对于数据的同步很有帮助,优点良多。

所以,中大型的项目还是在一开始就直接使用vuex是明智的决定,对于开发有很大的便利。

总结

以上所述是小编给大家介绍的vue非父子组件通信问题解决方法,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对我们网站的支持!

(0)

相关推荐

  • Vue非父子组件通信详解

    组件是Vue核心的一块内容,组件之间的通信也是很基本的开发需求.组件通信又包括父组件向子组件传数据,子组件向父组件传数据,非父子组件间的通信.前两种通信Vue的文档都说的很清楚,但是第三种文档上确只有下面的几句 具体如何去实现却没有很详细的说明,于是自己试着进行了实现.先看下简单的通信效果: 就是点击了一个组件,另一个组件的数字递加. html如下: <div id="app"> <component-a></component-a> <com

  • vue中的event bus非父子组件通信解析

    有时候非父子关系的组件也需要通信.在简单的场景下,使用一个空的Vue实例作为中央事件总线: var bus = new Vue() // 触发组件 A 中的事件 bus.$emit('id-selected', 1) // 在组件 B 创建的钩子中监听事件 bus.$on('id-selected', function (id) { // ... }) 在更多复杂的情况下,你应该考虑使用专门的 状态管理模式.就是用到了vuex eventBus是作为兄弟关系的组件之间的通讯中介. 代码示例: <

  • vue2.0父子组件及非父子组件之间的通信方法

    1.父组件传递数据给子组件 父组件数据如何传递给子组件呢?可以通过props属性来实现 父组件: <parent> <child :child-msg="msg"></child>//这里必须要用 - 代替驼峰 </parent> data(){ return { msg: [1,2,3] }; } 子组件通过props来接收数据: 方式1: props: ['childMsg'] 方式2 : props: { childMsg: Arr

  • 详解Vue 非父子组件通信方法(非Vuex)

    一提到两个非父子组件通信方法,有经验的 coder 肯定会说用 Vuex 啊,我个人建议不要为了用 Vuex 而用 Vuex,除非你的项目很大,耦合度很高,需要大量的储存一些 data,组件之间通信频繁.当然还是要根据自己的业务场景的来决定,总之还是那句话,不要为了用 Vuex 而用 Vuex! Vue 官网介绍了非父子组件通信方法: 不过官网说的太简单了,新手看完估计还是一脸懵逼.还有这个空的 Vue 实例放到哪里合适也值得商榷. 这篇文章的目的就是用一个简单的例子让你明白如何用 Bus

  • vue2利用Bus.js如何实现非父子组件通信详解

    前言 大家应该都知道,vue2中废弃了$dispatch和$broadcast广播和分发事件的方法.父子组件中可以用props和$emit().如何实现非父子组件间的通信,可以通过实例一个vue实例Bus作为媒介,要相互通信的兄弟组件之中,都引入Bus,然后通过分别调用Bus事件触发和监听来实现通信和参数传递.下面话不多说了,来一起看看详细的介绍吧. Bus.js可以是这样 import Vue from 'vue' export default new Vue() 在需要通信的组件都引入Bus

  • vue非父子组件通信问题及解决方法

    问题描述: 最近在做登录部分时遇到一个场景,当点击 "用户"按钮时,首先渲染login组件,在用户登录后直接跳转到用户信息界面(user组件).这里遇到了需要将login组件通过异步请求获得的用户信息传向 user组件,但是login和login组件不是父子组件,就暂称为兄弟组件吧. 简而言之,我要解决的就是兄弟组件之间的信息传递问题. //位于login.vue 中 export default { methods:{ login(){ this.$axios({ method: '

  • Vue非父子组件之间的通信方式详解

    目录 非父子组件的通信 1.Provide和Inject 1.1基本使用 1.2处理响应式数据(了解) 2.全局事件总线 总结 非父子组件的通信 此篇讲解的是, 在学习状态管理之前, 非父子间通信的方案 在开发中,我们构建了组件树之后,除了父子组件之间的通信之外,还会有非父子组件之间的通信. 这里我们主要讲两种方式: Provide/Inject: 全局事件总线: 1.Provide和Inject 1.1基本使用 Provide/Inject用于非父子组件之间共享数据: 比如有一些深度嵌套的组件

  • Vue.js 父子组件通信的十种方式

    面试官:Vue 中父子组件通信有哪些方式? 自己先想一分钟. 无可否认,现在无论大厂还是小厂都已经用上了Vue.js 框架,简单易上手不说,教程详尽,社区活跃,第三方套件还多.真的是前端开发人员必备技能.而且在面试当中也往往会问到关于 Vue 方面的各种问题,其中大部分面试官会问到如上这种问题. 最近一直在做 Vue项目代码层面上的优化,说实话,优化别人的代码真是件痛苦的事情,功能实现尚且不说,就说代码规范我就能再写出一篇文章来.真的是无规范不成方圆,规范这个东西太重要了!有点扯了,回到主题,咳

  • vue.js父子组件通信动态绑定的实例

    如下所示: <!DOCTYPE html> <html> <head lang="en"> <meta charset="UTF-8"> <title></title> </head> <body> <div id='app'> <!--这里的作用是将父组件渲染到页面上--> <father></father> </d

  • 如何用Vue实现父子组件通信

    一.父子组件的关系构成 本篇文章将要总结的是Vue中父子组件之间的通信方式. 那在vue中父子组件之间的关系是怎么构成的呢,或者说哪个组件可以称为是父组件,哪个组件又可以称为是子组件呢. 在我的理解中,父子组件的关系构成也比较简单. 使用vue-cli工具构建的项目中,我们经常会在一个组件中注册引用另外一个组件. Home.vue <template> <div class="home"> <p>这里是Home组件</p> </di

  • Vue中父子组件通信与事件触发的深入讲解

    目录 一.组件 子组件 父组件 二.父子组件通信 父组件给子组件通信 子组件向父组件通信 三.父子组件事件触发 父组件调用子组件中的事件方法 子组件调用父组件中的事件方法 四.总结 一.组件 子组件 <template> <div style="border:1px solid black;width:400px; height: 130px;"> <h3>我是子组件</h3> <button>子组件将值传递给父组件</

  • Vue中父子组件的值传递与方法传递

    一.Vue中父组件向子组件传值 利用v-bind向子组件传值,子组件中利用props接受 <div id="app"> <father></father> </div> <template id="father"> <div> <!--注意点: 组件是可以使用自己的数据的--> <p>{{name}}</p> <p>{{age}}</p>

随机推荐