for(var i=0;i<l;i++){ console.log(i);}
那么我们如何来捕获这种异常呢,有两种方法,
第一种是try..catch
第二种是 window.onerror
由于try.catch 没法捕捉到全局的错误事件,也即是说 只有try,catch的块里边运行出错才会被你捕捉到。所以我们这里排除它的这种方案,
来采用第二种方法,也就是window.onerror方法。
window.onerror
打开浏览器自带的开发者工具,当一个错误发生时,我们可以立刻得到提示,并且知道错误发生的位置以及调用的堆栈信息。
我们可以通过 window.onerror 来捕获页面上的各种脚本执行异常,它能帮助我们获取有用的信息。但是这个方法存在兼容性问题,在不同的浏览器上提供的数据不完全一致,
部分过时的浏览器只能提供部分数据。它的用法如下:window.onerror = function (message, url, lineNo, columnNo, error)
五个参数的含义如下:
1、message {String} 错误信息。直观的错误描述信息,不过有时候你确实无法从这里面看出端倪,特别是压缩后脚本的报错信息,可能让你更加疑惑。
2、url {String} 发生错误对应的脚本路径,比如是你的http://a.js报错了还是http://b.js报错了。
3、lineNo {Number} 错误发生的行号。
4、columnNo {Number} 错误发生的列号。
5、error {Object} 具体的 error 对象,包含更加详细的错误调用堆栈信息,这对于定位错误非常有帮助。
兼容性问题
不同浏览器对同一个错误的 message 是不一样的。
IE10以下浏览器只能获取到 message,url 和 lineNo这三个参数,获取不到columnNo 和 error
不过 window.event 对象提供了 errorLine 和 errorCharacter,以此来对应相应的行列号信息。
在使用onerror的时候,我们可以使用arguments.callee.caller 来递归出调用堆栈,这一类信息是最直接的错误信息信息,所以是必须要捕获并上报的。后面我们会用js去示范。
不同浏览器默认可获取的参数值:
写一个js报错的上报库
既然知道了window.onerror的用法,为啥我们不来写一个js库来监控我们的前端js,废话少说,写之。
实现思路:
1、收集window.onerror的五个参数
2、除了那五个参数,可以增加自定义参数
3、发送到后台服务器
我们暂且给我们的库起名为 badJsReport
原理比较简单,代码如下:
/** * Name: badJsReport.js * Version 1.1.0 * Author xianyulaodi * Address: https://github.com/xianyulaodi/badJsReport * Released on: December 22, 2016 */;(function(){ "use strict"; if (window.badJsReport){ return window.badJsReport}; /* * 默认上报的错误信息 */var defaults = {msg:"", //错误的具体信息url:"", //错误所在的urlline:"", //错误所在的行col:"", //错误所在的列error:"", //具体的error对象 }; /* *ajax封装 */ function ajax(options) {options = options || {};options.type = (options.type || "GET").toUpperCase();options.dataType = options.dataType || "json";var params = formatParams(options.data);if (window.XMLHttpRequest) { var xhr = new XMLHttpRequest();} else {var xhr = new ActiveXObject("Microsoft.XMLHTTP");}xhr.onreadystatechange = function () { if (xhr.readyState == 4) {var status = xhr.status;if (status >= 200 && status < 300) { options.success && options.success(xhr.responseText, xhr.responseXML);} else { options.fail && options.fail(status);} }}if (options.type == "GET") { xhr.open("GET", options.url + "?" + params, true); xhr.send(null);} else if (options.type == "POST") { xhr.open("POST", options.url, true); //设置表单提交时的内容类型 xhr.setRequestHeader("Content-Type", "application/x-www-form-urlencoded"); xhr.send(params);} } /* *格式化参数 */ function formatParams(data) {var arr = [];for (var name in data) { arr.push(encodeURIComponent(name) + "=" + encodeURIComponent(data[name]));}arr.push(("v=" + Math.random()).replace(".",""));return arr.join("&"); } /* * 合并对象,将配置的参数也一并上报 */ function cloneObj(oldObj) { //复制对象方法if (typeof(oldObj) != "object") return oldObj;if (oldObj == null) return oldObj;var newObj = new Object();for (var prop in oldObj)newObj[prop] = oldObj[prop];return newObj; }; function extendObj() { //扩展对象var args = arguments;if (args.length < 2) {return;}var temp = cloneObj(args[0]); //调用复制对象方法for (var n = 1,len=args.length; n <len; n++){for (var index in args[n]) { temp[index] = args[n][index];}}return temp; } /** * 核心代码区 **/ var badJsReport=function(params){if(!params.url){return}window.onerror = function(msg,url,line,col,error){ //采用异步的方式,避免阻塞 setTimeout(function(){//不一定所有浏览器都支持col参数,如果不支持就用window.event来兼容col = col || (window.event && window.event.errorCharacter) || 0;defaults.url = url;defaults.line = line;defaults.col = col;if (error && error.stack){ //如果浏览器有堆栈信息,直接使用 defaults.msg = error.stack.toString();}else if (arguments.callee){ //尝试通过callee拿堆栈信息 var ext = []; var fn = arguments.callee.caller; var floor = 3; //这里只拿三层堆栈信息 while (fn && (--floor>0)) {ext.push(fn.toString());if (fn === fn.caller) { break;//如果有环}fn = fn.caller; } ext = ext.join(","); defaults.msg = error.stack.toString();}// 合并上报的数据,包括默认上报的数据和自定义上报的数据var reportData=extendObj(params.data || {},defaults);// 把错误信息发送给后台ajax({ url: params.url,//请求地址 type: "POST", //请求方式 data: reportData,//请求参数 dataType: "json", success: function (response, xml) {// 此处放成功后执行的代码params.successCallBack&¶ms.successCallBack(response, xml); }, fail: function (status) {// 此处放失败后执行的代码params.failCallBack&¶ms.failCallBack(status); } }); },0); return true; //错误不会console浏览器上,如需要,可将这样注释}; } window.badJsReport=badJsReport;})();/*===========================badJsReport AMD Export===========================*/if (typeof(module) !== "undefined"){ module.exports = window.badJsReport;}else if (typeof define === "function" && define.amd) { define([], function () {"use strict";return window.badJsReport; });}我们封装了原生ajax,还有将上报的参数对象合并。并暴露了一个全局方法 badJsReport
badJsReport({ url:"http://www.baidu.com", //发送到后台的url *必须})3、如果需要新增上报参数,或者要知道发送给后台的回调。可以用下面的方法
badJsReport({ url:"http://www.baidu.com", //发送到后台的url *必须 data:{}, //自定义添加上报参数,比如app版本,浏览器版本 -可省略 successCallBack:function(response, xml){// 发送给后台成功的回调,-可省略 }, failCallBack:function(error){// 发送给后台失败的回调,-可省略 }})注意点: