1 组件通讯
1.1 :props(父传子)
// 数组:不建议使用props:[]// 对象props:{inpVal:{type:Number, //传入值限定类型// type 值可为String,Number,Boolean,Array,Object,Date,Function,Symbol// type 还可以是一个自定义的构造函数,并且通过 instanceof 来进行检查确认required: true, //是否必传default:200, //默认值,对象或数组默认值必须从一个工厂函数获取如 default:()=>[]validator:(value) {// 这个值必须匹配下列字符串中的一个return ['success', 'warning', 'danger'].indexOf(value) !== -1}}}
1.2 :$emit(子传父)
// 父组件<home @title="title">// 子组件this.$emit('title',[{title:'这是title'}])
1.3 :provide和inject(隔代传参)
//父组件:provide: { //provide 是一个对象,提供一个属性或方法foo: '这是 foo',fooMethod:()=>{console.log('父组件 fooMethod 被调用')}},// 子或者孙子组件inject: ['foo','fooMethod'], //数组或者对象,注入到子组件mounted() {this.fooMethod()console.log(this.foo)}//在父组件下面所有的子组件都可以利用inject
2 获取父类DOM
$root:获取顶层父类DOM,访问根组件,并且可以访问所有下层子组件的DOM
$parent:获取上一层DOM
例子:父组件A 包含 子组件B
子组件B 包含 子组件C
如果要从子组件C想要获取顶层父类,那么两者的使用:
this.$root 相当于 this.$parent.$parent
// 父组件<home ref="home"/>mounted(){console.log(this.$refs.home) //拿到子组件的DOM,可以进行数据操作和方法操作}
// 父组件mounted(){console.log(this.$root) //获取根实例,最后所有组件都是挂载到根实例上console.log(this.$root.$children[0]) //获取根实例的一级子组件console.log(this.$root.$children[0].$children[0]) //获取根实例的二级子组件}
3 : attrs与$listeners
3.1 attrs使用场景
①如果父传子有很多值那么在子组件需要定义多个解决attrs获取子传父中未在 props 定义的值
// 父组件<home title="这是标题" width="80" height="80" imgUrl="imgUrl"/>// 子组件mounted() {console.log(this.$attrs) //{title: "这是标题", width: "80", height: "80", imgUrl: "imgUrl"}}
②如果子组件定义了 props,打印的值就是剔除定义的属性
props: {width: {type: String,default: ''}},mounted() {console.log(this.$attrs) //{title: "这是标题", height: "80", imgUrl: "imgUrl"}}
3.2 listeners使用场景
子组件需要调用父组件的方法,解决父组件的方法可以通过listeners” 传入内部组件——在创建更高层次的组件时非常有用,如果是孙组件要访问父组件的属性和调用方法,直接一级一级传下去就可以
// 第一种父组件<home @change="change"/>// 子组件mounted() {console.log(this.$listeners) //即可拿到 change 事件}//第二种:子组件中methods: {childMethod() {this.$parent.fatherMethod();}}//第三种:子组件中,然后父组件监听这个事件,父: <child @fatherMethod="fatherMethod"></child>methods: {childMethod() {this.$emit('fatherMethod');}}//第四种:父组件把方法传入子组件中,在子组件里直接调用这个方法//父<div><child :fatherMethod="fatherMethod"></child></div>//子<button @click="childMethod()">点击</button>props: {fatherMethod: {type: Function,default: null}},methods: {childMethod() {if (this.fatherMethod) {this.fatherMethod();}}}
4 parent和children(两个均非响应式)
//父组件mounted(){console.log(this.$children)//可以拿到 一级子组件的属性和方法//所以就可以直接改变 data,或者调用 methods 方法}//子组件mounted(){console.log(this.$parent) //可以拿到 parent 的属性和方法}
5 render函数
//初级版 根据 props 生成标签,性能较低<template><div><div v-if="level === 1"> <slot></slot> </div><p v-else-if="level === 2"> <slot></slot> </p><h1 v-else-if="level === 3"> <slot></slot> </h1><h2 v-else-if="level === 4"> <slot></slot> </h2><strong v-else-if="level === 5"> <slot></slot> </stong><textarea v-else-if="level === 6"> <slot></slot> </textarea></div></template>
//优化版,利用render 函数减小了代码重复率,性能较高<template><div><child :level="level">Hello world!</child></div></template><script type='text/javascript'>import Vue from 'vue'Vue.component('child', {render(h) {const tag = ['div', 'p', 'strong', 'h1', 'h2', 'textarea'][this.level-1]return h(tag, this.$slots.default)},props: {level: { type: Number, required: true }}})export default {name: 'hehe',data() { return { level: 3 } }}</script>
6 递归组件
场景:如果开发一个 tree 组件,里面层级是根据后台数据决定的,这个时候就需要用到动态组件
// 递归组件: 组件在它的模板内可以递归的调用自己,只要给组件设置name组件就可以了。// 设置那么House在组件模板内就可以递归使用了,不过需要注意的是,// 必须给一个条件来限制数量,否则会抛出错误: max stack size exceeded// 组件递归用来开发一些具体有未知层级关系的独立组件。比如:// 联级选择器和树形控件<template><div v-for="(item,index) in treeArr">子组件,当前层级值: {{index}} <br/><!-- 递归调用自身, 后台判断是否不存在改值 --><tree :item="item.arr" v-if="item.flag"></tree></div></template><script>export default {// 必须定义name,组件内部才能递归调用name: 'tree',data(){return {}},// 接收外部传入的值props: {item: {type:Array,default: ()=>[]}}}</script>
7 components和 Vue.component
components:局部注册组件
export default{components:{home}}
Vue.component:全局注册组件
Vue.component('home',home)
