分享一个自定义的console类 让你不再纠结JS中的调试代码的兼容
分享一个自定义的console类 让你不再纠结JS中的调试代码的兼容
发布时间:2016-12-30 来源:查字典编辑
摘要:问题的产生在写JS的过程中,为了调试我们常常会写很多console.log、console.info、console.group、conso...

问题的产生

在写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:

Pattern Type
%s String
%d,%i Integer (numeric formatting is not yet supported)
%f Floating point number (numeric formatting is not yet supported)
%o Object hyperlink
%c Style formatting
其中的%c比较特殊,是给输出添加样式的,比如我们在firebug中这样写:

复制代码 代码如下:

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

运行后的结果是这样的:

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

这里%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>");

}

}

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格式的图片,就是上面代码中的三个长字符串,大家用时可以换成图片地址。

完整代码:

复制代码 代码如下:

JSCode

Login

Result

JavaScript

HTML

CSS

ALL

Edit

Share

DownLoad

<!DOCTYPE html>

<html>

<head>

<meta http-equiv="Content-Type" content="text/html; charset=utf-8"/>

<title>自定义console</title>

<style type="text/css">

.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;

}

</style>

<script type="text/javascript">

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(" "));

}

}

};

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>");

}

}

if(cla){

div.className=cla;

}

var oText=document.createTextNode(val);

div.appendChild(oText);

document.getElementById("showconsole").appendChild(div);

}

}

};

window.onload=function(){

console.group("log");

console.log('%c a %s This is red text on a green background','color:red; background-color:#EEE');

console.log("The %%s jumped over %d tall buildings",1, "abc",1);

console.log("I am %s and I have:", "abc", "bcd","cde");

console.log("abc",1);

console.log([1,2,3,4,5]);

console.log();

console.log(test);

console.group("info");

console.info('%c a %s This is red text on a green background','color:red; background-color:#EEE');

console.info("The %%s jumped over %d tall buildings",1, "abc",1);

console.info("I am %s and I have:", "abc", "bcd","cde");

console.info("abc",1);

console.info([1,2,3,4,5]);

console.info();

console.info(test);

console.group("warn");

console.warn('%c a %s This is red text on a green background','color:red; background-color:#EEE');

console.warn("The %%s jumped over %d tall buildings",1, "abc",1);

console.warn("I am %s and I have:", "abc", "bcd","cde");

console.warn("abc",1);

console.warn([1,2,3,4,5]);

console.warn();

console.warn(test);

console.group("error");

console.error('%c a %s This is red text on a green background','color:red; background-color:#EEE');

console.error("The %%s jumped over %d tall buildings",1, "abc",1);

console.error("I am %s and I have:", "abc", "bcd","cde");

console.error("abc",1);

console.error([1,2,3,4,5]);

console.error(test);

}

function test(){

alert("abc");

alert("abc");

alert("abc");

alert("abc");

alert("abc");

alert("abc");

alert("abc");

alert("abc");alert("abc");

}

</script>

</head>

<body>

<h1>自定义console(Artwl.cnblogs.com)</h1>

<div id="showconsole"></div>

</body>

</html>

小结

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

欢迎大家留言讨论。

作者:Artwl

推荐文章
猜你喜欢
附近的人在看
推荐阅读
拓展阅读
相关阅读
网友关注
最新Javascript教程学习
热门Javascript教程学习
编程开发子分类