美文网首页Vue
前端异常捕获和定位

前端异常捕获和定位

作者: Gopal | 来源:发表于2020-01-15 22:38 被阅读0次

    前言

    于前端而言,不管是开发还是生产阶段,异常的捕获和定位都是至关重要的。

    开发阶段,通过详细的报错信息,我们可以快速定位并解决问题。在生产,通过异常监控,根据异常埋点信息,我们可以第一时间知道异常信息,不至于造成严重后果。

    window.onerror

    全局监听异常来捕获

    借鉴下 MDN 的说明,当 JavaScript 运行时错误(包括语法错误)发生时候, window 会触发一个 ErrorEvent 接口的 error 事件,并执行 window.onerror()。加载一个全局的 error 事件处理函数可用于自动收集错误报告。

    语法:

    window.onerror = function(message, source, lineno, colno, error) { ... }
    

    参数说明:

    • message:错误信息(字符串)。可用于HTML onerror=""处理程序中的event
    • source:发生错误的脚本URL(字符串)
    • lineno:发生错误的行号(数字)
    • colno:发生错误的列号(数字)
    • error:Error 对象

    若该函数返回true,则阻止执行默认事件处理函数,也就是不会在控制台打印错误。

    好的,我们通过一个实例解析下

    直接上代码

    <html>
      <head>
        <script type="text/javascript">
          onerror = handleErr;
          var txt = "";
          function handleErr(msg, source, lineno, colno, error) {
            txt = "There was an error on this page.\n\n";
            txt += "错误信息: " + msg + "\n";
            txt += "发生错误的脚本URL: " + source + "\n";
            txt += "发生错误的行数: " + lineno + "\n\n";
            txt += "发生错误的列数: " + colno + "\n\n";
            txt += "Click OK to continue.\n\n";
            alert(txt);
            return false;
          }
          function message() {
            adddlert("Welcome guest!");
          }
        </script>
      </head>
    
      <body>
        <input type="button" value="View message" onclick="message()" />
      </body>
    </html>
    

    点击 View message 的时候,就会显示如下弹窗

    另外控制台会有报错,这是我们最后的 return false。假如 return true 是看不到相关的报错信息的

    在 onerror 的回调函数中,我们发送相关的埋点信息(相关的报错信息,行数,列数等等)到我们的监控平台,就可以实现基础的页面监控了

    try...catch...

    try...catch...。其中 try 指定要运行的代码块,catch 指定该代码块运行错误时候,抛出的响应。

    比如:

    try {
      nonExistentFunction();
    }
    catch(error) {
      console.error(error);
      // expected output: ReferenceError: nonExistentFunction is not defined
      // Note - error messages will vary depending on browser
    }
    

    我们使用 try...catch... 最主要是不会因为一处报错,导致我们页面挂掉。在catch 中我们也可以发送相关埋点到我们的监控平台。

    关于 Vue 异常捕获

    之所以会存在这种场景,是因为 Vue 自身已经通过 try...catch... 处理,而不会触发 window.onerror 事件,所以我们有时候也需要专门对 Vue 进行异常捕获

    我们可以使用 Vue.config.errorHandler 对 Vue 进行全局的异常捕获

    指定组件的渲染和观察期间未捕获错误的处理函数。这个处理函数被调用时,可获取错误信息和 Vue 实例。在处理函数中,我们除了发送相关的埋点信息,可以在控制台打印一下相关的报错信息,注意默认这个捕获的方法是不会在控制台打印的,这对于我们开发来讲是不友好的

    Vue.config.errorHandler = function (err, vm, info) {
      // handle error
      // `info` 是 Vue 特定的错误信息,比如错误所在的生命周期钩子
      // 只在 2.2.0+ 可用
    }
    

    关于跨域

    加载来自不同域的脚本发生错误的时候,为了避免信息泄露,语法细节不会再上报,而是简单的 "Script error"

    解决方法是,在 script 标签中使用 crossorigin 属性并要求服务器端发送适当的 CORS HTTP 响应头,则可以解决这个问题,也就是要求服务端设置 Access-Control-Allow-Origin

    <script src="http://cdn.xxx.com/index.js" crossorigin="anonymous"></script>
    

    在 webpack 中,我们可以设置 output 中 crossOriginLoading 为 anonymous,如下所示

    output: {
        path: path.resolve(__dirname, '../dist'),
        filename: '[name].js',
        publicPath: '',
        chunkFilename: '[name].js',
        library: 'MST',
        crossOriginLoading: 'anonymous'
      }
    

    关于 sourcemap

    当我们使用 webpack 打包我们 Vue 应用的时候,最后生成的代码都是混淆过的,主要出于安全和性能两个方面的考虑。但是在我们开发阶段这样是不利于我们定位和调试问题的。所以我们可以开启 source map 模式。我们只需要配置 webpack 的 devtool 选项即可,详见webpack devtool 官网。示例所示:

    devtool: 'eval-source-map'
    

    但是 sourcemap 很好用,但是生产上我们一般不能使用 sourcemap,主要还是安全方面的考虑,如果将 sourcemap 文件发布到线上,可能会造成代码泄露、业务流失、系统被攻击等等风险。那么线上的问题,我们怎么能够知道详细的异常信息呢?

    介绍一个 sourcemap 调试线上问题的技巧
    首先本地 webpack 打包依然生成 sourcemap 文件,但是我们不上传到服务器,只保留在本地服务器。当报错时候,我们使用 whistle 拦截和线上的 js 替换成我们本地 sourcemap 文件。这样就相当于加载我们本地的 sourmap 文件了。

    关于异步的异常捕获

    为什么 try...catch...不能捕获到异步的异常?

    这个涉及到了事件循环(Event Loop)相关知识了,首先 js 是单线程的,当我们 try 中执行的代码是异步的时候,当异步执行报错时候,可能同步代码已经从执行栈中取出并执行完毕了,所以没有办法捕获到异步的异常

    那我们应该如何捕获异步的异常呢?

    • 通过 Promise 的 catch 可以捕获到异常
    // reject
    const p1 = new Promise((reslove, reject) => {
      if(1) {
        reject();
      }
    });
    p1.catch((e) => console.log('p1 error'));
    
    // throw new Error
    const p2 = new Promise((reslove, reject) => {
      if(1) {
        throw new Error('p2 error')
      }
    });
    
    p2.catch((e) => console.log('p2 error'));
    

    Promise 不管内部是 reject 还是 throw new Error,都可以通过 catch 捕获

    • 使用 async 和 await 时候捕获异常
    run();
    async function run() {
        try {
            await Promise.reject(new Error("Oops!"));
        } catch (error) {
            error.message; // "Oops!"
        }
    }
    

    参考

    GlobalEventHandlers.onerror

    JS 拦截/捕捉 全局错误 全局Error onerror

    【webpack】你所不知道的sourceMap

    JS 异步错误捕获二三事

    欢迎大家来我杂货铺逛逛,不买东西都行,我们就聊聊天,谈谈心~

    欢迎大家关注我的前端大杂货铺

    相关文章

      网友评论

        本文标题:前端异常捕获和定位

        本文链接:https://www.haomeiwen.com/subject/utsnactx.html