最近重新翻开原生JS,又得到很多不同的体会,虽然现在开发框架那么多,但很多思想都还是离不开原生的基础。今天呢,我就根据自己的学习总结一下逻辑与(&&)和(逻辑或(||)和逻辑非(!)。

  基本定义

      ||:逻辑或,只有表达式的值都为false,才返回false,其他情况返回true  比如:(8>5)&&(4<6),返回true;(8<5)&&(4<6),返回false

      &&:逻辑与,若两边表达式的值都为true,才返回true;比如: (8>5)&&(4<6),返回true;(8<5)&&(4<6),返回false

    !:逻辑非,若表达式的值为true,则返回false;若表达式的值为false,则返回true  比如:!(9>2),返回false;!(9<2),返回true,

这是他们的基本定义,我们都再熟悉不过了,看似好像没有什么大用,其实这三个运算符在我们编写代码中所起到的重要作用。

 首先看几个例子

 var a = 0 || 1 || 2;// a = 1
var b = 1 || 0 || 3;// b = 1
var c = 0 || 0 || 0;// c = 0
var a = 0 && 1 && 2;// a = 0
var b = 1 && 0 && 3;// b = 0
var c = 0 && 0 && 0;// c = 0
var d = 1 || 2 || 4;// d = 1
OK.看完上面的栗子,你会发现逻辑运算符不是简单的返回true/false,而是返回了某一个内容,其实它的判断过程是这样的:
逻辑运算符是从前往后依次判断,判断到那个能得出最终结论的地方,就会停止往下判断并返回最后判断的那个内容,不管它是真是假。(记住这句话就行)
比如:||运算符,只要一个为真就为真,所以你可以认为它会从前往后一直按顺序在找真,找到真就返回那个真。找不到也返回最后一个判断的那个值。
0 || 1 || 2,判断到1就知道结果肯定是真,不再继续判断,返回1。
1 || 0 || 3, 判断到第一个就知道结果是真,不再继续判断,返回1。
0 || 0 || 0;一直判断到了最后一个,才知道结果为假,返回最后一个判断的0。
同理:&&运算符是只要一个为假就为假,所以它会从前往后一直找假的,返回最后一个判断的值。

实际应用

   1给函数参数定义默认值

 var a = a||2;  //判断过程:如果变量a是真就停止判断,返回a,如果a是假就继续判断,所以返回2. 实际上就是给a起了个默认值2。
也等同于:
if(a){
a = a
}else{
a =2
} //因为js不像php可以直接在型参数上定义func($attr=5)一个默认值,所以可以通过这种方法给参数定义默认值。(顺便提醒一下,最新的es6标准已经可以定义函数参数默认值了。)
可以看出用逻辑运算符既节省代码又实现功能

   2 大大减少代码量

  首先出个题:假设:成长速度为5显示1个箭头;成长速度为10显示2个箭头;成长速度为12显示3个箭头;其他都显示都显示0各箭头。

用代码怎么实现?(例子来源于网上的一个贴吧里)

最简单的

var add_level = 0;
if(add_step == 5){
add_level = 1;
}else if(add_step == 10){
add_level = 2;
}else if(add_step == 12){
add_level = 3;
}else {
add_level = 0;
}

 稍好些的switch:

var add_level = 0;
switch(add_step){
case 5 : add_level = 1;
break;
case 10 : add_level = 2;
break;
case 12 : add_level = 3;
break;
default : add_level = 0;
break;
}

更好的

var add_level = (add_step==5 && 1) || (add_step==10 && 2) || (add_step==12 && 3)  || 0;

更更好的

var add_level={'5':1,'10':2,'12':3}[add_step] || 0;
等同于下面的代码:
var obj = {'5':1,'10':2,'12':3};
var add_level=json[add_step] || 0;

如果需求改成成长速度为>12显示4个箭头;成长速度为>10显示3个箭头;成长速度为>5显示2个箭头;成长速度为>0显示1个箭头;成长速度为<=0显示0个箭头。

var add_level = (add_step>12 && 4) || (add_step>10 && 3) || (add_step>5 && 2) || (add_step>0 && 1) || 0;
console.log(add_level);

哇塞,是不是发现我们的代码一下子就少了很多,但需要注意的一点就是:一方面精简js代码,能实质性的减少网络流量,尤其是大量应用的js公用库。同时,也带来了代码可读性的降低,个人比较推荐的做法是:如果是相对复杂的应用,请适当地写一些注释。

我们可以不使用这些技巧,但是我们一定要能看懂,因为这些技巧已经广泛应用,不理解这些你就很难看懂别人的代码。

关于!运算符

很多代码写if(!!attr),为什么不直接写if(attr);其实这是一种更严谨的写法,如下:

console.log(typeof 5); //number
console.log(typeof !!5); //boolean 可以看出:!的作用是把一个其他类型的变量转成的bool类型。
        不管!后面是什么类型,逻辑非都会将它转成布尔类型

&& (逻辑与) 和||(逻辑或)的优先级

混合使用的时候要注意他们的优先级:&& (逻辑与) 优先级高于||(逻辑或)

return a && b || c 

先算a&&b,a 是 false ,a&&b就是返回a,再算a||c,则肯定返回 c;如果a是true ,则要看B,自己想想吧。。。

另附:JS运算符优先级(从高到低列出)

 运算符 描述
. [] () 字段访问、数组下标、函数调用以及表达式分组
++ -- - ~ ! delete new typeof void 一元运算符、返回数据类型、对象创建、未定义值
* / % 乘法、除法、取模
+ - + 加法、减法、字符串连接
<< >> >>> 移位
< <= > >= instanceof 小于、小于等于、大于、大于等于、instanceof
== != === !== 等于、不等于、严格相等、非严格相等
& 按位与
^ 按位异或
| 按位或
&& 逻辑与
|| 逻辑或
?: 条件
= oP= 赋值、运算赋值
, 多重求值

如有错误,欢迎指正。

你不得不知的逻辑或(||)与(&&)非(!)的更多相关文章

  1. [shell]shell 中| && || () {} 用法以及shell的逻辑与或非

    转自:https://www.jianshu.com/p/617c1ee1e46e | 运算符 管道符号,是unix一个很强大的功能,符号为一条竖线:"|".用法: command ...

  2. shell(2)-&& 与 || 逻辑或与非

    test 命令测试 -常见的测试类型–测试文件状态–字符串比较–整数值比较–逻辑测试&& 如果是“前面”(真),则“后面”[ -f /var/run/dhcpd.pid ] & ...

  3. shell脚本之 operater.sh 算术运算符;比较运算符;布尔运算符;逻辑与或非运算符;字符串运算符的使用

    1.注意 格式不能变 尤其是变量和变量表达式之间的空格:[ 空格] .[ $a == $b ] 中间的空格严格不能变:否则报错 关系运算符 关系运算符只支持数字,不支持字符串,除非字符串的值是数字. ...

  4. c++逻辑与或非优先级

    按优先级从高到低排列:!.&&.||,!的优先级最高,&&的优先级居中,||的优先级最低.

  5. numpy: np.logical_and/or/not (逻辑与/或/非)+python3-曲线拟合(polyfit/polyval)

    可以用拟合两个变量之间的关系,然后根据一个变量,去推测出另外一个变量的推测值

  6. ERwin创建逻辑模型

    1.逻辑实体添加非主键属性的三种的方式 属性1:在图中直接创建 属性2:在模型导航器中创建 属性3:在属性对话框中创建 2.实体显示选项(Entity Display) Rolename/Attrib ...

  7. linux逻辑卷管理

    近期在进行linux充电,依据网络资料自己整理的资料,分享一下 ---------------------------------------------------------- Linux逻辑卷管 ...

  8. [.NET产品源码保护].NET防止反编译(非混淆加密)

    .NET产品源码保护产生的背景: .NET源码加密方案支持C#及VB.NET等语言开发的ASP.NET及WINFORM应用.利用.NET支持托管代码与非托管代码共存的特性,将C#代码经过处理放于非托管 ...

  9. linux存储管理之逻辑卷

    LVM管理 ====================================================================================创建LVMVG扩展/ ...

随机推荐

  1. Java基础(下)(JVM、API)

    Java基础(下) 第三部分:Java源程序的编辑 我们知道,计算机是不能直接理解源代码中的高级语言,只能直接理解机器语言,所以必须要把高级语言翻译成机器语言,计算机才能执行高级语言编写的程序. 翻译 ...

  2. 【NodeJs】记录一个阿里云redis的坑

    背景 一个风和日丽的下午,一个上线的大好日子,以为一切准备好,上个线也就三五分钟的事.但是... 图样图森破. 背景是这样的,项目需要,在阿里云开通redis,在项目中配好redis的连接字符串后,以 ...

  3. java开发之阿里云对象存储OSS和云数据库Memcache的使用

    web开发中标配:aliyun ECS(阿里云服务器),aliyun RDS(阿里云数据库),aliyun OSS(阿里云对象存储),aliyun Memcache(阿里云缓存数据库). 今天就介绍下 ...

  4. MySQL创建数据库与创建用户以及授权

    1.create schema [数据库名称] default character set utf8 collate utf8_general_ci;--创建数据库 采用create schema和c ...

  5. TypeScript设计模式之中介者、观察者

    看看用TypeScript怎样实现常见的设计模式,顺便复习一下. 学模式最重要的不是记UML,而是知道什么模式可以解决什么样的问题,在做项目时碰到问题可以想到用哪个模式可以解决,UML忘了可以查,思想 ...

  6. solr 分词词库管理思路

    solr 分词词库管理思路 大概有以下几种思路: 1. 自定义 SolrRequestHandler        由 SolrRequestHandler 来进行对分词器,进行A)词库加载B)动态添 ...

  7. ubuntu auto mount自动挂载硬盘

    Ubuntu 挂载的文章在网上也不少,推荐一个: http://wenku.baidu.com/link?url=N2c7axijp_KYaYkt2CrZFNZPzzS8xBHLQSTUcI2F85I ...

  8. @Required 注释应用于 bean 属性的 setter 方法,它表明受影响的 bean 属性在配置时必须放在 XML 配置文件中,否则容器就会抛出一个 BeanInitializationException 异常。

    @Required 注释应用于 bean 属性的 setter 方法,它表明受影响的 bean 属性在配置时必须放在 XML 配置文件中,否则容器就会抛出一个 BeanInitializationEx ...

  9. Cordova原理一

    我们知道cordova中js要和native通信都是通过 cordova plugin来实现的.如果我们自己创建一个cordova plugin,它其实就是几个独立的文件按照一定的目录结构放在一起,然 ...

  10. java内部发送http请求并取得返回结果,修改response的cookie

    public Object userLogin(HttpServletRequest request, HttpServletResponse response, String email, Stri ...