文章目录
什么是组件?
组件 (Component) 是 Vue.js 最强大的功能之一。组件可以扩展 HTML 元素,封装可重用的代码。在较高层面上,组件是自定义元素,Vue.js 的编译器为它添加特殊功能。
二、创建全局组件
方式一
1、Vue.extend
var com1 = Vue.extend({
// 通过 template 属性,指定了组件要展示的HTML结构
template: '<h3>这是使用 Vue.extend 创建的组件</h3>'
})
2、Vue.component
Vue.component
(‘组件的名称’, 创建出来的组件模板对象) 注册组件
Vue.component('myCom1', com1)
注意:如果使用Vue.Component 注册全局组件的时候,组件的名称使用了驼峰命名,则在引用组件的时候需要把大写的驼峰改为小写的字母,同时,两个单词之前,使用 “–” 链接。如果不使用则直接拿名称来使用即可。
方式二
直接使用Vue.component
Vue.component('mycom2', {
template: '<div><h3>这是直接使用 Vue.component 创建出来的组件</h3>
<span>123</span></div>'
})
示例:
方式三
1、被控制的 #app 外面,使用 template 元素,定义组件的HTML模板结构。
<template id="tmpl">
<div>
<h1>这是通过 template 元素,在外部定义的组件结构</h1>
<h4>好用,不错!</h4>
</div>
</template>
2、使用id注册组件
Vue.component('mycom3', {
template: '#tmpl'
})
三、 创建局部组件
局部组件的创建和全局组件的创建方法一样。唯一区别的是,局部组件是在Vue实例中定义的。
四、组件中的data 和 methods
组件中的data 和实例中的data 有点不一样,实例中的data 可以为一个对象。但是组件中的data必须是一个函数,还必须返回一个对象。
data为什么是一个函数?
- 这是普通的写法------data函数的形式返回一个对象:
这时点击第一个组件实例的增加和减少,两个组件实例的值不会影响彼此。也就是说,这样每复用一次组件,就会返回一份新的data(类似于给每个组件实例创建一个私有的数据空间,让各个组件实例维护各自的数据)。如果换成下面的写法: - 直接以对象形式返回:
<template>
<div>
<h2>计数器:{{counter}}</h2>
<button @click="increment">+</button>
<button @click="decrement">-</button>
</div>
</template>
<script>
export default{
name:'ceshi',
data:{ counter:0 },
methods:{
increment(){
this.counter++
},
decrement(){
this.counter--
}
}
}
</script>
这时,两个计数器互相影响,保持同步,这不是我们想要的结果。Object是引用数据类型,里面保存的是内存地址,单纯的写成对象形式,就使得所有组件实例共用了一份data,就会造成一个变了全都会变的结果。
总结
组件中的data写成一个函数,数据以函数返回值形式定义,这样每复用一次组件,就会返回一份新的data【类似于给每个组件实例创建一个私有的数据空间,让各个组件实例维护各自的数据】。
从js原型链角度来解释:JavaScript只有函数构成作用域(注意理解作用域,只有函数{}
构成作用域,对象的{}
以及if(){}
都不构成作用域)。当data是一个函数时,每个组件实例都有自己的作用域,每个实例相互独立,就不会相互影响。
五、组件间的通信方式
props/$emit
父组件A通过props的方式向子组件B传递,B to A 通过在 B 组件中 $emit, A 组件中 v-on 的方式实现。
子组件:
<template>
<div class="hello">
<ul>
<li v-for="(user,index) in users" v-bind:key="index">{{ user }}</li>
</ul>
</div>
</template>
<script>
export default {
name: "users",
props: {
users: { //父组件中子标签自定义的名字
type: Array,
require: true
}
}
}
</script>
<style scoped>
li{
list-style-position: inside;
}
</style>
父组件:
<template>
<div id="app">
<img alt="Vue logo" src="./assets/logo.png">
<Users v-bind:users="users"> </Users>
</div>
</template>
<script>
import Users from "@/components/users";
export default {
name: 'App',
data(){
return {
users: ['西安邮电','西安石油','西北政法','西安工业','西安财经']
}
},
components: {
Users,
}
}
</script>
通过事件形式
子组件:
<template>
<header>
<h1 @click="changeTitle">{{ title }}</h1>
</header>
</template>
<script>
export default {
name: "Son",
data(){
return {
title: 'Vue.js Demo'
}
},
methods: {
changeTitle(){
this.$emit('titleChanged','西安邮电大学');
}
}
}
</script>
<style scoped>
h1{
background-color: greenyellow;
}
</style>
父组件:
<template>
<div id="app">
<Son v-on:titleChanged="updateTitle"></Son>
<h2>{{ title }}</h2>
</div>
</template>
<script>
import Son from "@/components/Son";
export default {
name: "Father",
data(){
return {
title: '传递的是一个值'
}
},
methods: {
updateTitle(e){
this.title = e
}
},
components:{
Son,
}
}
</script>
【总结】子组件通过events(事件)给父组件发送消息,实际上就是子组件把自己的数据发送到父组件。