try-catch仅捕获同步运行时错误(如ReferenceError、TypeError),无法捕获异步错误;Promise错误需用.catch()或await+try-catch处理;catch应分类处理、记录上下文、友好提示;finally用于清理但不改变错误流向。
它只捕获**运行时同步错误**,比如 ReferenceError、TypeError、SyntaxError(仅限 eval 内)、RangeError 等。异步操作(如 setTimeout、fetch、Promise 回调)里的错误默认逃逸出 try 块,不会被捕获。
常见误判场景:
try { fetch('/api').then(...); } → fetch 本身不抛错,但 .then 里 throw new Error() 不会被捕获try { JSON.parse('{"a":}') } → 会触发 SyntaxError,能捕获try { undefined.foo() } → 触发 TypeError,能捕获不能靠外层 try-catch 包住 .then(),得用 .catch() 或 await + try-catch。
推荐写法(现代 JS):
async function loadData() {
try {
const res = await fetch('/api');
if (!res.ok) throw new Error(`HTTP ${res.status}`);
const data = await res.json();
return data;
} catch (err) {
console.error('请求或解析失败:', err.message);
// 这里能拿到 fetch 网络错误、JSON 解析错误、手动 throw 的错误
}
}
注意点:
await 后必须是 Promise,否则不会挂起,也不会把 rejection 转为异常fetch 只在网络断开等极少数情况 reject;HTTP 4xx/5xx 仍返回 resolve 的 Response 对象,需手动检查 res.ok
catch 里吞掉错误又不 re-throw —— 后续逻辑可能因 data 是 undefined 再次崩核心原则:**明确错误来源,做最小必要响应,别掩盖问题**。
实用建议:
if (err instanceof TypeError) vs if (err.name === 'AbortError')
err.message)alert('保存失败,请检查网络后重试'),而非 alert(err.stack)
throw err 或 return Promise.reject(err),让上层决定是否兜底它不管成功失败都会
执行,适合清理工作,但**不能访问 err 或返回值**。
典型用例:
loading.value = false
controller.abort()(配合 fetch 的 signal)注意:finally 中抛出新错误会覆盖原错误;若需透传原错误,别在 finally 里 throw。
真正容易被忽略的是:很多人以为 finally 能“兜住”所有异常 —— 它只是保证执行,不改变错误流向。