Vue组件数据传递与props校验方式
作者:专注写bug
前言
组件与组件之间并不是完全独立的,他们之间可以进行一些数据的传递操作。
传递数据的解决方案就是props
选项。
组件数据传递的几种类型
简单字符串类型数据专递
比如定义两个页面Parent.vue
和Child.vue
,其中Parent.vue
包含Child.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>
- >Parent.vue
<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
还能针对传递的数据限定类型
、若不存在填充默认值
等操作。
限定数据类型 type
比如父级中传递的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类型,类型不一致出现了警告!
当然,在子级组件中,可以针对多个可能的类型进行限制,比如满足传入的数据是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>
刷新浏览器,查看信息。
给定默认值 default
如果子级组件中定义了某个变量的显示项,但在父级中未传入对应的值,此时子级组件在渲染显示的时候,不会将该变量标签进行显示。
- 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:"父级未传递值,默认显示这句话!" } } }
此时页面的显示效果如下所示:
如果父级传递了数据。那么显示效果又是怎么样的呢?
- >ComponentA.vue
此时浏览器中的显示效果如下所示:
【注意】数字Number和字符串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
在上面说了一个显示效果:
- 如果父级未传递指定变量数据,则在子级组件中会渲染对应的标签,但不会给变量赋值!
- 如果必须强制指定必须传递对应的值,此时则需要使用到
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值。
总结
以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。