前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >最佳实践:vue组件引用传值(续篇)

最佳实践:vue组件引用传值(续篇)

作者头像
奋飛
发布2022-09-08 12:19:24
3510
发布2022-09-08 12:19:24
举报
文章被收录于专栏:Super 前端Super 前端

本文是对 最佳实践:vue组件引用传值 的续篇:

上文中有提及,当父子组件对引用类型需要同步修改时,在子组件中需要切断相关引用,避免引用传值传递破坏 vue 单向数据流 的响应机制(如果直接修改 prop,vue 则会发出告警)。

但最近在项目代码中,看到了类似这样的写法(简化后):

index.vue

代码语言:javascript
复制
<div id="app">
  <el-input v-model="data.a"></el-input>
  <child v-model="data"></child>
  {{data}}
</div>
<script>
var Main = {
  data() {
    return {
      data: {
        a: '123'
      }
    }
  }
}
var Ctor = Vue.extend(Main)
new Ctor().$mount('#app')
</script>

child.vue

代码语言:javascript
复制
<el-input v-model="form.a" placeholder="请输入内容"></el-input>
<script>
export default {
  name: 'child',
  prop: ['value'],
  computed: {
    form: {
      get () {
        return this.value
      },
      set (val) {
        this.$emit('input', val)
      }
    }
  }
}  
</script>

上述写法貌似很优雅,也实现了父子组件联动响应的诉求。但仔细思考,会发现,这正是 “引用传值” 的副作用,违背了 vue 单向数据流的设计初衷,之所以没有告警是因为“不是使用同一变量,重新声明了变量”。

验证是否为同一引用值

代码语言:javascript
复制
// 在 child 中增加验证机制
watch: {
  'form': {
    handler (val) {
      console.log(this.form === this.value)	// true
    },
      deep: true
  }
}

所以,上述写法等价于

child.vue

代码语言:javascript
复制
<el-input v-model="form.a" placeholder="请输入内容"></el-input>
<script>
 export default {
  name: 'child',
  prop: ['value'],
  data () {
    return {
      form: this.value
    }
  }
}
</script>

这里,同样也实现了父子组件联动响应的诉求(借助 “引用传值” 的副作用)

综上,上述两种方式都不可取,需要大家深刻理解 vue 单向数据流思想,不要为了节省代码(或偷懒),产生难以维护的代码!!!

子组件 ① 切断引用;② 发生变化 $emit('input', form) 才是正解!!!

  • 切断一层:Object.assign()
  • 切断多层:JSON.parse(JSON.stringify())

延展思考:是否有场景传引用,就是为了使用一个对象呢?

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2022-07-15,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档