1 组件通讯

1.1 :props(父传子)

  1. // 数组:不建议使用
  2. props:[]
  3. // 对象
  4. props:{
  5. inpVal:{
  6. type:Number, //传入值限定类型
  7. // type 值可为String,Number,Boolean,Array,Object,Date,Function,Symbol
  8. // type 还可以是一个自定义的构造函数,并且通过 instanceof 来进行检查确认
  9. required: true, //是否必传
  10. default:200, //默认值,对象或数组默认值必须从一个工厂函数获取如 default:()=>[]
  11. validator:(value) {
  12. // 这个值必须匹配下列字符串中的一个
  13. return ['success', 'warning', 'danger'].indexOf(value) !== -1
  14. }
  15. }
  16. }

1.2 :$emit(子传父)

  1. // 父组件
  2. <home @title="title">
  3. // 子组件
  4. this.$emit('title',[{title:'这是title'}])

1.3 :provide和inject(隔代传参)

  1. //父组件:
  2. provide: { //provide 是一个对象,提供一个属性或方法
  3. foo: '这是 foo',
  4. fooMethod:()=>{
  5. console.log('父组件 fooMethod 被调用')
  6. }
  7. },
  8. // 子或者孙子组件
  9. inject: ['foo','fooMethod'], //数组或者对象,注入到子组件
  10. mounted() {
  11. this.fooMethod()
  12. console.log(this.foo)
  13. }
  14. //在父组件下面所有的子组件都可以利用inject

2 获取父类DOM

$root:获取顶层父类DOM,访问根组件,并且可以访问所有下层子组件的DOM
$parent:获取上一层DOM
例子:父组件A 包含 子组件B
子组件B 包含 子组件C
如果要从子组件C想要获取顶层父类,那么两者的使用:
this.$root 相当于 this.$parent.$parent

  1. // 父组件
  2. <home ref="home"/>
  3. mounted(){
  4. console.log(this.$refs.home) //拿到子组件的DOM,可以进行数据操作和方法操作
  5. }
  1. // 父组件
  2. mounted(){
  3. console.log(this.$root) //获取根实例,最后所有组件都是挂载到根实例上
  4. console.log(this.$root.$children[0]) //获取根实例的一级子组件
  5. console.log(this.$root.$children[0].$children[0]) //获取根实例的二级子组件
  6. }

3 : attrs与$listeners

3.1 attrs使用场景

①如果父传子有很多值那么在子组件需要定义多个解决attrs获取子传父中未在 props 定义的值

  1. // 父组件
  2. <home title="这是标题" width="80" height="80" imgUrl="imgUrl"/>
  3. // 子组件
  4. mounted() {
  5. console.log(this.$attrs) //{title: "这是标题", width: "80", height: "80", imgUrl: "imgUrl"}
  6. }

②如果子组件定义了 props,打印的值就是剔除定义的属性

  1. props: {
  2. width: {
  3. type: String,
  4. default: ''
  5. }
  6. },
  7. mounted() {
  8. console.log(this.$attrs) //{title: "这是标题", height: "80", imgUrl: "imgUrl"}
  9. }

3.2 listeners使用场景

子组件需要调用父组件的方法,解决父组件的方法可以通过listeners” 传入内部组件——在创建更高层次的组件时非常有用,如果是孙组件要访问父组件的属性和调用方法,直接一级一级传下去就可以

  1. // 第一种父组件
  2. <home @change="change"/>
  3. // 子组件
  4. mounted() {
  5. console.log(this.$listeners) //即可拿到 change 事件
  6. }
  7. //第二种:子组件中
  8. methods: {
  9. childMethod() {
  10. this.$parent.fatherMethod();
  11. }
  12. }
  13. //第三种:子组件中,然后父组件监听这个事件,父: <child @fatherMethod="fatherMethod"></child>
  14. methods: {
  15. childMethod() {
  16. this.$emit('fatherMethod');
  17. }
  18. }
  19. //第四种:父组件把方法传入子组件中,在子组件里直接调用这个方法
  20. //父
  21. <div>
  22. <child :fatherMethod="fatherMethod"></child>
  23. </div>
  24. //子
  25. <button @click="childMethod()">点击</button>
  26. props: {
  27. fatherMethod: {
  28. type: Function,
  29. default: null
  30. }
  31. },
  32. methods: {
  33. childMethod() {
  34. if (this.fatherMethod) {
  35. this.fatherMethod();
  36. }
  37. }
  38. }

4 parent和children(两个均非响应式)

  1. //父组件
  2. mounted(){
  3. console.log(this.$children)
  4. //可以拿到 一级子组件的属性和方法
  5. //所以就可以直接改变 data,或者调用 methods 方法
  6. }
  7. //子组件
  8. mounted(){
  9. console.log(this.$parent) //可以拿到 parent 的属性和方法
  10. }

5 render函数

  1. //初级版 根据 props 生成标签,性能较低
  2. <template>
  3. <div>
  4. <div v-if="level === 1"> <slot></slot> </div>
  5. <p v-else-if="level === 2"> <slot></slot> </p>
  6. <h1 v-else-if="level === 3"> <slot></slot> </h1>
  7. <h2 v-else-if="level === 4"> <slot></slot> </h2>
  8. <strong v-else-if="level === 5"> <slot></slot> </stong>
  9. <textarea v-else-if="level === 6"> <slot></slot> </textarea>
  10. </div>
  11. </template>
  1. //优化版,利用render 函数减小了代码重复率,性能较高
  2. <template>
  3. <div>
  4. <child :level="level">Hello world!</child>
  5. </div>
  6. </template>
  7. <script type='text/javascript'>
  8. import Vue from 'vue'
  9. Vue.component('child', {
  10. render(h) {
  11. const tag = ['div', 'p', 'strong', 'h1', 'h2', 'textarea'][this.level-1]
  12. return h(tag, this.$slots.default)
  13. },
  14. props: {
  15. level: { type: Number, required: true }
  16. }
  17. })
  18. export default {
  19. name: 'hehe',
  20. data() { return { level: 3 } }
  21. }
  22. </script>

6 递归组件

场景:如果开发一个 tree 组件,里面层级是根据后台数据决定的,这个时候就需要用到动态组件

  1. // 递归组件: 组件在它的模板内可以递归的调用自己,只要给组件设置name组件就可以了。
  2. // 设置那么House在组件模板内就可以递归使用了,不过需要注意的是,
  3. // 必须给一个条件来限制数量,否则会抛出错误: max stack size exceeded
  4. // 组件递归用来开发一些具体有未知层级关系的独立组件。比如:
  5. // 联级选择器和树形控件
  6. <template>
  7. <div v-for="(item,index) in treeArr">
  8. 子组件,当前层级值: {{index}} <br/>
  9. <!-- 递归调用自身, 后台判断是否不存在改值 -->
  10. <tree :item="item.arr" v-if="item.flag"></tree>
  11. </div>
  12. </template>
  13. <script>
  14. export default {
  15. // 必须定义name,组件内部才能递归调用
  16. name: 'tree',
  17. data(){
  18. return {}
  19. },
  20. // 接收外部传入的值
  21. props: {
  22. item: {
  23. type:Array,
  24. default: ()=>[]
  25. }
  26. }
  27. }
  28. </script>

递归组件必须设置name 和结束的阀值

7 components和 Vue.component

components:局部注册组件

  1. export default{
  2. components:{home}
  3. }

Vue.component:全局注册组件

  1. Vue.component('home',home)