视频1 视频21 视频41 视频61 视频文章1 视频文章21 视频文章41 视频文章61 推荐1 推荐3 推荐5 推荐7 推荐9 推荐11 推荐13 推荐15 推荐17 推荐19 推荐21 推荐23 推荐25 推荐27 推荐29 推荐31 推荐33 推荐35 推荐37 推荐39 推荐41 推荐43 推荐45 推荐47 推荐49 关键词1 关键词101 关键词201 关键词301 关键词401 关键词501 关键词601 关键词701 关键词801 关键词901 关键词1001 关键词1101 关键词1201 关键词1301 关键词1401 关键词1501 关键词1601 关键词1701 关键词1801 关键词1901 视频扩展1 视频扩展6 视频扩展11 视频扩展16 文章1 文章201 文章401 文章601 文章801 文章1001 资讯1 资讯501 资讯1001 资讯1501 标签1 标签501 标签1001 关键词1 关键词501 关键词1001 关键词1501 专题2001
如何使用vue中的watcher
2020-11-27 19:35:34 责编:小采
文档


这篇文章主要介绍了vue 中的 watcher的相关资料,需要的朋友可以参考下

观察 Watchers

虽然计算属性在大多数情况下更合适,但有时也需要一个自定义的 watcher 。这是为什么 Vue 提供一个更通用的方法通过watch 选项,来响应数据的变化。当你想要在数据变化响应时,执行异步操作或开销较大的操作,这是很有用的。

大家对于 watch 应该不陌生,项目中都用过下面这种写法:

watch: {
 someProp () {
 // do something
 }
}
// 或者
watch: {
 someProp: {
 deep: true,
 handler () {
 // do something
 }
 }
}

上面的写法告诉 vue,我需要监听 someProp 属性的变化,于是 vue 在内部就会为我们创建一个 watcher 对象。(限于篇幅,我们不聊 watcher 的具体实现,感兴趣的可以直接看源码 watcher)

然而在 vue 中,watcher 的功能并没有这么单一,先上段代码:

<template>
 <p>
 <p>a: {{ a }}</p>
 <p>b: {{ b }}</p>
 <button @click="increment">+</button>
 </p>
</template>
<script>
export default {
 data () {
 return {
 a: 1
 }
 },
 computed: {
 b () {
 return this.a * 2
 }
 },
 watch: {
 a () {
 console.log('a is changed')
 }
 },
 methods: {
 increment () {
 this.a += 1
 }
 },
 created () {
 console.log(this._watchers)
 }
}
</script>

在线demo

上面代码非常简单,我们现在主要关注 created 钩子中打印的 this._watchers,如下:

分别展开三个 watcher,观察每一个 expression,从上到下分别为:

b() { return this.a * 2;? }
"a"
function () { vm._update(vm._render(), hydrating);? }

上面三个 watcher 代表了三种不同功能的 watcher,我们将其按功能分为三类:

  • 在 watch 中定义的,用于监听属性变化的 watcher (第二个)

  • 用于 computed 属性的 watcher (第一个)

  • 用于页面更新的 watcher (第三个)

  • normal-watcher

    我们在 watch 中定义的,都属于这种类型,即只要监听的属性改变了,都会触发定义好的回调函数

    computed-watcher

    每一个 computed 属性,最后都会生成一个对应的 watcher 对象,但是这类 watcher 有个特点,我们拿上面的 b 举例:

    属性 b 依赖 a,当 a 改变的时候,b 并不会立即重新计算,只有之后其他地方需要读取 b 的时候,它才会真正计算,即具备 lazy(懒计算)特性

    render-watcher

    每一个组件都会有一个 render-watcher, function () {? vm._update(vm._render(), hydrating);? }, 当 data/computed

    中的属性改变的时候,会调用该 render-watcher 来更新组件的视图

    三种 watcher 的执行顺序

    除了功能上的区别,这三种 watcher 也有固定的执行顺序,分别是:

    computed-render -> normal-watcher -> render-watcher

    这样安排是有原因的,这样就能尽可能的保证,在更新组件视图的时候,computed 属性已经是最新值了,如果 render-watcher 排在 computed-render 前面,就会导致页面更新的时候 computed 值为旧数据。

    下面从一段实例代码中看下vue中的watcher

    在这个示例中,使用 watch 选项允许我们执行异步操作(访问一个 API),我们执行该操作的频率,并在我们得到最终结果前,设置中间状态。这是计算属性无法做到的。

    <p id="watch-example">
    <p>
    Ask a yes/no question:
    <input v-model="question">
    </p>
    <p>{{ answer }}</p>
    </p>
    <!-- Since there is already a rich ecosystem of ajax libraries -->
    <!-- and collections of general-purpose utility methods, Vue core -->
    <!-- is able to remain small by not reinventing them. This also -->
    <!-- gives you the freedom to just use what you're familiar with. -->
    <script src="https://unpkg.com/axios@0.12.0/dist/axios.min.js"></script>
    <script src="https://unpkg.com/lodash@4.13.1/lodash.min.js"></script>
    <script>
    var watchExampleVM = new Vue({
    el: '#watch-example',
    data: {
    question: '',
    answer: 'I cannot give you an answer until you ask a question!'
    },
    watch: {
     // 如果 question 发生改变,这个函数就会运行
    question: function (newQuestion) {
    this.answer = 'Waiting for you to stop typing...'
    this.getAnswer()
    }
    },
    methods: {
     // _.debounce 是一个通过 lodash 操作频率的函数。
     // 在这个例子中,我们希望访问yesno.wtf/api的频率
     // ajax请求直到用户输入完毕才会发出
     // 学习更多关于 _.debounce function (and its cousin
    // _.throttle), 参考: https://lodash.com/docs#debounce
    getAnswer: _.debounce(
    function () {
    var vm = this
    if (this.question.indexOf('?') === -1) {
    vm.answer = 'Questions usually contain a question mark. ;-)'
    return
    }
    vm.answer = 'Thinking...'
    axios.get('https://yesno.wtf/api')
    .then(function (response) {
    vm.answer = _.capitalize(response.data.answer)
    })
    .catch(function (error) {
    vm.answer = 'Error! Could not reach the API. ' + error
    })
    },
    // 这是我们为用户停止输入等待的毫秒数
    500
    )
    }
    })
    </script>

    上面是我整理给大家的,希望今后会对大家有帮助。

    相关文章:

    在Vue中有关响应式原理(详细教程)

    在angularjs中如何实现柱状图动态加载

    在Angular作用域中scope的如何使用

    使用react如何实现菜单权限控制

    详细解读vue.js中props如何传递参数

    下载本文
    显示全文
    专题