Vue.js:Nuxt 错误处理
使用 vuex 设置错误处理有点困难。似乎有很多方法可以做到这一点,但关于正确错误处理的文档却很少。我一直在尝试四种替代方案,但还没有找到令人满意的解决方案。
替代方案 1 - 在组件上捕获和处理错误
在 pages/login.vue 中:
export default {
methods: {
onLogin() {
this.$store.dispatch('auth/login', {
email: this.email,
password: this.password,
}).then(() => {
this.$router.push('/home');
}).catch((error) {
// handle error in component
});
},
},
}
在 store/auth.js 中:
export const actions = {
login({ commit }, { email, password }) {
return this.$axios.post('/api/login', {
email,
password,
}).then((res) => {
doSomething(res);
});
},
}
优点
- 嗯。
缺点
- 未在 vuex 中处理和存储错误。
- 在组件方法中引入大量样板代码。
替代方案 2 - 捕获并在 vuex 中处理错误
在 pages/login.vue 中:
export default {
methods: {
onLogin() {
this.$store.dispatch('auth/login', {
email: this.email,
password: this.password,
}).then(() => {
this.$router.push('/home');
});
},
},
}
在 store/auth.js 中:
export const actions = {
login({ commit }, { email, password }) {
return this.$axios.post('/api/login', {
email,
password,
}).then((res) => {
doSomething(res);
}).catch((error) => {
// store error in application state
commit('SET_ERROR', error);
});
},
}
优点
- 可以使用 vuex 从任何组件访问错误对象
- 可以在布局中使用反应式错误组件,当错误状态发生变化时会显示该组件。
缺点
- 我不确定是否有办法跟踪错误的来源,即从哪个组件抛出错误。
替代方案 3 - 使用 axios 捕获错误拦截器
在 plugins/axios.js 中:
export default function({ $axios, store }) {
$axios.onError(error => {
store.dispatch('setError', error);
});
}
在 pages/login.vue 中:
export default {
methods: {
onLogin() {
this.$store.dispatch('auth/login', {
email: this.email,
password: this.password,
}).then(() => {
this.$router.push('/home');
});
},
},
}
在 store/auth.js 中:
export const actions = {
login({ commit }, { email, password }) {
return this.$axios.post('/api/login', {
email,
password,
}).then((res) => {
doSomething(res);
});
},
}
优点
- 全局错误处理
- 无需在 vuex 或组件中捕获错误
- 没有样板代码
缺点
- 所有异常均未处理,这意味着未捕获非 axios 错误。
替代方案4 - 自定义错误插件
我一直在尝试实现一个捕获所有异常的自定义插件,但我无法让它工作。
在 plugins/catch.js 中:
export default (ctx, inject) => {
const catchPlugin = function (func) {
return async function (args) {
try {
await func(args)
} catch (e) {
return console.error(e)
}
}
};
ctx.$catch = catchPlugin;
inject('catch', catchPlugin);
}
在 pages/login.vue 中:
export default {
methods: {
onLogin: this.$catch(async function () {
await this.$store.dispatch('auth/login', { email: this.email, password: this.password });
this.$router.push('/home');
}),
},
}
优点
- 没有样板。
- 插件中捕获所有错误。
缺点
- 我无法让它工作。 :(
我的印象是,vue/nuxt 中缺少有关错误处理的文档。有人能让第四种替代方案奏效吗?这是理想的吗?还有其他替代方案吗?什么是常规的?
感谢您的时间!
选项 #4 不起作用的原因是您返回了一个永远不会执行的函数:
function catchPlugin(outerFunction) {
return function async innerFunction(args) {
try {
const data = await outerFunction(args);
return { data }
} catch (error) {
return { error }
}
}
}
用法:
const execute = catchPlugin((args) => {
// do something
})
execute('myArgument');
如您所见,您还需要执行内部函数,以使您的示例正常工作:
onLogin: this.$catch(async function () {
await this.$store.dispatch('auth/login', { email: this.email, password: this.password });
this.$router.push('/home');
})(), // mind the () :-)
但是......我相信处理组件中的错误并不是一件坏事,因为这与您的视图组件紧密相关。例如,考虑一个登录组件,我们现在看到的是一个全局错误处理程序(toastr),如果用户名/密码不正确,它将显示一个 toast 消息。根据我的经验,这不是最好的行为,这是一个很好的起点,但更好的方法是将错误消息添加到靠近组件的位置,显示到底出了什么问题。这意味着您将始终必须在组件本身中添加错误处理(与 UI 相关)。
我们公司也有同事在开发同一产品,我们也在为此苦苦挣扎。一个是添加错误处理,另一个则没有。在我看来,唯一的解决方案是教育开发人员始终添加适当的错误处理。
async/await
的语法并不是那么糟糕:
methods: {
async login (email, password) {
try {
await this.$store.dispatch('auth/login', { email, password })
// do something after login
} catch (error) {
// handle error
}
}
}
关于您的
缺点
的最后一件事:
错误未处理并存储在 vuex 中。
。为什么这是一个缺点?您是否需要让错误全局可用?我看到
很多
的是人们在
vuex
中放入了太多无用的状态,而这些状态仅在组件本身中使用。拥有本地组件状态并不坏。由于它与登录有关,因此只有在登录组件中才应该知道此错误。
在
action
中使用
Promise
vuex 中的示例:
NEW_AUTH({ commit }) {
return new Promise((resolve, reject) => {
this.$axios.$get('/token').then((res) => {
...
resolve();
}).catch((error) => {
reject(error);
})
})
}
在页面中:
this.$store.dispatch('NEW_AUTH')
.then(() => ... )
.catch((error) => ... )
要解决 替代方案 2 中的 缺点 ,您可以
(a) 传入组件的名称,甚至是对组件的引用,或者
(b) 您可以将错误保留在进行调用的组件的状态中。然后,您可以在组件中检查是否存在错误并显示它。为此,您可以使用 mixin 来放弃对样板的需求。
在 store/auth.js 中:
export const actions = {
login({ commit }, { email, password }) {
return this.$axios.post('/api/login', {
email,
password,
}).then((res) => {
doSomething(res);
commit('save_to_state', { response: res });
}).catch((error) => {
commit('save_to_state', { error });
});
},
}