Vue异步请求最佳实践

一、当前存在的问题

目前项目前端请求后台数据的方式是这样的:

  1. 页面中methoddispatchaction

  2. action调用mutation,请求axios

  3. 请求到数据后存储到state

  4. 页面中在computed中获取state,使用watch监听到数据变化之后做业务逻辑。

调用流程如图:
现在的请求方式

在当前的项目中,这样的调用方式可以解决异步请求,对于接口的响应速度也很快。但依然有着几点不足。

1.代码冗余

页面中的每一个请求都需要一个method,一个comuputed和一个watch。我们知道computed是计算属性,是Vue数据驱动的重要实现。但在计算属性里获取state这个操作并没有去"计算"。实际上是用了mapstate获取了全局的state,计算属性在这里成了一个state的容器。

2.业务逻辑依赖watch

我们知道watch是监听器,但是用监听器去监听计算属性(数据驱动的)的变化,是本末倒置了。可以看看Vue官方文档是如何描述监听的:

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

watch应该是作为computed的补充,而不是代替。所以当前的问题是大量的业务逻辑写在watch里,而不是在调用接口,数据返回之后立即处理

3.错误的mutation使用

为什么是错误的呢?我们先看看为什么会有上面两个错误的出现。当我们把axios写到mutation的时候,我们发现computed获取到state的数据不是同步的。

image

二、解决问题

我们理想的情况是希望页面调用method之后去请求axios,后端数据返回时立即进行业务处理,处理完成之后界面立刻响应渲染。另外请求出错的时候,我们也希望能有对应的处理。实现上述需求,需要使用Promise来解决。了解Promise的用法,可以参考这篇文章Promise使用详解

使用Promise之后的请求流程变为下图:
image

在当前的请求逻辑下,我们可能觉得很疑惑的一点是,为什么删除的时候需要使用state?按照Vue的定义,state应该是一个公共的变量,所以删除之后的数据我不需要保存到state,但是为了保持页面的响应我们只能使用state来保存,使用computed去获取state,使用watch去响应数据变化。这样的步骤明显是多余的,那么我们看看使用了Promise之后的代码是什么样的,这里以一个删除功能为例:
首先我们需要在action中写一个Promise,在Promise中调用axios。理论上我们可以在页面上直接调用axios,但是由于需要保持API接口与页面解耦,所以页面的method方法依然是调用action。

    actions:{
        /* 删除资源action,进行了HTTP请求状态的判断,页面请求该方法后需要进行处理*/
        async delResBaseInfo(context, id) {
            return new Promise(function(resolve, reject) {
                axios
                    .delete(`${api}/${id}`)
                    .then((response) => {
                        if (response.status == 200) {
                            resolve(Immutable.fromJS(response.data));
                        } else {
                            reject(Immutable.fromJS(response.data));
                        }
                    })
                    .catch(function(error) {
                        console.log(error);
                    });
            });
        }
    }

页面method的调用函数,里面包含了业务逻辑。

    // 删除数据,调用action的axios请求
        deletData(id) {
            this.loading = true;
            this.$store
                .dispatch('resourceBase/delResBaseInfo', id)
                .then((response) => {
                    if (response.get('code') == 1) {
                        this.getData();
                        this.loading = false;
                        successMessage(this, textStandard.deleteSuccess(''));
                    } else {
                        errorMessage(this, textStandard.deleteError(''));
                    }
                })
                .catch((error) => {
                    console.log(error);
                });
        },

使用了Promise之后的代码得到了简化,并且因为Promise的特性,异步问题也能完美解决。

转载于:https://www.cnblogs.com/rever/p/11231685.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值