最后
正值招聘旺季,很多小伙伴都询问我有没有前端方面的面试题!
开源分享:【大厂前端面试题解析+核心总结学习笔记+真实项目实战+最新讲解视频】
try {
// 可能会导致错误的代码
} catch (error) {
// 在错误发生时怎么处理
}
如果 try 块中的任何代码发生了错误,就会立即退出代码执行过程,然后执行 catch 块。此时 catch 块会接收到一个包含错误信息的对象,这个对象中包含的信息因浏览器而异,但共同的是有一个保存着错误信息的 message 属性。
finally 子句在 try-catch 语句中是可选的,但是 finally 子句一经使用,其代码无论如何都会执行。换句话说,try 语句块中代码全部正常执行,finally 子句会执行;如果因为出错执行了 catch 语句,finally 子句照样会执行。只要代码中包含 finally 子句,则无论 try 或 catch 语句中包含什么代码——甚至是 return 语句,都不会阻止 finally 子句执行。来看下面函数的执行结果:
function testFinally {
try {
return “出去玩”;
} catch (error) {
return “看电视”;
} finally {
return “做作业”;
}
return “睡觉”;
}
表面上调用这个函数会返回 “出去玩”,因为返回 “出去玩” 的语句位于 try 语句块中,而执行此语句又不会出错。实际上返回 “做作业”,因为最后还有 finally 子句,结果就会导致 try 块里的 return 语句被忽略,也就是说调用的结果只能返回 “做作业”。如果把 finally 语句拿掉,这个函数将返回 “出去玩”。因此,在使用 finally 子句之前,一定要非常清楚你想让代码怎么样。(思考一下如果 catch 块和 finally 块都抛出异常,catch 块的异常是否能抛出)
但令人遗憾的是,try-catch 无法处理异步代码和一些其他场景。接下来让我具体分析几种异常场景及其处理方案。
四、异常分析
1. JS 代码错误
下面为我司内部错误监控平台一次日常报错的调用堆栈截图:
错误还是比较明显的,this 指向导致的问题。onOk 使用普通函数时,函数内执行语句的 this 上下文为 Antd.Modal 组件的实例,而 Antd.Modal 组件不存在 changeFilterType 这个方法。将 onOK 方法像 onCancel 方法一样改成箭头函数,将 this 指向父组件即可。
TypeError 类型在 JavaScript 中会经常遇到,在变量中保存着意外类型时,或者在访问不存在的方法时,都会导致这种错误。错误的原因虽然多种多样,但归根结底还是由于在执行特定类型的操作时,变量的类型并不符合要求所致。再看几个例子:
class People {
constructor(name) {
this.name = name;
}
sing() {}
}
const xiaoming = new People(“小明”);
xiaoming.dance(); // 抛出 TypeError
xiaoming.girlfriend.name; // 抛出 TypeError
代码错误一般在开发和测试阶段就能发现。用 try-catch 也能捕获到:
// 代码
try {
xiaoming.girlfriend.name;
} catch (error) {
console.log(xiaoming.name + “没有女朋友”, error);
}
// 运行结果
// 小明没有女朋友 TypeError: Cannot read property ‘name’ of undefined
2. JS 语法错误
我们修改一下代码,我们把英文分号改成中文分号:
try {
xiaoming.girlfriend.name;// 结尾是中文分号
} catch(error) {
console.log(xiaoming.name + “没有女朋友”, error);
}
// 运行结果
// Uncaught SyntaxError: Invalid or unexpected token
SyntaxError 语法错误我们无法通过 try-catch 捕获到,不过语法错误在我们开发阶段就可以看到,应该不会顺利上到线上环境。
不过凡事总有例外,线上还是能收到一些语法错误的告警,但多半是 JSON 解析出错和浏览器兼容性导致。
再看几个例子:
JSON.parse(‘{name:xiaoming}’); // Uncaught SyntaxError: Unexpected token n in JSON at position 1
JSON.parse(‘{“name”:xiaoming}’); // Uncaught SyntaxError: Unexpected token x in JSON at position 8
JSON.parse(‘{“name”:“xiaoming”}’); // 正常
var testFunc () => { }; // 在 IE 下会抛出 SyntaxError,因为 IE 不支持箭头函数,需要通过Babel等工具事先转译下
使用 JSON.parse 解析时出现异常就是一个很好的使用 try-catch 的场景:
try {
JSON.parse(remoteData); // remoteData 为服务端返回的数据
} catch {
console.error(“服务端数据格式返回异常,无法解析”, remoteData);
}
并不是捕获到错误就结束了,捕获到错误后,我们需要思考当错误发生时:
-
错误是否是致命的,会不会导致其它连带错误
-
后续的代码逻辑还能不能继续执行,用户还能不能继续操作
-
是不是需要将错误信息反馈给用户,提示用户如何处理该错误
-
是不是需要将错误上报服务端
对应上面的问题这里就会有很多解决方案了,譬如:
- 如果是服务器未知异常导致,可以阻塞用户操作,弹窗提示用户"服务器异常,请稍后重试"。并提供给用户一个刷新的按钮;
try {
return JSON.parse(remoteData);
} catch (error) {
Modal.fail(“服务器异常,请稍后重试”);
return false;
}
- 如果是数据异常导致,可阻塞用户操作,弹窗提示用户"服务器异常,请联系客服处理~",同时将错误信息上报异常服务器,开发人员通过异常堆栈和用户埋点定位问题原因;
try {
return JSON.parse(remoteData);
} catch (error) {
Modal.fail(“服务器异常,请联系客服处理~”);
logger.error(“JSON数据解析出现异常”, error);
return false;
}
- 如果数据解析出错属于预料之中的情况,也有替代的默认值,那么当解析出错时直接使用默认值也可以;
try {
return JSON.parse(remoteData);
} catch (error) {
console.error(“服务端数据格式返回异常,使用本地缓存数据”, erorr);
return localData;
}
任何错误处理策略中最重要的一个部分,就是确定错误是否致命。
3. 异步错误
try {
setTimeout(() => {
undefined.map(v => v);
}, 1000)
} catch(e) {
console.log(“捕获到异常:”, e);
}
Uncaught TypeError: Cannot read property ‘map’ of undefined
at :3:15
并没有捕获到异常,try-catch
对语法和异步错误却无能为力,捕获不到,这是需要我们特别注意的地方。
五、异常捕获
5.1 window.onerror
当 JS
运行时错误发生时,window
会触发一个 ErrorEvent
接口的 error
事件,并执行window.onerror()
。
/**
* @param {String} message 错误信息
* @param {String} source 出错文件
* @param {Number} lineno 行号
* @param {Number} colno 列号
* @param {Object} error Error对象(对象)
*/
window.onerror = function (message, source, lineno, colno, error) {
console.log(“捕获到异常:”, { message, source, lineno, colno, error });
};
同步错误可以捕获到,但是,请注意 window.error
无法捕获静态资源异常和 JS 代码错误。
5.2 静态资源加载异常
方法一:onerror 来捕获
这样可以拿到静态资源的错误,但缺点很明显,代码的侵入性太强了,每一个静态资源标签都要加上 onerror 方法。
方法二:addEventListener(“error”)

由于网络请求异常不会事件冒泡,因此必须在捕获阶段将其捕捉到才行,但是这种方式虽然可以捕捉到网络请求的异常,但是无法判断 HTTP 的状态是 404 还是其他比如 500 等等,所以还需要配合服务端日志才进行排查分析才可以。
5.3 Promise 异常
Promise 中的异常不能被 try-catch 和 window.onerror 捕获,这时候我们就需要监听 unhandledrejection 来帮我们捕获这部分错误。
window.addEventListener(“unhandledrejection”, function (e) {
e.preventDefault();
console.log(“捕获到 promise 错误了”);
console.log(“错误的原因是”, e.reason);
console.log(“Promise 对象是”, e.promise);
return true;
});
Promise.reject(“promise error”);
new Promise((resolve, reject) => {
reject(“promise error”);
});
new Promise((resolve) => {
resolve();
}).then(() => {
throw “promise error”;
});
5.4 React 异常
React 处理异常的方式不同。虽然 try-catch 适用于许多非普通 JavaScript 应用程序,但它只适用于命令式代码。因为 React 组件是声明性的,所以 try-catch 不是一个可靠的选项。为了弥补这一点,React 实现了所谓的错误边界。错误边界是 React 组件,它“捕获子组件树中的任何地方的 JavaScript 错误”,同时还记录错误并显示回退用户界面。
class ErrorBoundary extends React.Component {
constructor(props) {
super(props);
this.state = { hasError: false };
}
componentDidCatch(error, info) {
// 展示出错的UI
this.setState({ hasError: true });
// 将错误信息上报到日志服务器
logErrorToMyService(error, info);
}
最后
由于文档内容过多,为了避免影响到大家的阅读体验,在此只以截图展示部分内容
vaScript 错误”,同时还记录错误并显示回退用户界面。
class ErrorBoundary extends React.Component {
constructor(props) {
super(props);
this.state = { hasError: false };
}
componentDidCatch(error, info) {
// 展示出错的UI
this.setState({ hasError: true });
// 将错误信息上报到日志服务器
logErrorToMyService(error, info);
}
最后
[外链图片转存中…(img-z6nmotw3-1715748065620)]
[外链图片转存中…(img-djrRX1pJ-1715748065620)]
由于文档内容过多,为了避免影响到大家的阅读体验,在此只以截图展示部分内容