当前位置:   article > 正文

vue3.0 setup简单使用示例_vue3 setup demo

vue3 setup demo

setup

vue2.x

了解vue3的setup之前,可以先回顾一下vue2.x中组件传值的写法。

// App.vue
<template>
  <div class="app">
    App
    <Demo name="张三" age="18" />
  </div>
</template>

<script>
import Demo from "./components/demo.vue";
export default {
  name: "App",
  components: {
    Demo,
  },
};
</script>

<style>
.app {
  width: 100%;
  height: 100px;
  background-color: beige;
}
</style>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
// demo
<template>
  <div class="demo">demo组件</div>
</template>

<script>
export default {
  name: "demo",
  props: ["name", "age"],
  mounted() {
    console.log("this", this);
  },
};
</script>

<style>
.demo {
  width: 50%;
  height: 50px;
  background-color: pink;
}
</style>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22

通过打印this可以看到this上存在name和age
在这里插入图片描述
当组件中不写props时,虽然可以在 a t t r 中 也 能 获 取 到 , 但 是 区 别 于 p r o p s , attr中也能获取到,但是区别于props, attrprops,attr中无法对传参进行类型约束,且无法在template中直接使用。
在这里插入图片描述
另外,当使用props接收时,$attr上是拿不到name和age。
在这里插入图片描述
再来看一下slot,即使没有在组件中使用slot时,会作为虚拟节点显示在 $slot中,使用后就会作为 $slot中的一个对象。
在这里插入图片描述
具名插槽,当组件中不接收时
在这里插入图片描述
以上分别演示了 $attr和 $slot写法,现在看vue3的setup

-setup的执行时机

  • 在beforeCrate之前,并且this是undefined。
    setup的参数
  • props:值为对象,包含:组件外部传递过来,且组件内部声明接受了的属性。
  • context:上下文对象
    attrs:值为对象,包含:组件外部传递过来,但没有在props配置中声明的属性,相当于this.$attrs
    slots:收到的插槽内容,相当于this. $slots
    emit:分发自定义事件的函数,相当于this. $emit

先写一个简单的模板

// views/home.vue
<template>
  <Demo></Demo>
</template>

<script>
import Demo from "@/components/demo.vue";
export default {
  components: {
    Demo,
  },
};
</script>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
// components/demo.vue
<template>
  <p>个人信息</p>
  <p>姓名:{{ person.name }}</p>
  <p>年龄:{{ person.age }}</p>
</template>

<script lang="ts">
import { defineComponent, ref, reactive } from "vue";
export default defineComponent({
  name: "Home",
  components: {},
  setup() {
    const person = reactive({
      name: "张三",
      age: 18,
    });

    return {
      person,
    };
  },
});
</script>

  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25

显示正常
在这里插入图片描述
首先代码验证setup执行时机,在beforeCrate之前,并且this是undefined

// components/demo.vue
<template>
  <p>个人信息</p>
  <p>姓名:{{ person.name }}</p>
  <p>年龄:{{ person.age }}</p>
</template>

<script lang="ts">
import { defineComponent, ref, reactive } from "vue";
export default defineComponent({
  name: "Home",
  components: {},
  beforeCreate() {
    console.log("---beforeCreate---");
  },
  setup() {
    console.log("---setup---");
    const person = reactive({
      name: "张三",
      age: 18,
    });

    return {
      person,
    };
  },
});
</script>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28

打印后可以发现,setup执行时机是在beforeCrate之前,且this为undefined
在这里插入图片描述
再来看setup的参数

export default {
  name: "Home",
  setup(a, b, c) {
    console.log("a", a);
    console.log("b", b);
    console.log("c", c);
    const person = reactive({
      name: "张三",
      age: 18,
    });

    return {
      person,
    };
  },
};
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16

在这里插入图片描述
先来第一个参数,官方定义props,为组件传值,来写一个例子并且打印

// views/home.vue
<template>
  <Demo hobby="学习"></Demo>
</template>

<script>
import Demo from "@/components/demo.vue";
export default {
  components: {
    Demo,
  },
};
</script>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
// components/demo.vue
<template>
  <p>个人信息</p>
  <p>姓名:{{ person.name }}</p>
  <p>年龄:{{ person.age }}</p>
</template>

<script>
import { reactive } from "vue";
export default {
  name: "Home",
  setup(props) {
    console.log("props", props);
    const person = reactive({
      name: "张三",
      age: 18,
    });

    return {
      person,
    };
  },
};
</script>

  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25

在这里插入图片描述
会发现打印的内容是空的,并且有一段警告,意思是外部传递了hobby,组件内部并没有调用。
正确写法与2.x一致

// components/demo.vue
<script>
import { reactive } from "vue";
export default {
  name: "Home",
  props: ["hobby"],
  setup(props) {
    console.log("props", props);
    const person = reactive({
      name: "张三",
      age: 18,
    });

    return {
      person,
    };
  },
};
</script>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19

在这里插入图片描述
并且在多传参数的情况下,不会有警告

// components/demo.vue
<script>
import { reactive } from "vue";
export default {
  name: "Home",
  props: ["hobby", "interest"],
  setup(props) {
    console.log("props", props);
    const person = reactive({
      name: "张三",
      age: 18,
    });

    return {
      person,
    };
  },
};
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18

在这里插入图片描述
正确的打印(再添加一个传值interest),可以发现props将传值整理成一个带有响应式的对象。
在这里插入图片描述
第二个参数叫context,可以理解成一个上下文对象

<script>
import { reactive } from "vue";
export default {
  name: "Home",
  props: ["hobby", "interest"],
  setup(props, context) {
    console.log("props", props);
    console.log("context", context);
    const person = reactive({
      name: "张三",
      age: 18,
    });

    return {
      person,
    };
  },
};
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18

在这里插入图片描述

先打印一下attrs,发现并没有需要的属性
在这里插入图片描述
当注释掉props时,再打印attrs
在这里插入图片描述
与vue2.x相似,会有个警告和props的数据被attrs接收了,接着试一下事件emit和打印结果

// view/home.vue 父组件
<template>
  <Demo @hello="showHobby" hobby="学习" interest="还是学习"></Demo>
</template>

<script>
import Demo from "@/components/demo.vue";
export default {
  components: {
    Demo,
  },
  setup() {
    function showHobby(val) {
      alert(`触发hello事件,收到参数是:${val}`);
    }
    return {
      showHobby,
    };
  },
};
</script>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
// components/demo.vue 子组件
<template>
  <p>个人信息</p>
  <p>姓名:{{ person.name }}</p>
  <p>年龄:{{ person.age }}</p>
  <button @click="test">测试组件的hello事件</button>
</template>

<script>
import { reactive } from "vue";
export default {
  name: "Home",
  props: ["hobby", "interest"],
  emits: ["hello"],
  setup(props, context) {
    console.log("props", props);
    // console.log("context.attrs", context.attrs);
    console.log("context.attrs", context.emit);
    const person = reactive({
      name: "张三",
      age: 18,
    });
    function test() {
      context.emit("hello", 666);
    }
    return {
      person,
      test,
    };
  },
};
</script>

  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33

在这里插入图片描述
最后看一下slots

<template>
  <Demo @hello="showHobby" hobby="学习" interest="还是学习">
    <template v-slot:qwerb>
      <span>在家</span>
    </template>
  </Demo>
</template>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7

控制台看一下打印结果

在这里插入图片描述

<template>
  <p>个人信息</p>
  <p>姓名:{{ person.name }}</p>
  <p>年龄:{{ person.age }}</p>
  <p>
    <slot name="qwerb"></slot>
  </p>
  <button @click="test">测试组件的hello事件</button>
</template>

  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10

在这里插入图片描述

声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:【wpsshop博客】
推荐阅读
相关标签
  

闽ICP备14008679号