zoukankan      html  css  js  c++  java
  • 分享一个自定义的 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:

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

      欢迎大家留言讨论。

    版权

    作者:Artwl

    出处:http://artwl.cnblogs.com

    本文首发博客园,版权归作者跟博客园共有。转载必须保留本段声明,并在页面显著位置给出本文链接,否则保留追究法律责任的权利。

  • 相关阅读:
    【PowerDesigner】【2】将工具栏显示出来
    【PowerDesigner】【1】简单介绍
    【服务器】【Windows】【4】删除Windows系统中不想要的服务
    【Java】【8】StringUtils中isNotEmpty和isNotBlank的区别
    【Java】【7】枚举类
    hdu 1285
    Codeforces Round #198 (Div. 2) —— D
    Codeforces Round #198 (Div. 2) —— C
    Codeforces Round #198 (Div. 2) —— B
    Codeforces Round #198 (Div. 2) —— A
  • 原文地址:https://www.cnblogs.com/jiangxiaobo/p/5314197.html
Copyright © 2011-2022 走看看