分享一個(gè)自定義的 console 類,讓你不再糾結(jié)JS中的調(diào)試代碼的兼容
問題的產(chǎn)生
在寫JS的過程中,為了調(diào)試我們常常會(huì)寫很多 console.log、console.info、console.group、console.warn、console.error代碼來查看JS的運(yùn)行情況,但發(fā)布時(shí)又因?yàn)镮E不支持console,又要去掉這些代碼,一不小心就會(huì)出錯(cuò)。
本文分享自己昨晚寫的一個(gè)console類來試圖解決這一問題。當(dāng)然,更好的做法是把測(cè)試代碼分開寫,那樣就不會(huì)有這個(gè)問題。
解決思路
如何解決IE下不兼容的問題呢,那就是我們自己定義一個(gè)console類來覆蓋瀏覽器提供的console功能,這樣只要在頁面中引用此JS文件就可以了。
另外,此類還提供了查看輸出的調(diào)試信息功能,console 定義了哪些功能呢,我們可以在這里看到:http://getfirebug.com/wiki/index.php/Console_API,我們可以看到這里提供了很多方法,我們常用的有 console.log、console.info、console.group、console.warn、console.error、console.profile、console.time,最后兩個(gè)是分析代碼性能的,比較復(fù)雜,本文沒有實(shí)現(xiàn)。
代碼解析
第一步,當(dāng)然是搭一個(gè)結(jié)構(gòu),覆蓋瀏覽器(firebug、chrome)提供的console功能,這樣直接引用此JS文件即可保證瀏覽器(主要是IE)中不出錯(cuò):
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(){ } };
第二步,實(shí)現(xiàn) console.log方法。在所實(shí)現(xiàn)的幾個(gè)方法中這個(gè)是最復(fù)雜的。
從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比較特殊,是給輸出添加樣式的,比如我們?cè)趂irebug中這樣寫:
console.log('%cTest output', 'color:white; background-color:blue');
運(yùn)行后的結(jié)果是這樣的:

這里%c也可以跟 %s、%d等混用。
所以,在代碼中我直接用replace進(jìn)行替換,由于JS中的replace默認(rèn)只替換第一個(gè)匹配項(xiàng),這里剛好,代碼如下:
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可以接受多個(gè)參數(shù),且個(gè)數(shù)不確定,所以這里直接沒有寫形參。對(duì)于%c雖然firebug中寫在中間也是有效的,這里為了簡(jiǎn)單直接只對(duì)寫在開頭的有效。代碼中先把參數(shù)轉(zhuǎn)換為數(shù)組,然后對(duì)數(shù)組進(jìn)行分情況處理。
當(dāng)參數(shù)個(gè)數(shù)大于1時(shí),對(duì)后面的參數(shù)用replace進(jìn)行替換,然后把剩下的參數(shù)連接(join)起來進(jìn)行輸出。
當(dāng)參數(shù)個(gè)數(shù)為1時(shí),還要分兩種情況,一是數(shù)組,二是方法。對(duì)于數(shù)組,按firebug中的格式,在兩端加中括號(hào),對(duì)于函數(shù),把字的顏色變?yōu)榫G色
當(dāng)參數(shù)個(gè)數(shù)為0時(shí),直接輸出空字符串
后面的consoleHelper.showlog是為了輸出方便另外寫的一個(gè)方法,在這個(gè)方法中把各種調(diào)試信息的結(jié)果顯示在頁面上的一個(gè)div(如果存在)中。
其他幾個(gè)方法的思路跟這個(gè)差不多,只是樣式不同,功能比這個(gè)簡(jiǎn)單,直接把參數(shù)連接起來輸出即可。
整個(gè)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); } } };
注:如果想在頁面中看到調(diào)試信息,直接在頁面上添加一個(gè)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; }
這里為了演示方便,三個(gè)小圖標(biāo)直接用的是base64格式的圖片,就是上面代碼中的三個(gè)長(zhǎng)字符串,大家用時(shí)可以換成圖片地址。
小結(jié)
寫這個(gè)JS一方面是工作中有這方面的需求,另外也是因?yàn)樵诓﹩栔锌吹接腥藛?nbsp;JavaScript中如何獲得console.log的值? ,前段時(shí)間有個(gè)國(guó)外學(xué)編程網(wǎng)站可以把console.log的結(jié)果直接顯示在頁面上,不知道是不是用了本文類似的方案。
歡迎大家留言討論。

作者:Artwl
本文首發(fā)博客園,版權(quán)歸作者跟博客園共有。轉(zhuǎn)載必須保留本段聲明,并在頁面顯著位置給出本文鏈接,否則保留追究法律責(zé)任的權(quán)利。

浙公網(wǎng)安備 33010602011771號(hào)