分享一个自定义的 console 类,让你不再纠结JS中的调试代码的兼容

问题的产生

  在写JS的过程中,为了调试我们常常会写很多 console.log、console.info、console.group、console.warn、console.error代码来查看JS的运行情况,但发布时又因为IE不支持console,又要去掉这些代码,一不小心就会出错。

  本文分享自己昨晚写的一个console类来试图解决这一问题。当然,更好的做法是把测试代码分开写,那样就不会有这个问题。

解决思路

  如何解决IE下不兼容的问题呢,那就是我们自己定义一个console类来覆盖浏览器提供的console功能,这样只要在页面中引用此JS文件就可以了。

  另外,此类还提供了查看输出的调试信息功能,console 定义了哪些功能呢,我们可以在这里看到:http://getfirebug.com/wiki/index.php/Console_API,我们可以看到这里提供了很多方法,我们常用的有 console.log、console.info、console.group、console.warn、console.error、console.profile、console.time,最后两个是分析代码性能的,比较复杂,本文没有实现。

代码解析

  第一步,当然是搭一个结构,覆盖浏览器(firebug、chrome)提供的console功能,这样直接引用此JS文件即可保证浏览器(主要是IE)中不出错:

console
var console={
assert:function(){
},
clear:function(){
},
count:function(){
},
debug:function(){
},
dir:function(){
},
dirxml:function(){
},
error:function(){
},
exception:function(){
},
group:function(name){
},
groupCollapsed:function(){
},
groupEnd:function(){
},
info:function(){
},
log:function(){
},
memoryProfile:function(){
},
memoryProfileEnd:function(){
},
profile:function(){
},
profileEnd:function(){
},
table:function(){
},
time:function(){
},
timeEnd:function(){
},
timeStamp:function(){
},
trace:function(){
},
warn:function(){
}
};

  第二步,实现 console.log方法。在所实现的几个方法中这个是最复杂的。

  从firebug的API中我们可以看到,console.log不仅仅可以输出信息,还提供了类似 string.Format的功能,直接引用原文如下:

  Here is the complete set of patterns that you may use for string substitution:

PatternType
 %sString
 %d, %iInteger (numeric formatting is not yet supported)
 %fFloating point number (numeric formatting is not yet supported)
 %oObject hyperlink
 %cStyle formatting

 

  其中的%c比较特殊,是给输出添加样式的,比如我们在firebug中这样写:

console.log('%cTest output', 'color:white; background-color:blue');

  运行后的结果是这样的:

  这里%c也可以跟 %s、%d等混用。

  所以,在代码中我直接用replace进行替换,由于JS中的replace默认只替换第一个匹配项,这里刚好,代码如下:

    var args=Array.prototype.slice.call(arguments);if(args.length>1){var i=1,hasstyle=false;if(args[0].indexOf("%c")==0){args[0]=args[0].replace(/%c/,"");i=2;hasstyle=true;}for(;i<args.length;i++){if(/%s|%d|%i|%o/.test(args[0])){args[0]=args[0].replace(/%s|%d|%i|%o/,args[i]);}else{break;}}if(i<args.length){args[0]=args[0]+" "+args.slice(i).join(" ");}if(hasstyle){consoleHelper.showlog(args[0],args[1]);}else{consoleHelper.showlog(args[0]);}}else if(args.length==1){if(arguments[0] instanceof Array){consoleHelper.showlog("["+args[0]+"]");}else if(arguments[0] instanceof Function){consoleHelper.showlog(args[0],null,"console_log_function");}else{consoleHelper.showlog(args[0]);}}else{consoleHelper.showlog("");}

  由于console.log可以接受多个参数,且个数不确定,所以这里直接没有写形参。对于%c虽然firebug中写在中间也是有效的,这里为了简单直接只对写在开头的有效。代码中先把参数转换为数组,然后对数组进行分情况处理。

  当参数个数大于1时,对后面的参数用replace进行替换,然后把剩下的参数连接(join)起来进行输出。

  当参数个数为1时,还要分两种情况,一是数组,二是方法。对于数组,按firebug中的格式,在两端加中括号,对于函数,把字的颜色变为绿色

  当参数个数为0时,直接输出空字符串

  后面的consoleHelper.showlog是为了输出方便另外写的一个方法,在这个方法中把各种调试信息的结果显示在页面上的一个div(如果存在)中。

  其他几个方法的思路跟这个差不多,只是样式不同,功能比这个简单,直接把参数连接起来输出即可。

  整个console类代码如下:

console全部代码
var console={
assert:function(){
},
clear:function(){
},
count:function(){
},
debug:function(){
},
dir:function(){
},
dirxml:function(){
},
error:function(){var args=Array.prototype.slice.call(arguments);consoleHelper.showerror(args.join(" "));
},
exception:function(){
},
group:function(name){consoleHelper.showgroup(name);
},
groupCollapsed:function(){
},
groupEnd:function(){
},
info:function(){var args=Array.prototype.slice.call(arguments);if(args.length==1){if(arguments[0] instanceof Array){consoleHelper.showinfo("["+args[0]+"]");}else if(arguments[0] instanceof Function){consoleHelper.showinfo(args[0],"console_log_function");}else{consoleHelper.showinfo(args[0]);}}else{consoleHelper.showinfo(args.join(" "));}
},
log:function(){var args=Array.prototype.slice.call(arguments);if(args.length>1){var i=1,hasstyle=false;if(args[0].indexOf("%c")==0){args[0]=args[0].replace(/%c/,"");i=2;hasstyle=true;}for(;i<args.length;i++){if(/%s|%d|%i|%o/.test(args[0])){args[0]=args[0].replace(/%s|%d|%i|%o/,args[i]);}else{break;}}if(i<args.length){args[0]=args[0]+" "+args.slice(i).join(" ");}if(hasstyle){consoleHelper.showlog(args[0],args[1]);}else{consoleHelper.showlog(args[0]);}}else if(args.length==1){if(arguments[0] instanceof Array){consoleHelper.showlog("["+args[0]+"]");}else if(arguments[0] instanceof Function){consoleHelper.showlog(args[0],null,"console_log_function");}else{consoleHelper.showlog(args[0]);}}else{consoleHelper.showlog("");}
},
memoryProfile:function(){
},
memoryProfileEnd:function(){
},
profile:function(){
},
profileEnd:function(){
},
table:function(){
},
time:function(){
},
timeEnd:function(){
},
timeStamp:function(){
},
trace:function(){
},
warn:function(){var args=Array.prototype.slice.call(arguments);if(args.length==1){if(arguments[0] instanceof Array){consoleHelper.showwarn("["+args[0]+"]");}else if(arguments[0] instanceof Function){consoleHelper.showwarn(args[0],"console_log_function");}else{consoleHelper.showwarn(args[0]);}}else{consoleHelper.showwarn(args.join(" "));}
}
};

  consoleHelper代码如下:

var consoleHelper={
showlog:function(val,style,cla){if(cla){cla="console_log "+cla;}else{cla="console_log";}this.show(val,style,cla);
},
showinfo:function(val,cla){if(cla){cla="console_info "+cla;}else{cla="console_info";}this.show(val,null,cla);
},            
showwarn:function(val,cla){if(cla){cla="console_warn "+cla;}else{cla="console_warn";}this.show(val,null,cla);
},
showerror:function(val){this.show(val,null,"console_error");
},
showgroup:function(val){if(!val){val="";}this.show(val+":",null,"console_group");
},
show:function(val,style,cla){if(document.getElementById("showconsole")){var div=document.createElement("div");if(div.setAttribute){if(style){div.setAttribute("style",style);}}else{if(style){div=document.createElement("<div style="+style+">");}}if(cla){div.className=cla;}var oText=document.createTextNode(val);div.appendChild(oText);document.getElementById("showconsole").appendChild(div);}
}
};

  注:如果想在页面中看到调试信息,直接在页面上添加一个id 为 showconsole 的隐藏的div即可。

  样式(尽量跟FireBug保持一致):

.console_log{border:1px solid #CCC;color:#333;padding:0px 5px;min-height:24px;line-height:24px;margin-bottom:-1px;
}
.console_info{border:1px solid #CCC;color:#333;padding:0px 5px;min-height:24px;line-height:24px;margin-bottom:-1px;background: url("") no-repeat scroll 0 1px #EBF5FF;padding-left:30px;
}
.console_warn{border:1px solid #CCC;color:#333;padding:0px 5px;min-height:24px;line-height:24px;margin-bottom:-1px;background: url("") no-repeat scroll 0 1px #FFFFC8;padding-left:30px;
}
.console_error{border:1px solid #CCC;color:#FF0000;padding:0px 5px;min-height:24px;line-height:24px;margin-bottom:-1px;background: url("") no-repeat scroll 0 1px #FFEBEB;padding-left:30px;
}
.console_group{margin-top:20px;font-size:16px;font-weight:bolder;
}
.console_log_function{color:green;
}

  这里为了演示方便,三个小图标直接用的是base64格式的图片,就是上面代码中的三个长字符串,大家用时可以换成图片地址。

小结

  写这个JS一方面是工作中有这方面的需求,另外也是因为在博问中看到有人问 JavaScript中如何获得console.log的值? ,前段时间有个国外学编程网站可以把console.log的结果直接显示在页面上,不知道是不是用了本文类似的方案。

  欢迎大家留言讨论。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/404186.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

OS X 平台的 8 个终端实用工具

From: http://blog.jobbole.com/72238/ 本文由 伯乐在线 - shinancao 翻译自 mitchchn。欢迎加入iOS小组。转载请参见文章末尾处的要求。 OS X 终端对外开放了许多很强大的UNIX实用工具和脚本。如果你是从Linux转过来的&#xff0c;你会发现许多熟悉的命令正是按照你想的那样…

内存操作流

可以将输出的位置设置在内存上&#xff0c;此时就要使用ByteArrayInputStream、ByteArrayOutputStream来完成输入和输出功能。 ByteArrayInputStream主要完成将内容写入到内存中 ByteArrayOutputStream的功能主要是将内存中的数据输出 import java.io.ByteArrayInputStream; im…

mysql 中 group_concat()用法

基本语法&#xff1a;group_concat([DISTINCT] 要连接的字段 [Order BY 排序字段 ASC/DESC] [Separator 分隔符]) 初始数据&#xff1a; 以name分组&#xff0c;将money打印在一行&#xff0c;默认,分隔&#xff1a; select aa.namename , GROUP_CONCAT( money ) money from a…

H264(NAL简介与I帧判断)

From: http://blog.csdn.net/jefry_xdz/article/details/8461343 1、NAL全称Network Abstract Layer, 即网络抽象层。 在H.264/AVC视频编码标准中&#xff0c;整个系统框架被分为了两个层面&#xff1a;视频编码层面&#xff08;VCL&#xff09;和网络抽象层面&#…

让VS2010添加新类时自动添加public关键字(来自dudu博文)

以下是dudu 老大的详细步骤&#xff1a; http://www.cnblogs.com/dudu/archive/2011/08/30/2159894.html 里面有一个重要步骤&#xff0c;如何修改模板class.cs文件&#xff0c;请看下面截图 1、下载一个解压软件7-zip_9.25Alpha&#xff08;本人是win7 64位&#xff0c;不支持…

H.264视频编码在VC++.Net中的实现

From: http://blog.csdn.net/xwchen/article/details/5052981 引言&#xff1a;H.264编码技术是俱乐部在过去一段时间内研究的一个方向,对该编码技术进行过实际的开发和应用&#xff0c;并取得了很大的收获。下面将重点介绍H.264视频编码在VC.Net中的实现。 1. H.264编码的介…

TCP/IP ---封装与分用

封装 当应用程序用T C P传送数据时&#xff0c;数据被送入协议栈中&#xff0c;然后逐个通过每一层直到被当作一串比特流送入网络。其中每一层对收到的数据都要增加一些首部信息&#xff08;有时还要增加尾部信息&#xff09;&#xff0c;该过程如图1 - 7所示。T C P传给I P的数…

远程计算机需要网络级别身份验证,而您的计算机不支持该验证,请联系您的系统管理员或者技术人员来获得帮助...

故障&#xff1a;“远程计算机需要网络级别身份验证&#xff0c;而您的计算机不支持该验证&#xff0c;请联系您的系统管理员或者技术人员来获得帮助” 故障症状&#xff1a;当您使用Windows XP“远程桌面连接”工具去连接Windows Vistas或Windows Server 2008的远程桌面、终端…

CRC32算法详细推导(1)

From: http://blog.csdn.net/sparkliang/article/details/5671510 CRC算法详解&#xff08;1&#xff09; 作为blog再次发出来&#xff0c;详细描述一下CRC32算法的推导过程。 CRC 算法的数学基础 CRC 算法的数学基础就不再多啰嗦了&#xff0c;到处都是&#xff0c;简单提一…

CRC32算法详细推导(2)

From: http://blog.csdn.net/sparkliang/article/details/5671977 CRC算法详解&#xff08;2&#xff09; 初见 Table-Driven 变换到上面的方法后&#xff0c;我们离 table-driven 的方法只有一步之遥了&#xff0c;我们知道一个字节能表示的正整数范围是 0~255&#xff0c;步…

每天一点Swift(五)控制器的生命周期和SizeClass

字数358 阅读19 评论0 喜欢0 初始化init-->awakeFromNib--> prepare a segue --> SB去设置outlets --> viewDidLoad 1. viewDidLoad 在viewDidLoad中&#xff0c;outlets已经被设置&#xff1b;但是几何位置&#xff08;bounds&#xff09;并没有被设置 viewDidLoa…

h264 I帧的判断

From:http://blog.csdn.net/dxpqxb/article/details/13289205 H264数据的NALU 头的格式如图2 所示&#xff1a; F&#xff1a;forbidden_zero_bit.1 位&#xff0c;如果有语法冲突&#xff0c;则为 1。当网络识别此单元存在比特错误时&#xff0c;可将其设为 1&#xff0c;以便…

单硬盘上mac + win7双系统,GUID-GPT分区

首先&#xff0c;当然要介绍下我的环境了。 一、环境介绍 1. 主板是支持UEFI启动的&#xff0c;但是我感觉自己没用到。 2. 我有两个硬盘: hd0: 准备安装mac和win7_x64 hd1: GUID格式&#xff0c;GPT分区&#xff0c;已装有mac10.9.3环境&#xff0c;还有个FAT32的分区&a…

将clover安装到硬盘EFI分区, 解决root device uuid is ...问题

其实说白了&#xff0c;很简单&#xff0c;照着配置图来&#xff0c;一招搞定&#xff0c;但是当你不知道这些配置的时候&#xff0c;将可能耗费你好几天的时间&#xff0c;而且还不一定得到最优解。 附“Clover v2k r2703.pkg"下载地址: http://download.csdn.net/deta…

Silverlight 5 新特性

微软Silverlight 团队在12月9日早上低调的发布了Silverlight 第五个版本-Silverlight 5. Silverlight 5 Available is Here 这也可能是Silverlight最后一个官方主要版本. Silverlight 5 可能作为官方发布最后一个Silverlight 版本. 官方提供各方面支持直到2021年. 支持的时间相…

VMware10出现VMware Workstation 不可恢复错误: (vmx)

在我的VMware10中安装了Ubuntu12&#xff0c; 昨晚还正常关机的呢&#xff0c;今天早上一打开&#xff0c;竟然报错了&#xff0c;错误如图&#xff1a; 上网搜索了下&#xff0c;没有找到比较合适的方法&#xff0c;最后&#xff0c;我在没有卸载VMware的情况下&#xff0c;重…

DB2中ixf文件的导入导出

1. 导出数据 语法&#xff1a;EXPORT TO <文件路径>/文件名.IXF OF IXF SELECT * FROM 表名 2. 导入数据 语法&#xff1a;db2 IMPORT FROM <路径>/<文件名>.IXF OF IXF MODIFIED BY FORCECREATE COMMITCOUNT 10000 REPLACE_CREATE INTO <表名> 简单语…