分享一个自定义的 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)中不出错:

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

  运行后的结果是这样的:

  这里%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类代码如下:

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的结果直接显示在页面上,不知道是不是用了本文类似的方案。

  欢迎大家留言讨论。

时间: 2024-12-20 19:05:01

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

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

分享一个自定义的console类 让你不再纠结JS中的调试代码的兼容 在写JS的过程中,为了调试我们常常会 写很多 console.log.console.info.console.group.console.warn.console.error代码来查看JS 的运行情况,但发布时又因为IE不支持console,又要去掉这些代码,一不小心就会出错 问题的产生 在写JS的过程中,为了调试我们常常会写很多 console.log.console.info.console.group.console.

分享一个自定义打印套打方案(一),概述

最近项目中需要实现单据套打的功能,实现后,留此日记以备忘,同时为有类似需求的同学提供一种解决方案. 原始需求: 1. 打印模板支持灵活自定义,支持可视化设置(含设置打印项是否打印,及位置,大小,字体等…). 2. 支持自定义打印项. 3. 常规的打印项设置(如自动打印小计/合计,表头表尾等…) 4. 在设置模板的时候,支持实时预览. 首先,先上个图, 给大家看一下打印套打模板的大致效果图,各位同学看着顺眼再往下看. 接下来,我们简单的来讨论一下如何实现这个需求,简单分析一下,在这个功能中,可能需

分享一个PHP数据库分页类

本帖最后由 luenmicro 于 2014-11-12 23:19 编辑 分享一个PHP数据库分页类. [code]<?php    class page    {        private $pagesize;        private $lastpage;        private $totalpages;        private $nums;        private $numPage=1; function __construct($page_size,$tota

分享一个自定义打印套打方案(二),扩展Panel,以支持鼠标随意拖动

接上一章节,本篇主要介绍一种支持鼠标随意拖动Panel内部控件位置的方法.为了简单起见,这里我们不妨就暂称我们将要扩展的Panel容器名称为 MoveControlPanel,该容器至少需要实现以下功能 1. 识别当前鼠标位置,是否处于某个内部组件的边框位置,以让鼠标显示出对应的图标(拉伸,移动-) 2.移动鼠标以改变内部某个组件的大小及坐标, 3.保存容器内每个组件的当前坐标及大小. 为了便于描述当前光标状态,我们不妨定义一个枚举.姑且称其为 EMousePointPosition, 其至少应

Android 分享一个SharedPreferences的工具类,方便保存数据

我们平常保存一些数据,都会用到SharedPreferences,他是保存在手机里面的,具体路径是data/data/你的包名/shared_prefs/保存的文件名.xml, SharedPreferences的使用也很简单,我自己就写了一个SharedPreferences的工具类,然后就保存在这里,等自己以后需要保存数据直接从这里copy代码,哈哈 工具类如下 [java] view plaincopy package com.example.shortcut; import androi

创建一个自定义的Application类

由于每个应用程序必须创建一个Application对象,vs为开发人员提供了模板来减轻开发人员的重复工作.当使用vs创建一个WPF应用程序是,vs会自动创建一个app.xaml文件, <Application x:Class="WpfApplication8.App" xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x="http://schemas.mi

福利到~分享一个基于jquery的智能提示控件intellSeach.js

一.需求 我们经常会遇到[站内搜索]的需求,为了提高用户体验,我们希望能做到像百度那样的即时智能提示.例如:某公司人事管理系统,想搜索李XX,只要输入“李”,系统自然会提示一些姓李的员工,这样方便用户使用.说白了,就是用户边输入,系统会提示相关的结果:或者,当用户点击搜索框时,就推荐一些内容,如360.百度都会提示今天的主要新闻或搜索量大的内容. jquery 已经有一个这样的插件了,叫 autocomplete, 但我觉得不好用.关于autocomplete的介绍也很多,有兴趣的朋友可以去试试

我也分享一个c# ini操作类

刚刚看了一篇 @云菲菲 的关于基于正则的INI辅助类文章:http://www.cnblogs.com/yunfeifei/p/4081977.html,作者写的不错.还看到评论处有一个的地址:https://devlib.codeplex.com/SourceControl/latest#main/product/Codes/DevLib.Configuration/IniEntry.cs,也是基于ini的一些便捷性封装类.后者显然比前者场合通用性强一些,前者则是简单易用,很方便.但是总体来说

[转帖] 分享一个java正则验证类

原址:http://blog.csdn.net/jarvis_java/article/details/5949096 package com.tool.util; import java.util.regex.Matcher; import java.util.regex.Pattern; /** * * @author Jarvis * 90%的验证都调用了Regular方法 但是本类也可删除大部分方法 涉及到正则的判断都直接穿参数和正则表达式 * 但是为了方便业务类调用和有更直观的含义 建