Java自学者论坛

 找回密码
 立即注册

手机号码,快捷登录

恭喜Java自学者论坛(https://www.javazxz.com)已经为数万Java学习者服务超过8年了!积累会员资料超过10000G+
成为本站VIP会员,下载本站10000G+会员资源,会员资料板块,购买链接:点击进入购买VIP会员

JAVA高级面试进阶训练营视频教程

Java架构师系统进阶VIP课程

分布式高可用全栈开发微服务教程Go语言视频零基础入门到精通Java架构师3期(课件+源码)
Java开发全终端实战租房项目视频教程SpringBoot2.X入门到高级使用教程大数据培训第六期全套视频教程深度学习(CNN RNN GAN)算法原理Java亿级流量电商系统视频教程
互联网架构师视频教程年薪50万Spark2.0从入门到精通年薪50万!人工智能学习路线教程年薪50万大数据入门到精通学习路线年薪50万机器学习入门到精通教程
仿小米商城类app和小程序视频教程深度学习数据分析基础到实战最新黑马javaEE2.1就业课程从 0到JVM实战高手教程MySQL入门到精通教程
查看: 430|回复: 0

如何做好前端页面异常监控?

[复制链接]
  • TA的每日心情
    奋斗
    2024-11-24 15:47
  • 签到天数: 804 天

    [LV.10]以坛为家III

    2053

    主题

    2111

    帖子

    72万

    积分

    管理员

    Rank: 9Rank: 9Rank: 9

    积分
    726782
    发表于 2021-4-23 00:02:23 | 显示全部楼层 |阅读模式

    在开发工作中难免会出现bug,一般项目都是测试检查通过后就可以发线上,可是在线上仍旧会出现各种意料之外或者未测试到的问题,这个时候有的用户会向客服反馈说哪里哪里有问题,这是一种被动的错误上报方式,毕竟不是所有的用户都会上报问题,更多的则是出现问题后直接离开我们的APP。所以异常监控这块就显得越来越重要。

    页面异常分类

    在我们的项目中我将页面异常分为以下几种情况:

    1. javascript异常(语法错误,运行时错误,跨域脚本)
    2. 资源加载异常(img js css)
    3. ajax请求异常
    4. promise异常
    5. vue项目中全局异常捕获

    接下来介绍如何捕获这些异常

    前端页面异常捕获方式

    window.onerror捕获javascript异常

    /**
    * 捕获javascript异常
    * @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 });
    }
    

    跨域脚本异常捕获

    一般涉及跨域的js运行错误时会抛出错误提示script error.,但没有具体信息(如出错文件,行列号提示等), 可利用资源共享策略来捕获跨域js错误
    客户端:在script标签增加crossorigin="anonymous"属性
    服务端:静态资源响应头Access-Control-Allow-Origin: *

    window.addEventListener('error',cb,true)捕获资源加载异常

    img加载异常时会触发img.onerror函数

    // 捕获资源加载异常
    window.addEventListener('error',function(e){
        const err = e.target.src || e.target.href
        if(err){
            console.log('捕获到资源加载异常',err)
        }
    },true)
    

    ajax接口请求异常捕获

    // 统一拦截ajax请求
    function ajaxEventTrigger (event) {
        var ajaxEvent = new CustomEvent(event, { detail: this })
        window.dispatchEvent(ajaxEvent)
    }
    
    var oldXHR = window.XMLHttpRequest
    function newXHR () {
        var realXHR = new oldXHR()
        realXHR.addEventListener('readystatechange', function () { ajaxEventTrigger.call(this, 'ajaxReadyStateChange') }, false)
        return realXHR
    }
    window.XMLHttpRequest = newXHR
    var startTime = 0
    var gapTime = 0 // 计算请求延时
    window.addEventListener('ajaxReadyStateChange', function (e) {
        var xhr = e.detail
        var status = xhr.status
        var readyState = xhr.readyState
        /**
         * 计算请求延时
         */
        if (readyState === 1) {
            startTime = (new Date()).getTime()
        }
        if (readyState === 4) {
            gapTime = (new Date()).getTime() - startTime
        }
        /**
         * 上报请求信息
         */
         if (readyState === 4) {
            if(status === 200){
                // 接口正常响应时捕获接口响应耗时
                console.log('接口',xhr.responseURL,'耗时',gapTime)
             }else{
                // 接口异常时捕获异常接口及状态码
                console.log('异常接口',xhr.responseURL,'状态码',status)
            }
         }
    })
    

    promise异常捕获

    promise 中的报错顺序是:
    如果有catch 等捕获函数,则走catch 捕获函数。catch 捕获函数如果没有抛出新的异常,则下一个then将会认为没有什么报错,会继续执行。
    如果没有catch 等捕获函数,我们需要注册 window.addEventListener('unhandledrejection') 来处理

    /**
     * Promise catch错误上报,需要在使用promise的地方显示调用.catch(),否则不会捕获错误
     */
    if (typeof Promise !== 'undefined') {
        var _promiseCatch = Promise.prototype.catch
        Promise.prototype.catch = function (foo) {
            return _promiseCatch.call(this, catCatch(foo))
        }
    }
    function catCatch (foo) {
        return function (args) {
            let msg = args.stack ? args.stack : args
            console.log('捕获到catch中的异常',msg)
            foo && foo.call(this, args)
        }
    }
    
    /**
     * 监听promise未处理的reject错误, 跨域情况下监控不到
    */
    window.addEventListener('unhandledrejection', event => {
        console.log('捕获到未处理的promise异常',event.reson)
    })
    

    vue项目全局异常捕获

    Vue.config.errorHandler = function (err, vm, info) {
        // `info` 是 Vue 特定的错误信息,比如错误所在的生命周期钩子
        // 只在 2.2.0+ 可用
        let msg = `错误发生在:${info}中,具体信息:${err.stack}`
        console.log(msg)
    }
    

    捕获到这些异常后我们需要将这些异常上报给服务器,我们直接以请求图片的形式发送上报内容

    异常上报

    function report (msg) {
        var reportUrl = 'http://xxxx/report'
        new Image().src = reportUrl + encodeURIComponent(JSON.stringify(msg))
    }
    
    哎...今天够累的,签到来了1...
    回复

    使用道具 举报

    您需要登录后才可以回帖 登录 | 立即注册

    本版积分规则

    QQ|手机版|小黑屋|Java自学者论坛 ( 声明:本站文章及资料整理自互联网,用于Java自学者交流学习使用,对资料版权不负任何法律责任,若有侵权请及时联系客服屏蔽删除 )

    GMT+8, 2025-1-24 08:39 , Processed in 0.060043 second(s), 30 queries .

    Powered by Discuz! X3.4

    Copyright © 2001-2021, Tencent Cloud.

    快速回复 返回顶部 返回列表