十二月已经过半,冬季是一个美妙的季节,寒冷的空气逼得人们不得不躲在安逸舒适的环境里生活。冬季会给人一种安静祥和的氛围,让人沉浸在其中,仿佛是一个旧的阶段的结束,同时也是一个新的阶段的开始。这么说来,西方和中国的圣诞节和春节都选择在了冬季也不是没有道理,在一年中最寒冷的时候,人们拥簇在温暖的环境里,彼此诉说着过去一年里自己的成就,展望着新的一年里美好的愿望,相互挂念的人团聚,天气的寒冷和人情的温暖形成了强烈的对比。而在天寒地冻之中,仿佛更有利于人们思考,去探寻知识的真谛。

  这次想分享的是 JS 当中的逻辑运算符与、或,也就是 &&|| ,初来乍到的同学们看到这里就会觉得没趣了,这玩意有什么好分享的,刚开始学 JS 的时候不就会了吗,我用了无数遍都没有什么问题啊。而有经验的同学可能会陷入沉思,难不成这其中会有什么奥秘所在?没错,别看这简简单单的几个运算符,虽然这是最基础的知识,但其中隐藏的奥秘却十分耐人寻味,接下来我就为大家一一揭开这简答的运算符背后的奇妙之处。

  基础的作用我就不说了,这两个符号是个程序员都能明白,这里首先我想先来说一说 JS 当中的隐式转换。

  众所周知,JS 在做逻辑判断的时候会自动将非布尔类型的值进行隐式转换,转换成布尔类型的值然后在进行逻辑运算。在初学 JS 的时候,都会讲到在隐式转换中,除了几个特定的假值,其他的均会转换成真值,这些假值有:

1 NaN;
2 "";
3 undefined;
4 null;
5 0;

  有了这些隐式转换的规则,便构成了 JS 当中逻辑运算的核心基础。

  其实在 JS 当中,要说“逻辑运算符”其实并不完全正确,Kyle Simpson 在《You Don't Know JS》系列书当中提到:“与其说是‘逻辑运算符’,不如说是‘选择器运算符’。” 为什么大师要这样说呢?其实我们大多数人都被 JS 的表象给蒙蔽了,比如下面一段非常简单的代码:

1 if( "hello" && 0 ) {
2     console.log(true);
3 } else {
4     console.log(false);
5 }

  如果你对 JS 了解的不够深刻,你可能会这样解释这段代码:首先在逻辑判断中,"hello" 是一个真值,0 是一个假值,一个真值和一个假值进行与运算,结果为 false 。这也可能是大多数人的理解,但其实不然,其内部的原理可不止这么简单,因为 && 和 || 返回的并不是判断条件的真假 ,而是判断条件中的一个原始值。它将依次对条件判断中的值进行判断,如果是非布尔值,则转换成布尔值做判断,然后再根据判断条件来决定返回哪一个值。

  对于 && :该运算符返回条件语句中的第一个假值,如果所有的值都为真,则返回最后一个值,&& 也被称为 “守护运算符” 。比如下面一段代码:

1 var a = "hello" && "world";
2 console.log(a);  //world
3 var b = 0 && 1;
4 console.log(b);  //0

  可以看出,逻辑运算符其实返回的并不是条件的真假,而是原始值。如果条件语句中有多个 && 运算符,则一样遵循以上原则,从左向右依次判断,如果遇到了假值,就返回该假值,如果所有值都为真,则返回最后一个值。

  对于 ||:该运算符与 && 运算符相反,它返回条件语句中的第一个真值,如果所有值都为假,则返回最后一个值。比如下面一段代码:

1 var a = "hello" || 0;
2 console.log(a);  //hello
3 var b = 0 || NaN;
4 console.log(b);  //NaN

  同样,|| 返回的也不是布尔值。如果有多个 || 则同样遵循相同的原则,从左向右依次扫描。

  讲到这里也就来到了本篇文章的核心,在 JS 当中,条件判断语句都是建立在隐式转换之上的,也就是说所谓的逻辑运算符,实际上是在条件判断语句中从左向右依次扫描,如果是一个布尔值,则判断该布尔值的真假,如果是一个非布尔值,则先对该值进行隐式转换,然后再判断真假,如果满足条件,则返回该值,如果没有满足条件值,则返回最后一个值,然后在对返回的这个值做判断,如果是一个布尔值,则直接判断,如果是一个非布尔值,则先隐式转换成布尔值,再做判断。所以我们也可以把 && 称为 “取假运算符” ,把 || 称为 “取真运算符” ,因为这两个运算符的实质都是取条件语句中的第一个真值或者假值,如果始终没有找到,则返回最后一个值。而这样的算法也恰好满足逻辑判断的需求,比如 && 运算符,如果所有的值都是真值,那么返回哪个值其实都无所谓,因为所有值都能够被隐式转化为 true ,而只要有一个假值,则判断条件不成立,所以会返回第一个遇到的假值。而 || 运算符,如果所有的值都是假值,返回任意一个都会被隐式转换成 false ,但只要遇到了一个真值,则判断条件成立,所以会返回第一个遇到的真值。&& 和 || 运算符都是 “短路” 的。

  所以我们可以自己实现一个逻辑运算的函数:

 1 // && 等价于:
 2 function AND () {
 3     for (var i = 0; i < arguments.length; i++) {
 4         if (!arguments[i]) {
 5             return arguments[i];
 6         }
 7     }
 8     return arguments[i-1];
 9 }
10
11 // || 等价于:
12 function OR () {
13     for (var i = 0; i < arguments.length; i++) {
14         if(arguments[i]) {
15             return arguments[i];
16         }
17     }
18     return arguments[i-1];
19 }

  (注:在这里我同时也想对 ! 这个运算符做讲解,但考虑到内容和篇幅的问题,暂时不做深入探究,仅做简单讲述。 ! 运算符实际上运行机制与 && 和 || 是一样的,首先会对参数值做判断,如果是一个布尔值,则进行取反运算,如果是一个非布尔值,则先进行隐式转换,再进行取反运算。而我们通常写的 if (something) 语句,实际上的意思 if (!!something))

  然后我们可以这样使用:

1 var a = ["hello", undefined, "world"];
2 console.log(AND.apply(null, a));  //undefined
3 var b = ["", 0, NaN];
4 console.log(OR.apply(null, b));  //NaN

  进而,我们就可以推断出一下结论:

1 a = x || y;
2 //等价于:
3 a = x ? x : y;
4
5 a = x && y;
6 //等价于:
7 a = x ? y : x;

  这通常也是一些压缩工具所做的事情,它们尽可能的将繁杂的条件判断语句转换成 && 或者 || ,因为这样代码更加的精简,但是可读性则就不那么可观了。

  对于最开始的那一段代码:

1 if( "hello" && 0 ) {
2     console.log(true);
3 } else {
4     console.log(false);
5 }

  我们现在就要这样解释:首先这是个与运算符,与运算符的作用是取第一个假值,如果所有的值都为真,那么则返回最后一个值。所以在这条语句中,第一个值是 "hello" ,因为该值是一个非布尔值,JS 引擎会先将它隐式转换成布尔值,而该值不在假值的范围内,所以会被转化成 true 。随后 JS 引擎会继续查找,第二个值是 0 ,该值同样也不是一个布尔值,所以 JS 引擎也会先将它隐式转换成布尔值,而该值在假值的范围内,所以会被转化成 false ,满足 && 运算符的查找条件,则将值 0 返回。而条件判断语句接受到了值 0 ,该值不是一个布尔类型的值,所以会先对它进行隐式转换,而该值在假值范围内,所以会被转化成 false ,然后控制台会输出 false。

  所以说以后当我们看到 && 和 || 时候,就不要仅仅的从字面上的意义去理解了,在看完了这篇文章之后,你就可以很自豪很有底气的对别人说,你真的会用逻辑运算符吗?

  好了,就这么两个小玩意居然背后也有着这么多的精髓,看来知识的深度是无穷的,冬季还真是一个能引发人们思考的季节。圣诞节即将到来,在这里提前预祝大家圣诞快乐,Merry Christmas!

转载于:https://www.cnblogs.com/dong-xu/p/6196844.html

探寻 JavaScript 逻辑运算符(与、或)的真谛相关推荐

  1. JavaScript逻辑运算符的使用技巧

    前言 !, &&, || 三个运算符是JavaScript中重要的逻辑运算符,本文将介绍这三个运算符在JavaScript实际编程中的有趣使用技巧. 取反运算符(!) 如果对一个值连续 ...

  2. JavaScript逻辑运算符“”和“||”短路原则的应用

    文章出自个人博客https://knightyun.github.io/2018/06/01/js-logic-compute,转载请申明 逻辑运算符 在Javascript中,有逻辑运算符 与 &a ...

  3. web前端开发学什么,JavaScript逻辑运算符与赋值运算符

    问题: 从浏览器地址栏输入url到请求返回发生了什么 你一看这种烂掉牙的问题,小case,但996面试大佬由此延展的问题已经远远超越了这个问题本身了,不信你就接着看. 我回答了首先会进行 url 解析 ...

  4. JavaScript逻辑运算符

    或与非:&&  ||  ! ----------------------------------------------------------- 一.流程控制: 1.条件判断: if ...

  5. 详细解析 JavaScript 获取元素的坐标

    随时随地技术实战干货,获取项目源码.学习资料,请关注源代码社区公众号(ydmsq666) from:https://www.cnblogs.com/dong-xu/p/7150715.html?utm ...

  6. JavaScript王者归来

    内容简介回到顶部↑ 你手中的这本<javascript王者归来>不仅是一本传播知识的书,更是一本求道的书. 本书分为五个部分循序渐进地与读者讨论了javascript的方方面面,从简单的语 ...

  7. (4)javascript的运算符以及运算符的优先级

    运算符的使用方法 在javascript的程序中要完成各种各样的运算,是离不开运算符的. 在javascript中,按运算符类型可以分为算术运算符.赋值运算符.比较运算符.逻辑运算符.条件运算符等. ...

  8. javascript进制转换_「js基础」JavaScript逻辑和位运算符归纳

    上一篇文章呢,我们讲述了JavaScript运算符中的关系运算符和逻辑运算符,那么紧接上一篇的文章我们今天来说说逻辑运算符. 引言 逻辑运算符不是很难,也不是很多,我们只要记住三个就可以啦!分别是: ...

  9. 32 位的有符号整数_「js基础」JavaScript逻辑和位运算符归纳

    上一篇文章呢,我们讲述了JavaScript运算符中的关系运算符和逻辑运算符,那么紧接上一篇的文章我们今天来说说逻辑运算符. 引言 逻辑运算符不是很难,也不是很多,我们只要记住三个就可以啦!分别是: ...

最新文章

  1. 你知道Redis可以实现延迟队列吗?
  2. 用UML做好系统分析
  3. 基于外卖评论的舆情风控
  4. python 自定义函数和循环_Python循环语句——对for循环和while循环应用自定义函数公式的实践,套用,练习...
  5. php怎么克隆,利用php怎么对对象进行克隆
  6. 【M17】考虑使用缓式评估
  7. redis事务原理,使用,详解
  8. SpringMVC拦截器(interceptors)
  9. java 模拟电梯_java模拟电梯运行简单实现,swing界面
  10. hive拉链表方案二
  11. a as as big rat_Whats the meaning of give a rats as..._考试资料网
  12. Docker--docker ps 命令与结果解析
  13. spring boot中如何实现在手机注册和登录时获取验证码(阿里短信服务)
  14. 图片文件Exif信息详细说明
  15. Python全栈最全学习之路-Python基础(十一)
  16. shiro的受权管理
  17. 可调光的LED驱动解决方案
  18. 流行编曲(5)采样、小打、Pad、声场
  19. 【密码学】C 语言实现 RSA 模幂运算
  20. p4est 2.3.2 安装

热门文章

  1. python谱聚类算法_谱聚类 - python挖掘 - 博客园
  2. ajax post提交数据_第三十五天JavaScript中的ajax
  3. stm32中断优先级_浅谈STM32串口USART1的使用
  4. jquery ajax实例 php,jquery中各个ajax实例操作
  5. php连接云数据库服务器,服务器使用PHP连接sqlserver数据库
  6. UVa10340 - All in All
  7. JavaScript流程分支结构(1)
  8. 2017蓝桥杯B组:取数位(递归,求余和除法)
  9. bzoj 1113: [Poi2008]海报PLA(栈)
  10. bzoj 3111: [Zjoi2013]蚂蚁寻路(DP)