JS装逼实用技巧

作者: 绰号陆拾柒 | 来源:发表于2017-10-28 02:16 被阅读562次

    评价一段代码的好坏,往往是看代码的几个点:1、简洁易读,优雅的代码,读起来赏心悦目,更易于让人理解。2、执行效率,高效的执行速度,让代码的性能更省时间,省内存。3、注释,每个人的思维模式都不同,如果适当加一些注释,可以使你的代码及编程思维,更容易让人接受。
    通过一些工作积累,我在项目开发中,也意识到写代码,不是做完就了事,还要重新阅读自己写的代码,并且考虑是否可以进行优化,这样,才是健壮的代码。如果普通的一些用法及语法,这里不再多说,现只把一些工作中逼格比较高的实用经验分享出来,感兴趣的同学,可以一观。让我们来见识到JS强大的表现力吧!

    操作符的妙用

    利用按位与“&”判断奇偶数

    按位与运算的逻辑是这样: 0001 & 0011 = 0001,也就是两个位都是1,才是1,其它位都是0。
    我们经常要做一个条件,判断一个数的奇偶性,会这样写:

    function fn(n) {
        if(n % 2 === 1) {
            //奇数
        } else {
            //偶数
        }
    }
    

    利用按位与运算,则可以简写为这样:

    function fn(n) {
        if(n & 1) {
            //奇数
        } else {
            //偶数
        }
    }
    

    利用按位或“|”取整

    按位或运算的逻辑是这样: 0001 | 0011 = 0011,也就是两个位都是0,才是0,其它位都1。

    0 | 0 //0
    0 | 1 //1
    1 | 0 //1
    1 | 1 //1
    3 | 5 //7 即 0000 0011 | 0000 0101 = 0000 0111   因此,3|5的值得7
    

    按位或|有一个作用,通过搭配0进行运算可以用来实现数字取整。

    1.1 | 0 //1
    -2.0 | 0 //-2
    
    //非数字,则根据值强转为 0 或者 1 再取整,不过这种没啥意义
    null | 0 //0
    true | 0 //1
    

    Tips: 其实浮点数是不支持位运算的,所以会先把1.1转成整数1再进行位运算,就好像是对浮点数向下求整。所以 1 | 0 的结果就是1。

    还有一个就是在设计vue组件的时候,也常用到的地方如下:(具体为啥这么用,我也说不清楚,没研究过原理,知道的同学可以告诉我)。

    export default {
        //... 其它参数
        props {
            params: Object | Array //只要其中任意一个满足条件即可
        }
    }
    

    利用按位非“~”简化表达式

    普及一下~运算符,这个符号学名叫“按位非”,它是一个一元运算符。按位非操作符由一个波浪线(~)表示,按位非就是求二进制的反码。不管什么值使用了这个运算符,结果总是一个数字。按位非运算符,简单的理解就是改变运算数的符号并减去1。

    ~5 //-6
    ~0 //-1
    ~-2 //1
    ~true //-2 这里true被强转为1
    ~null //-1 这里null被强转为0
    ~undefined //-1 这里undefined被强转为0
    

    利用这个原理,我们可以在实际工作中这样应用:

    var str = 'hello world!'
    if(str.indexOf('w') != -1) { /*...*/ }
    //或者
    if(str.indexOf('w') > -1) { /*...*/ }
    
    //根据str.indexOf('w')的值,无外乎两种情况: -1, 0及正整数,
    //则,从-1到正整数中,经过按位运算~之后,则为:0或者任意负数,
    //实际表示为:false与true
    //上面的条件语句,可以改为:
    if(~str.indexOf('w')) { /*...*/ }
    

    Tips: 按位运算,简单的地方,可以使用。但一些比较复杂的、难理解的,我觉得应该尽量少用,因为会给阅读者带来困难,也会给自己带来麻烦。

    左移<<求2的n次方

    //二进制运算:
    01 << 2 //0100,十进制为4
    
    //实际运用可以这样:
    1 << 2 //4 即2的平方
    1 << 3 //8 即2的立方
    1 << 4 //16 即2的4次方
    

    无符号右移>>>判断数的正负

    正数的无符号右移与有符号右移结果是一样的。负数的无符号右移会把符号位也一起移动,而且无符号右移会把负数的二进制码当成正数的二进制码。即:

    1 >>> 0 //1
    2 >>> 0 //2
    4 >>> 0 //4
    4 >>> 1 //2
    4 >>> 2 //1
    -4 >>> 0 //4294967292
    -4 >>> 1 //2147483646
    -2 >>> 0 //4294967294
    -1 >>> 0 //4294967295
    

    观察上面的例子,我们得出一个结论,正数右移0位,值不变,而负数右移0位,值已经变化了。即可以通过这种关系判断一个数的正负:

    function isPos(n) {
        return n === n >>> 0
    }
    isPos(-1) //false
    isPos(1) //true
    

    利用&&连结条件与表达式

    通常,我们为了给一个变量赋值,正常的逻辑是:if语句判断,通过if逻辑控制语句,则赋值成功。利用&&符号,则可以直接在前面写上条件语句,后面写上执行语句。

    var a;
    if(true) {
        a = 10
    }
    
    //也可以写成
    var a;
    true && (a = 10);
    //或者
    var a = true && 10; //这种情况尽量不要使用,因为如果条件不成立,则会给a赋上一个false值。
    
    //常见使用:
    function fn(obj) {
        obj && (return true);
        return false;
    }
    
    var a = b = c = true, d;
    a && b && c && (d = 10);
    console.log(d) //10
    

    Tips: 对于&&,需要注意以下几点:

    1. 对于布尔值,逻辑与是非常简单的,只要有一个false,就返回false;
    2. 对于不是布尔值的情况则:
      如果第一个操作数是对象,则返回第二个数
      如果第二个操作数是对象,则只有在第一个操作数的求值结果为true的情况下才会返回该对象;
      如果第两个操作数都是对象,则返回第二个数操作数
      如果有一个操作数是null,则返回null
      如果有一个操作数是NaN,则返回第NaN
      如果第一个操作数是undefined,则返回undefined

    利用||取值(设置默认值)

    通常,我们声明一个变量,可能要根据条件进行赋值。正常的逻辑是:if ... else if ... else或者使用switch这两种语句块。

    var a, b;
    if(b === 0) {
        a = 1
    } else if(b === 1) {
        a = 2
    } else {
        a = 3
    }
    //或者
    var a, b = 1;
    switch(b) {
        case 0:
            a = 1;
            break;
        case 1:
            a = 2;
            break;
        default:
            a = 3;
            break;
    }
    

    如果直接使用||或运算符,则按从左到右,取第一个非空的一个值,否则直接使用false作为值使用。

    var b = 4;
    var a = (b === 0 && 1) || (b === 1 && 2) || 3; // 3
    
    //常用到的或运算取值方式一般是在函数里面。
    function doSomething(obj) {
        var o = obj || {} //如果obj不存在,则使用默认值
        var oo;
        o && (oo = obj || {}) //结合与运算,作变判断的前置条件
    }
    

    Tips: 关于||,需要注意以下几点:

    1. 对于布尔值,逻辑或是非常简单的,只要有一个true,就返回true;
    2. 对于不是布尔值的情况则:
      如果第一个操作数是对象,则返第一个操作数
      如果第一个操作数的求值结果为false,则返回第二个操作数
      如果两个操作数都是对象,则返回第一个操作数
      如果两个操作数是null,则返回null
      如果两个操作数是NaN,则返回NaN
      如果两个操作数是undefined,则返回undefined

    利用~~取整

    如果要将一个小数取整数部分,正常的逻辑是:parseInt()强转为整数方法,或者Math.floor()向下取整。其实,使用~~操作符可以更快速的取整。

    parseInt(12.55) //12
    Math.floor(12.55) //12
    ~~12.55 //12
    

    其实,除了取整的作用,它还可以达到强转数字的作用,比如:

    ~~true //1
    ~~false //0
    ~~[] //0
    ~~{} //0
    ~~undefined //0
    ~~!undefined //1
    ~~null //0
    ~~!null //1
    

    Tips: 虽然~~用起来比较骚气,但是为了可读性,本人还是建议使用Math.floor()更为稳妥,谁知道领导review代码的时候,会不会说你过于装逼,要被喷死。

    利用+将字符串转为数字

    如果要将一个表现字符串的数字转化为真正的数字,正常逻辑是:Number()或者parseInt()、parseFloat()实现转化。实际上,我们还可以更简捷,只需要在前面添加一个+就可以了。当然,你也可以用-来实现,只不过,这样子,则值就成了负数。

    Number('123') //123
    parseInt('123') //123
    parseFloat('123') //123
    parseFloat('123.0') //123
    parseFloat('123.0') //123
    parseFloat('123.1') //123.1
    
    //只要使用+,则可以实现数字的转化
    +'123' //123
    +'12.22' //12
    -'12.1' //-12.1
    
    //如果要取整并且转数字的话,使用~~
    ~~'123.33' //123
    

    实际上,+也适用于Date对象。

    var date = new Date //Fri Oct 27 2017 14:38:49 GMT+0800 (中国标准时间)
    date.getTime() //1509086332914
    
    //使用+直接输出时间缀
    +new Date //1509086332914
    

    利用!逻辑非强转布尔值

    逻辑非会将所有值,转为布尔值:

    !{} //false 一个空对象,实际上是一个引用,属于存在的引用地址值
    ![] //false 存在引用地址值
    !'' //true
    !0 //true
    !'hello' //false
    !null //true
    !NaN //true
    !undefined //true
    

    利用逻辑非运算符,可以省去一些多余的判断。比如经常要判断一个值非空:

    var a, b;
    if(a !== undefined && a !== null && a !== '' && a !== {} && a!== 0 && a!== undefined) {
        b = true;
    }
    

    实际上,如果像这面这种判断,只需要一个逻辑非。

    var a, b;
    !a && (b = true);
    

    利用!!实现变量检测

    如果要将一个值强转为布尔类型,正常的逻辑是:Boolean()强转为布尔值。

    Boolean(123) //true
    Boolean('hello') //true
    Boolean('false') //true
    Boolean(null) //false
    Boolean(undefined) //false
    Boolean('undefined') //true
    
    

    然而,通过!!两个非逻辑符,则可以将一个值强转为布尔值。

    !!123 //true
    !!'hello' //true
    !!'false' //true
    !!null //false
    !!undefined //false
    !!0 //false 注意,0也会转为false,在数字中,只有0会转为false,其它非0值,都会转为true
    !!'' //false
    !!NaN //false
    

    Tips: 任意的javascript的值都可以转换成布尔值。这些值会被转换成false:undefined,null,0,-0,NaN,"",而其它都变强转为true。通常,利用!!符号来检测一个变量是否存在。

    表达式的妙用

    自执行函数

    函数,只是声明,并不能直接执行,需要调用才会执行。而如果,变成了表达式,则会自动执行。自执行函数,则是利用了表达式的这种特性。那些匿名函数附近使用括号或一些一元运算符来引导解析器,指明运算符附近是一个表达式。
    按照这个理解,可以举出五类,超过十几种的让匿名函数表达式立即调用的写法:

    ( function() {}() );
    ( function() {} )();
    [ function() {}() ];
    
    ~ function() {}();
    ! function() {}();
    + function() {}();
    - function() {}();
    
    delete function() {}();
    typeof function() {}();
    void function() {}();
    new function() {}();
    new function() {};
    
    var f = function() {}();
    
    1, function() {}();
    1 ^ function() {}();
    1 > function() {}();
    

    Tips:另外值得再次注意的是,括号的含混使用——它可以用来执行一个函数,还可以做为分组运算符来对表达式求值。比如使用圆括号或方括号的话,可以在行首加一个分号,避免被用做函数执行或下标运算:

    ;( function() {}() )
    ;(function(){})()
    

    isNaN判断是否为合法数字

    isNaN(x) 函数用于检查其参数是否是非数字值。如果x是特殊的非数字值NaN(或者能被转换为这样的值),返回的值就是 true。如果x是其他值,则返回 false。
    经常,我们在页面功能模块开发过程中,拿到的交互数据,有很大可能是字符串形式的“数字”。这时,我们如果要做计算,就得先判断是否合法的数字。

    isNaN('111') //false
    isNaN(111) //false
    isNaN(12.2) //false
    isNaN('12.2') //false
    isNaN('aa11') //true
    isNaN(undefined) //true 一切非数字,返回都是true
    

    Tips: 我们可以这样总结,只要是表现得像数字(字符串形式,或者真正数字)的都可以检测出是属于“数字”,否则属于不合法数字。

    缓存Array.length

    写for循环的时候,经常是这样:

    var arr = [1,2,3]
    for(var i = 0; i < arr.length; i++) {/**/}
    

    在处理一个很大的数组循环时,对性能的影响将是非常大的。为了提升运行性能,需要将数组使用一个变量缓存起来使用。

    for(var i = 0, len = arr.length; i < len; i++){/**/}
    

    判断属于是否存在于对象中

    1、使用in运算符

    var obj = { a: 11 }
    'a' in obj //true
    'b' in obj //false
    

    2、使用!!

    var obj = { a: 11 }
    !!obj.a //true
    !!obj.b //false
    
    //或者使用undefined来判断,但是可能也会属于值本身就是undefined,这样子就判断不出来了。
    obj.a !== undefined //true
    obj.b !== undefined //false
    

    3、hasOwnProperty()方法

    var obj = { a: 11 }
    obj.hasOwnProperty('a') //true
    obj.hasOwnProperty('b') //false
    //该方法只能判断自有属性是否存在,对于继承属性会返回false。
    

    获取数组最后一个元素

    获取数组最后一个元素的方式有多种,常用的有array.pop或者array[array.length - 1]方式。实际上,还可使用Array.slice(-1)

    var arr = [1,2,3,4,5]
    arr.pop() //5 此方法会改变数组的结构,不推荐
    arr[arr.length - 1] //5
    arr.slice(-1)[0] //5 不需要计算数组的长度,直接拿到最后一个元素
    

    数组截断

    数组有时候需要设置一个上限,或者删除数组中的一些元素,使用array.length = [长度值] 这种方式非常有用。

    var arr = [1,2,3,4,5]
    arr.slice(0,3)
    console.log(arr) //[1,2,3]
    arr = arr.splice(0,3) //splice方法会改变数组结构
    console.log(arr) //[1,2,3]
    
    //直接设置长度值
    arr.length = 3
    console.log(arr) //[1,2,3]
    
    //还可以实现数组的清空操作
    arr.length = 0; //arr => []
    

    数组合并

    合并数组常用的方法是:concat。不过现有一种更快速的合并方式:Array.push.apply(arr1,arr2)

    var arr1 = [1,2,3], arr2 = [4,5];
    
    arr1 = arr1.concat(arr2) //常规方式
    //或者
    arr1 = arr1.push.apply(arr1,arr2); //装逼方式,但运行速度更快!
    
    console.log(arr1) //[1,2,3,4,5] 两种方式都可以达到合并的结果
    

    类数组利用数组的方法

    类数组拥有length属性,但不具有数组所具有的方法。为了方便操作,常需要将其转化为数组结构。

    var arrLike = { '0': 'a', '1': 'b', length: 2 }; //下标为 '0' '1'才符合数组的数据结构
    Array.prototype.join.call(arrLike, '-'); //a-b 模拟数组的join()方法,使用-分隔。
    Array.prototype.slice.call(arrLike); //['a','b'] //模拟数组slice方法,返回一个新的数组
    
    function fn() {
        var args = Array.prototype.slice.call(arguments); //获取所有参数的列表
    }
    

    利用arguments.callee实现递归

    普通的递归方法是这样写的:

    function fn(n) {
        if(n <= 1) {
            return 1
        } else {
            return n * fn(n - 1)
        }
    }
    

    但是当这个函数变成了一个匿名函数时,我们就可以利用callee来递归这个函数。

    function fn(n) {
        if(n <= 1) {
            return 1
        } else {
            return n * arguments.callee(n - 1)
        }
    }
    

    Tips: 这个方法虽然好用,但是有一点值得注意,ECMAScript4中为了限制js的灵活度,让js变得严格,新增了严格模式,在严格模式中我们被禁止不使用var来直接声明一个全局变量,当然这不是重点,重点是arguments.callee这个属性也被禁止了。

    给回调函数传递参数

    经常,函数一般是可以当作参数来使用。但是有时候,一些函数自身需要带参,这时候,把函数当参数使用的话,就显得比较麻烦。一般的处理方法是,通过附加的传参的办法。

    function callback(obj) {
        console.log(obj)
    }
    
    function fn(callback, obj) {
        callback(obj)
    }
    

    但是有些时候,函数的参数限定了,只能传一个回调函数,这时,这种附加传参的显得无力了。采用闭包的方式可以解决此问题。

    function callback(obj) {
        return function() {
            console.log(obj)
        }
    }
    
    document.body.addEventListener('click',callback('hello')); //执行callback('hello'),则返回的是一个函数
    

    利用正则表达式处理实际问题

    1、获取有价值部分
    经常,在项目中,可能会获取到一些带有带有各种干扰因素的值。而我们需要提取相应的数字部分,可以使用正则表达式获取。

    var val = "98.8元/斤";
    val.replace(/[^(0-9).]/ig,'') //98.8
    

    2、获取URL中的value值
    经常,我们会从url中获取到location.search,但却是一个字符串,然后还需要将这个字符串进行分拆才能得到想要的值,而使用正则,则快速方便得多。

    //假设当前url为:https://www.baidu.com?name=jawil&age=23
    
    //普通实现:
    function fn(key) {
        var obj = {}
        var searchStr = window.location.search.substring(1);
        searchStr.split('&').forEach(function(item) {
            var arr = item.split('=');
            obj[arr[0]] = arr[1];
        });
        return obj[key]
    }
    
    console.log(fn('name')) //jawil
    console.log(fn('age')) //23
    
    //使用正则实现:
    function fn(key) {
        var reg = new RegExp("(^|&)"+ key +"=([^&]*)(&|$)"); //url正则表达式
        var flag = window.location.search.substring(1).match(reg);
        return flag != null ? unescape(flag[2]) : null;
    }
    
    console.log(fn('name')) //jawil
    

    3、格式化数字
    通常,我们需要将一个大的数字 19456212 转化为 19,456,212 这种样子。使用正则表达式可以很好的处理:

    var num = 19456212.85;
    num.toLocaleString() //19,456,212.85 toLocaleString()会根据你机器的本地环境来返回字符串,它和toString()返回的值在不同的本地环境下使用的符号会有微妙的变化
    
    //为了保险起见,可以使用方法实现
    function fn(s, n) {
        n = n > 0 && n <= 20 ? n : 2;
        s = parseFloat((s + '').replace(/[^\d\.-]/g, '')).toFixed(n) + '';
        var l = s.split('.')[0].split('').reverse(),
            r = s.split('.')[1];
        t = '';
        for (i = 0; i < l.length; i++) {
            t += l[i] + ((i + 1) % 3 == 0 && (i + 1) != l.length ? ',' : '');
        }
        return t.split('').reverse().join('') + '.' + r;
    }
    
    console.log(fn(19456212.85)) //19,456,212.85
    

    总结

    我归纳的东西比较杂,都是平时遇到的一些小技巧。基本上都是一些表达式上的优化,之后,再有啥好东西,我继续在这篇中添加。
    javascript日常的使用过程中,很多语句,其实都是可以有多种写法,尝试着使用不同的写法,对比一下哪种写法更简洁,更易读,更能提高执行效率,可以实现对代码的优化。而我在项目开发过程中使用到的一些简写,很大部分是了解别人的代码写法,从中学以致用。另外一个很好的办法就是,查调试自己的压缩代码的时候,不防读一下被压缩优化过的代码,变成了啥样子的,会有发现,压缩过的代码,实际上编译器已经帮我们实现了最佳写法,这样,不妨借鉴一下,用到自己的项目中去。

    相关文章

      网友评论

      本文标题:JS装逼实用技巧

      本文链接:https://www.haomeiwen.com/subject/xhglpxtx.html