组件与组件之间并不是完全独立的,他们之间可以进行一些数据的传递操作。传递数据的解决方案就是props
选项。
比如定义两个页面Parent.vue
和Child.vue
,其中Parent.vue
包含Child.vue
。
<template>
<div class="div">
<h1>子类组件</h1><br>
<p>msg: {{ msg }}</p><br>
<p>title: {{ title }}</p><br>
</div>
</template>
<script>
export default{
data(){
return{
}
},
// props 数组类型,其中保存父级传入子级数据时,标签上的属性名称
props:["msg","title"]
}
</script>
<style scoped>
.div{
border: 1px solid;
}</style>
<template>
<h1>父类组件</h1><br>
<ChildDemo msg="专注写bug 父级传入子级数据" :title="tittleMsg"/>
</template>
<script>
// 父类中引入子类
import Child from './Child.vue';
export default{
data(){
return{
tittleMsg:"父级传入子级信息2"
}
},
// script 增加 setup,则不能写逻辑,出现报错,所以此处手动注入
components:{
// key-value 结构 别名:对应引入子类
ChildDemo:Child
}
}
</script>
案例效果展示:
如果按照Java语言理解,就很简单。
万物皆对象。既然字符串是这种方式,那么其他类型也大差不差了!
直接看例子:
Parent.vue
<template>
<h1>父类组件</h1><br>
<ChildDemo msg="专注写bug 父级传入子级数据"
:title="tittleMsg"
:age="userAge"
:arrays="userLists"
:userInfo="userInfos" />
</template>
<script>
// 父类中引入子类
import Child from './Child.vue';
export default{
data(){
return{
tittleMsg:"父级传入子级信息2", // 字符串
userAge:28, // number 数字类型
userLists:["lilei","jack","tom"], // 数组类型
userInfos:{ // object 对象类型
id:5173,
name:"lilei"
}
}
},
// script 增加 setup,则不能写逻辑,出现报错,所以此处手动注入
components:{
// key-value 结构 别名:对应引入子类
ChildDemo:Child
}
}
</script>
Child.vue
<template>
<div class="div">
<h1>子类组件</h1><br>
<p>msg: {{ msg }}</p><br>
<p>title: {{ title }}</p><br>
<p>age: {{ age }}</p><br>
<ul>
<li v-for="(item,index) of arrays" :key="index">{{ item }}</li>
</ul>
<p>用户基本信息编号:{{ userInfo.id }} </p>
<p>用户基本信息名称: {{ userInfo.name }}</p>
</div>
</template>
<script>
export default{
data(){
return{
}
},
props:["msg","title","age","arrays","userInfo"]
}
</script>
<style scoped>
.div{
border: 1px solid;
}</style>
props
传递数据操作时,只能从父级传递至子级中,即:从外至内
。
不能反其道而行!
在上面的案例中,父级组件Parent.vue
向子级组件Child.vue
进行了传递数据测试。除了能满足数据传递之外,props
还能针对传递的数据限定类型
、若不存在填充默认值
等操作。
比如父级中传递的userAge
是String
类型,若子级组件中定义的是Number
类型。则会出现什么样的问题呢?看下面的案例。
ComponentA.vue
<template>
<h1>父类组件</h1><br>
<ComponentBDemo
:age="userAge" />
</template>
<script>
// 父类中引入子类
import ComponentB from './ComponentB.vue';
export default{
data(){
return{
userAge:"28", // 传递字符串类型
}
},
// script 增加 setup,则不能写逻辑,出现报错,所以此处手动注入
components:{
// key-value 结构 别名:对应引入子类
ComponentBDemo:ComponentB
}
}
</script>
在子级组件中的props
换一个写法,指定数据的类型。
ComponentB.vue
<template>
<h1>子级组件</h1><br>
<p>age: {{ age }}</p><br>
</template>
<script>
export default{
data(){
return{
}
},
props:{
age:{ // 限定类型
type:Number
}
}
}
</script>
运行后,浏览器查看显示效果。
当然,在子级组件中,可以针对多个可能的类型进行限制,比如满足传入的数据是String
或Number
等。
修改子级组件ComponentB.vue
<template>
<h1>子级组件</h1><br>
<p>age: {{ age }}</p><br>
</template>
<script>
export default{
data(){
return{
}
},
props:{
age:{
//type:Number // 限定单个类型
type:[Number,String,Object,Array] // 支持多种类型范围
}
}
}
</script>
刷新浏览器,查看信息。
如果子级组件中定义了某个变量的显示项,但在父级中未传入对应的值,此时子级组件在渲染显示的时候,不会将该变量标签进行显示。
ComponentB.vue
<template>
<h1>子级组件</h1><br>
<p>age: {{ age }}</p><br>
<p>{{ userName }}</p><br>
</template>
<script>
export default{
data(){
return{
}
},
props:{
age:{
//type:Number // 限定单个类型
type:[Number,String,Object,Array]
},
userName:{
type:String
}
}
}
</script>
子级组件定义userName
变量的显示,但父级未传递值,此时浏览器中的显示信息如下:
如果说父级组件未传递值时,需要子级组件中默认显示一些信息,可以写成下面这种形式。
<template>
<h1>子级组件</h1><br>
<p>age: {{ age }}</p><br>
<p>{{ userName }}</p><br>
</template>
<script>
export default{
data(){
return{
}
},
props:{
age:{
//type:Number // 限定单个类型
type:[Number,String,Object,Array]
},
userName:{
type:String,
default:"父级未传递值,默认显示这句话!"
}
}
}
</script>
核心就是针对未传递值的变量增加default
标识 。
export default{
data(){
return{
}
},
props:{
age:{
//type:Number // 限定单个类型
type:[Number,String,Object,Array]
},
userName:{
type:String,
default:"父级未传递值,默认显示这句话!"
}
}
}
此时页面的显示效果如下所示:
验证数组类型的数据默认值定义。
<template>
<h1>子级组件</h1><br>
<p>age: {{ age }}</p><br>
<p>{{ userName }}</p><br>
<ul>
<li v-for="(item,index) of arrays" :key="index">{{ item }}</li>
</ul>
</template>
<script>
export default{
data(){
return{
}
},
props:{
age:{
//type:Number // 限定单个类型
type:[Number,String,Object,Array]
},
userName:{
type:String,
default:"父级未传递值,默认显示这句话!"
},
arrays:{ // 数组类型的变量
type:Array,
default(){ // 工厂函数返回默认对象
return ["这只是默认的数组展示项"]
}
}
}
}
</script>
数组类型默认值展示效果:
在上面说了一个显示效果:
如果父级
未传递
指定变量数据,则在子级组件
中会渲染对应的标签,但不会给变量赋值!
如果必须强制指定必须传递对应的值,此时则需要使用到required:true
标识。如下所示:
父级未传递值
msg
,子级组件对应变量指定必传!
<template>
<h1>子级组件</h1><br>
<p>age: {{ age }}</p><br>
<p>{{ userName }}</p><br>
<ul>
<li v-for="(item,index) of arrays" :key="index">{{ item }}</li>
</ul>
</template>
<script>
export default{
data(){
return{
}
},
props:{
age:{
//type:Number // 限定单个类型
type:[Number,String,Object,Array]
},
userName:{
type:String,
default:"父级未传递值,默认显示这句话!"
},
arrays:{
type:Array,
default(){
return ["这只是默认的数组展示项"]
}
},
msg:{ // 父级未传递该变量
type:String,
required:true
}
}
}
</script>
此时浏览器中的显示效果如下:
丢失必选项msg值。