04、手把手教Vue--props

PS:转载请注明出处
作者: TigerChain
地址: https://www.jianshu.com/p/87ff57037d9c
本文出自 TigerChain 简书 手把手教 Vue 系列

本节大纲

本节大纲

教程简介

  • 1、阅读对象
    本篇教程适合新手阅读,老手直接略过
  • 2、教程难度
    初级,本人水平有限,文章内容难免会出现问题,如果有问题欢迎指出,谢谢
  • 3、Demo 地址:https://github.com/tigerchain/vue-lesson 请看 02、vue组件 组件这一节

正文

一、v-bind

在说 props 之前,我们先说说 v-bind 这个指令

1、什么是 v-bind?

v-bind 用来动态的绑定一个或多个属性,或一个组件的 prop ,比如绑定图片地址、标签的样式等,v-bind 的缩写是 :

2、v-bind 的用法

比如,我们给 img 一个图片资源地址

<img v-bind:src="imageSrc"/>

再比如,我给一个 div 设置 css 样式,如下

<div v-bind:class="{divstyle:isShowStyle}"></div>

<style>
 .divstyle {
    background-color: red;
 }
</style>

<script>
 var vm = new Vue({
   el:"#root",
   data:{
      // 如果 isShowStyle 为 true 的话,那么 
      // v-bind:class="{divstyle:isShowStyle}" 表示此 div 的 class 就是 divstyle
      isShowStyle:true
   }
 })
</script>

等等,拿官方的举例来说,v-bind 可以绑定属性「以数据驱动」,在绑定 class 或 style「内联样式」 的时候支持数据或对象,以下是官方举例基本上把 v-bind 的使用场景介绍完了,html 的属性只能使用 v-bind 来绑定「除非你使用原生写法,不使用 v-bind」

<!-- 绑定一个属性 -->
<img v-bind:src="imageSrc">

<!-- 缩写 -->
<img :src="imageSrc">

<!-- 内联字符串拼接 -->
<img :src="'/path/to/images/' + fileName">

<!-- class 绑定 -->
<div :class="{ red: isRed }"></div>
<div :class="[classA, classB]"></div>
<div :class="[classA, { classB: isB, classC: isC }]">

<!-- style 绑定 -->
<div :style="{ fontSize: size + 'px' }"></div>
<div :style="[styleObjectA, styleObjectB]"></div>

<!-- 绑定一个有属性的对象 -->
<div v-bind="{ id: someProp, 'other-attr': otherProp }"></div>

<!-- 通过 prop 修饰符绑定 DOM 属性 -->
<div v-bind:text-content.prop="text"></div>

<!-- prop 绑定。“prop”必须在 my-component 中声明。-->
<my-component :prop="someThing"></my-component>

<!-- 通过 $props 将父组件的 props 一起传给子组件 -->
<child-component v-bind="$props"></child-component>

<!-- XLink -->
<svg><a :xlink:special="foo"></a></svg>

3、写个 demo 感受一下吧,这里直接在 script 引入 vue.js 来写

  • 1、核心代码如下:
<body>
  <div id="container">
    <!-- 第一种写法 其中 img 是宽度为 35 的 style 样式-->
    <img v-bind:src="imgSrc" v-bind:class="{'img':show}" >
    <!-- 第二种写法 -->
    <img v-bind:src="'./imgs/logo.png'" v-bind:style="imgStyle" >
    <p v-bind:style="{color:'red'}">{{msg}}</p>
    <!-- 标签的属性的都可以使用 v-bind 替换 -->
    <input v-bind:type="'text'"  v-bind:placeholder="'输入内容'" v-model="clickBtnText"><!-- :value="''"  和 v-model 同时出现会报错-->
    {{clickBtnText}} <br/>
    <!-- 这里是行内样式 -->
    <input type="button" value="测试" v-bind:style="{'margin-top':'10px'}">
  </div>
  <script>
    // 定义一个样式
    var imgStyle = {
      height: '40px',
      width: '40px'
    }

    var vm = new Vue({
      el: '#container',
      data: {
        msg:"123",
        imgStyle:imgStyle,
        imgSrc:'./imgs/logo.png',
        show:true,
        clickBtnText:''
      }
    })
  </script>
</body>
  • 2、查看结果
v-bind 结果

我们看到我们使用 v-bind 设置图片,设置样式「行内样式和行外样式」,并且代替 html 原有的属性等,并且成功显示,调试窗口也没有报错

二、props 属性

Vue 和 React 一样,是使用 props「单向数据流」 来传递数据的「父组件的数据传递给后代组件」,props 在本组件中是不能修改的「和 React 也是一毛一样」

1、一般格式

我们知道组件有四种格式,不管那种格式 props 用法都是相同的,这里以全局组件为例子

Vue.component('customcomponent',{
    // 定义接收的 props 等待父组件传递
    props:['message'],
    template: '<span>{{message}}</span>',
    ... 数据和方法等省略

})

2、直接写代码感受一下

在以下 demo 中,我们使用 props 来传递字符串、对象、样式来感受一下 props

  • 1、直接在上面的代码基础上修改,添加以下代码
<hr>
<h4>以下是 props 例子</h4>
<div id="app2">
<!-- 如果要使用 props 来传递对象,就要使用 v-bind  -->
<mycomponent message="你好" :mydata="{username:'tigerchain',age:28}" name-style="color:red"></mycomponent>
</div>

<script>
// 定义一个组件
var myComponent = Vue.extend({
  // 定义 props 
  props: {
    message:"",
    mydata:{},
    //样式 ,如果这里使用驼峰标识 ,则在标签中使用就要使用 name-style 传递
    nameStyle:{}
  },
  template: '<div>{{ message }} <div> <span v-bind:style="nameStyle">{{ mydata.username}}</span></div> </div>'
})

// 全局注册组件 组件的别名要是小写,否则会报错
Vue.component('mycomponent', myComponent)

var vm2 = new Vue({
  el:'#app2',
})
</script>

为了便于观察,我们重新定义了一个 id 为 app2 的 div 和一个新的 Vue 实例「vm2」 ,我们定义一个组件并且添加如下的 props

添加 props属性

在使用组件的时候就可以传递数据了

props 传递方法
  • 2、运行查看结果
传递 props 结果

我们看到,传递过来的数据完美的接收到并显示出来了,在这里注意以几下几点

  • 如果 props 中声明的是数据是驼峰标识,那么传值的时候就要使用 - 标识

比如 props 中的数据是 myData 那么在组件中传值时就要使用 my-data="" ,如果没有使用驼峰那不牵扯

  • 在组件中使用 props 传值的时候默认传的是字符串

如果想要传值「比如对象,数据等」那么就要使用 v-bind="" 来传值,上面的 message 和 mydata 就显示了这两种方式

  • props 中就定义一些默认的值,不能直接修改 props,只能间接的通过 data 来修改「在 data 里面定义数据,值就是 props ,这一点 React 是一样的」,其实 Vue 中的 data 就和 React 中的 state 是一样的

  • props 中的值,我们在 Vue 实例中可以使用 this.props 的值来访问「和仿问 data 是一样的」

三、 props 验证

想要写一个有“良心”的组件,指定 props 的验证规则非常有必要,比如我们在 props 中定义传递过来的 name 必须是字符串否则就发出警告「这对使用插件的人来说是非常友好的」

1、验证规则

直接引用官方的验证规则吧

 props: {
    // 基础类型检测 (`null` 指允许任何类型)
    propA: Number,
    // 可能是多种类型
    propB: [String, Number],
    // 必传且是字符串
    propC: {
      type: String,
      // 必须传
      required: true
    },
    // 数值且有默认值
    propD: {
      type: Number,
      default: 100
    },
    // 数组/对象的默认值应当由一个工厂函数返回
    propE: {
      type: Object,
      default: function () {
        return { message: 'hello' }
      }
    },
    // 自定义验证函数
    propF: {
      validator: function (value) {
        return value > 10
      }
    }
  }

其中 type 有以下几种类型:

  • String:字符串
  • Number:数字
  • Boolean: boolean 类型
  • Function: props 能传递方法,这样的话,组件方法的回调是非常 easy 的
  • Object: 对象
  • Array: 数组
  • Symbol: 原始类型

2、写段代码看一下吧,我们在原有代码基础上添加如下代码「红框所示」

props 中添加规则

在这里我们就可以传递方法「实现了方法的回调」,并且规定了 age 只能传递数字,然后再看 vm2 实例中添加方法,如下所示

props-method

那么我们的 show 方法是如何来的呢,细心的朋友会发现,我们在组件中调用了 props 中声明的方法,这个 show 就是我们在使用组件的时候伟递过来的,我们看看代码

<div id="app2">
    <!-- 如果要使用 props 来传递对象,就要使用 v-bind  -->
    <mycomponent
      message="你好"
      :mydata="{username:'tigerchain',age:28}"
      name-style="color:red"
      age="100"
      :clickme="show()"
      />
</div>

由于 clickme 是 props 中规定的必须是方法,所以这里传递一个方法过去,在 vue 实例中直接实现这个方法就 ok 了「这样我们就可以在多组件开发中使用 props 来传递方法来达到方法回调的作用」,并且我们为了验证 age 故意传了一个字符串,我们在 chrome 调试容器能看到报错了,说 age 要传一个数字「这里就不截图了,大家可以自行实验」,上面的例子虽然把方法回调回来了,但是这在实际开发中还是有缺陷的,回调回来的方法没有带个参数「这肯定不能满足常用的开发呀」,我们修改一下上面的代码来实现回调传参

修改一:props 传递方法处

props-pass-funvalue.png

修改二:调用 props 传参方法

props-pass-vaule-call.png

修改三:回调方法实现

props-callback-fun.png

经过以上三处修改,我们就使用 props 来回调子组件的方法「并且带参数」完美实现了

3、显示结果如下

props-pass-fun-callback.gif

我们可以看到,使用 props 传递方法,并且把子组件中的"你好"当作方法参数回传回来

以上我们就把 props 基本上介绍完了,细心的朋友可以发现,我们在 props 中传递方法的时候,未修改之前的方法是带括号的,后面的方法没有带括号,那么这有什么区别呢?再给大家赠送一些 Vue 中的方法的一些知识

三、Vue 中的方法

Vue 中的方法可以定义在 methods 中「这不废话吗」,也可以定义在 data 中,针对于无参数方法可以带括号,也可以不带括号来说明「有参数方法则必须带括号」,那么这些到底有什么区别呢?我们实际写一个例子来总结一下吧

  • 1、新建一个 vue-method.html 文件「并引入 vue.js」

核心代码如下

<h3>Vue 中的方法</h3>
  <div id="container">
    <button v-bind:class="{bg:isShowbgStyle}" @click="testDataFun">在 data 调用 testDataFun 方法</button>
    <button v-bind:class="{bg:isShowbgStyle}" @click="testDataFun()">在 data 调用 testDataFun() 方法</button>
    <button v-bind:class="{bg:isShowbgStyle}" @click="testInMethodsFun">在 methods 调用 testInMethodsFun 方法</button>
    <button v-bind:class="{bg:isShowbgStyle}" @click="testInMethodsFun()">在 data 调用 testInMethodsFun() 方法</button>
  </div>

  <script>
    var vm = new Vue({
      el:"#container",
      data:{
        isShowbgStyle:true,
        testDataFun(){
          console.log(this,arguments)
        }
      },
      methods: {
        testInMethodsFun() {
          console.log(this,arguments)
        }
      }
    })
  </script>
  • 运行查看结果
vue-method.png

通过 Demo 我们清楚的知道以下结论

  • 1、在 data 或是 methods 中调用无参方法,那么默认会把 Event 对象传递进去,比如上面的 1、3 方法
  • 2、如果想让 this 指向 Vue 实例化的对象,如果在 data 方法调用就必须加上括号
  • 3、在 Vue methods 中的方法不管加参数还是不加参数「带括号或是不带括号」,this 都指向 Vue 的实例化对象
  • 4、综上所术,我们把 Vue 中的方法都放在 methods 中是最好的选择,如果想要想绑定事件,那么不加括号是最好的选择。

有的朋友可能会问了,那我想在 data/methods 中声明方法,既绑定事件又传参数如何搞呢?可以做到吗,肯定可以,我们添加测试代码如下

<button v-bind:class="{bg:isShowbgStyle}" @click="testInMethodsFun('msg',$event)">在 data 调用 testInMethodsFun(参数,事件) 方法</button>

我们可以使用 $event 显示的把原生事件传递过去,查看一下输出结果

Vue$3 {_uid: 0, _isVue: true, $options: {…}, _renderProxy: Proxy, _self: Vue$3, …} 

Arguments(2) ["msg", MouseEvent, callee: ƒ, Symbol(Symbol.iterator): ƒ]

怎么样,Arguments 参数中第一个是方法参数,第二个就是传递过来的事件,在 data 方法是同理

到此为止,我们把 vue props 就说完了,请大家动手写写感受一下

四、回顾一下

  • v-bind 是用来绑定属性的和 props 传递对象或值使用的「props 默认传递的是字符串,如果想要传值、对象,那么就要使用 v-bind」

  • 如果 props 中声明的是数据是驼峰标识,那么传值的时候就要使用 - 标识

    如:props 中的数据是 myData 那么在组件中传值时就要使用 my-data="" ,如果没有使用驼峰那不牵扯

  • props 用来定义一些默认值,我们不能直接修改 props 的值,只能通过 data 来间接修改

  • 要写一个"合格"的 vue 组件,那么 props 最好加上验证

  • props 也可以用来传递方法,达到父组件调用子组件方法并且回调的目标

  • vue 中的方法建议写在 methods 中,无参方法调用时就不要加括号

怎么样,说了这么多就这么几个知识点,再次提示一定要动手写一写「感受一下」

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,951评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,606评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,601评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,478评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,565评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,587评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,590评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,337评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,785评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,096评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,273评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,935评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,578评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,199评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,440评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,163评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,133评论 2 352