задний план
- У вас все еще есть проблемы с Uncaught (в обещании) ReferenceError?
- Вы все еще постоянно пишете try catch для перехвата исключений?
- Вы все еще пишете уловки для каждого обещания?
Пришло время для единой унифицированной обработки исключений! ! ! (для проектов vue, включая vuex)
глобальный захват исключений
Vue.config.errorHandler = function (err, vm, info) {
// 指定组件的渲染和观察期间未捕获错误的处理函数。这个处理函数被调用时,可获取错误信息和 Vue 实例。
// handle error
// `info` 是 Vue 特定的错误信息,比如错误所在的生命周期钩子
// 只在 2.2.0+ 可用
}
Примечание. При обработке исключений синхронные исключения и асинхронные исключения следует обрабатывать отдельно.
Анализ исходного кода vue core
Прочтите исходный код, чтобы увидеть, как vue предоставляет пользователям интерфейс Vue.config.errorHandler.
Решение для синхронной обработки исключений
// 定义异常处理函数,判断用户是否自定义Vue.config.errorHandler,定义则直接调用,未定义执行vue本身异常处理。
function globalHandleError(err, vm, info) {
if (Vue.config.errorHandler) {
try {
return config.errorHandler.call(null, err, vm, info)
} catch (e) {
logError(e, null, 'config.errorHandler');
}
}
logError(err, vm, info);
}
try {
// vue正常执行代码被包裹在try内,有异常会调用globalHandleError
} catch (e) {
globalHandleError(e, vm, '对应信息');
}
Решение для асинхронной обработки исключений
// 定义异步异常处理函数,对于自身没有捕获异常的promise统一执行catch
function invokeWithErrorHandling(
handler,
context,
args,
vm,
info
) {
var res;
try {
res = args ? handler.apply(context, args) : handler.call(context);
if (res && !res._isVue && isPromise(res) && !res._handled) {
res.catch(function (e) { return handleError(e, vm, info + " (Promise/async)"); });
// 异步代码例如promise可以统一为其定义Promise.prototype.catch()方法。
res._handled = true;
}
} catch (e) {
handleError(e, vm, info);
}
return res
}
// 所有的钩子函数调用异常处理函数
function callHook(vm, hook) {
var handlers = vm.$options[hook];
// 为所有钩子增加异常处理
var info = hook + " hook";
if (handlers) {
for (var i = 0, j = handlers.length; i < j; i++) {
invokeWithErrorHandling(handlers[i], vm, null, vm, info);
}
}
}
расширение знаний
// vue接口是能处理同步异常以及部分钩子中的异步异常,对于方法中的异常无法有效处理,我们可以仿照源码增加方式中的异步异常处理,避免为每一个promise写catch
Vue.mixin({
beforeCreate() {
const methods = this.$options.methods || {}
Object.keys(methods).forEach(key => {
let fn = methods[key]
this.$options.methods[key] = function (...args) {
let ret = fn.apply(this, args)
if (ret && typeof ret.then === 'function' && typeof ret.catch === "function") {
return ret.catch(Vue.config.errorHandler)
} else { // 默认错误处理
return ret
}
}
})
}
})
// 针对vuex中的action也做了异常统一处理,核心代码见下面
function registerActionHandle(actions) {
Object.keys(actions).forEach(key => {
let fn = actions[key]
actions[key] = function (...args) {
let ret = fn.apply(this, args)
if (isPromise(ret)) {
return ret.catch(errorHandler)
} else { // 默认错误处理
return ret
}
}
})
}
const registerVuex = (instance) => {
if (instance.$options['store']) {
let actions = instance.$options['store']['_actions'] || {}
if (actions) {
let tempActions = {}
Object.keys(actions).forEach(key => {
tempActions[key] = actions[key][0]
})
registerActionHandle(tempActions)
}
}
}
полный код
Ниже приведен полный код для глобальной обработки исключений, который был инкапсулирован в плагин.
errorPlugin.js
/**
* 全局异常处理
* @param {
* } error
* @param {*} vm
*/
function isPromise(ret) {
return (ret && typeof ret.then === 'function' && typeof ret.catch === "function")
}
const errorHandler = (error, vm, info) => {
console.error('抛出全局异常')
console.error(vm)
console.error(error)
console.error(info)
}
function registerActionHandle(actions) {
Object.keys(actions).forEach(key => {
let fn = actions[key]
actions[key] = function (...args) {
let ret = fn.apply(this, args)
if (isPromise(ret)) {
return ret.catch(errorHandler)
} else { // 默认错误处理
return ret
}
}
})
}
const registerVuex = (instance) => {
if (instance.$options['store']) {
let actions = instance.$options['store']['_actions'] || {}
if (actions) {
let tempActions = {}
Object.keys(actions).forEach(key => {
tempActions[key] = actions[key][0]
})
registerActionHandle(tempActions)
}
}
}
const registerVue = (instance) => {
if (instance.$options.methods) {
let actions = instance.$options.methods || {}
if (actions) {
registerActionHandle(actions)
}
}
}
let GlobalError = {
install: (Vue, options) => {
/**
* 全局异常处理
* @param {
* } error
* @param {*} vm
*/
Vue.config.errorHandler = errorHandler
Vue.mixin({
beforeCreate() {
registerVue(this)
registerVuex(this)
}
})
Vue.prototype.$throw = errorHandler
}
}
export default GlobalError
использовать
// 在入口文件中引入
import ErrorPlugin from './errorPlugin'
import Vue from 'vue'
Vue.use(ErrorPlugin)
напиши в конце
Добавление глобальной обработки исключений помогает
- Улучшить надежность кода
- уменьшить количество сбоев
- Быстро найти ошибки
полный код
Ссылка на данные
Сделайте код более художественным!!!
о нас
Передовая команда Kuaigou Taxi занимается обменом передовыми технологиями и регулярно публикует высококачественные статьи. Добро пожаловать, обратите внимание и лайкните.