《JavaScript核心概念及实践》——2.3 运算符

简介: 事实上,很多新手会犯的错误都与JavaScript自身的灵活性有关系,比如由于JavaScript的弱类型特性,解释器会在运行时进行类型的转换,这样会隐藏一些出错的可能性,又比如是采取中括号运算符还是使用点运算符,以及相等还是等同等,JavaScript本身不会强制限制开发者,但是有一个好的编码标准仍然是非常必要的。

本节书摘来自异步社区《JavaScript核心概念及实践》一书中的第2章,第2.3节,作者:邱俊涛著,更多章节内容可以访问云栖社区“异步社区”公众号查看

2.3 运算符

运算符,通常是容易被忽略的一个内容,但是一些比较古怪的语法现象仍然可能需要用到运算符的结合率或者其作用来进行解释。JavaScript中,运算符是一定需要注意的地方,有很多具有JavaScript编程经验的人仍然免不了被搞得晕头转向。

我们在这一节主要讲解这样几个运算符,诸如算术运算、位运算等和其他的主流程序设计语言类似,在这里不做讨论。

2.3.1 中括号运算符([])
中括号([])运算符可用在数组对象和对象上,从数组中按下标取值。

var array = ["one", "two", "three", "four"];
array[0]

而[]同样可以作用于对象,一般而言,对象中的属性的值是通过点(.)运算符来取值,如:

var object = {
    field : "self",
    printInfo : function(){
        print(this.field);
    }
}

object.field;
object.printInfo();

但是考虑到这样一种情况,我们在遍历一个对象的时候,对其中的属性的键(key)是一无所知的,我们怎么通过点(.)来访问呢?这时候我们就可以使用[]运算符。

for(var key in object){
    print(key + ":" + object[key]);
}

运行结果如下。

field:slef
printInfo:function (){
        print(this.field);
}

2.3.2 点运算符(.)
点运算符的左边为一个对象(属性的集合),右边为属性名,应该注意的是右边的值除了作为左边的对象的属性外,同时还可能是它自己的右边的值的对象。

var object = {
    field : "self",
    printInfo : function(){
        print(this.field);
    },
    outter:{
        inner : "inner text",
        printInnerText : function(){
            print(this.inner);
        }
    }
}

object.outter.printInnerText();

这个例子中,outter作为object的属性,同时又是printInnerText()的对象。

但是点(.)运算符并不总是可用的,考虑这样一种情况,如果一个对象的属性本身就包含点(.)的键(self.ref),点运算符就无能为力了。

var ref = {
    id : "reference1",
    func : function(){
        return this.id;
    }
};

var obj = {
    id : "object1",
    "self.ref" : ref
};

当我们尝试访问obj的“self.ref”这个属性的时候:obj.self.ref,解释器会以为obj中有个名为self的属性,而self对象又有个ref的属性,这样会发生不可预知的错误,一个好的解决方法是使用中括号([])运算符来访问。

print(obj["self.ref"].func());
在这种情况下,中括号运算符成为唯一可行的方式,因此,建议在不知道对象的内部结构的时候(比如要遍历对象来获取某个属性的值),一定要使用中括号运算符,这样可以避免一些意想不到的问题。

2.3.3 相等与等同运算符
运算符==读作相等,而运算符===则读作等同。这两种运算符都是在JavaScript代码中经常见到的,但是意义则不完全相同,简而言之,相等运算符会对两边的操作数做类型转换,而等同则不会。我们还是通过例子来说明。

print(1 == true);
print(1 === true);
print("" == false);
print("" === false);

print(null == undefined);
print(null === undefined);

运行结果如下。

true
false
true
false
true
false

相等和等同运算符的规则分别如下。

相等运算符
如果操作数具有相同的类型,则判断其等同性,如果两个操作数的值相等,则返回true(相等),否则返回false(不相等)。

如果操作数的类型不同,则按照这样的情况来判断:

  • null和undefined相等;
  •  其中一个是数字,另一个是字符串,则将字符串转换为数字,再做比较;
  •  其中一个是true,先转换成1(false则转换为0)再做比较;
  •  如果一个值是对象,另一个是数字/字符串,则将对象转换为原始值(通过toString()或者valueOf()方法);
  •  其他情况,则直接返回false。

等同运算符
如果操作数的类型不同,则不进行值的判断,直接返回false。

如果操作数的类型相同,分下列情况来判断:

 - 都是数字的情况,如果值相同,则两者等同(有一个例外,就是NaN,NaN与其本身也不相等),否则不等同;

  •  都是字符串的情况,与其他程序设计语言一样,如果串的值不等,则不等同,否则等同;
  •  都是布尔值,且值均为true/false,则等同,否则不等同;
  •  如果两个操作数引用同一个对象(数组,函数),则两者完全等同,否则不等同;
  •  如果两个操作数均为null/undefined,则等同,否则不等同。

比如:

var obj = {
    id : "self",
    name : "object"
};

var oa = obj;
var ob = obj;

print(oa == ob);
print(oa === ob);

会返回:

true
true

再来看一个对象的例子。

var obj1 = {
    id : "self",
    name : "object",
    toString : function(){
        return "object 1";
    }
}

var obj2 = "object 1";

print(obj1 == obj2);
print(obj1 === obj2);

返回值为:

true
false

obj1是一个对象,而obj2是一个结构与之完全不同的字符串,而如果用相等操作符来判断,则两者是完全相同的,因为obj1重载了顶层对象的toString方法。

而不等(!=)和不等同(!==),则与相等(==)/等同(!==)相反。因此,在JavaScript中,使用相等/等同、不等/不等同的时候,一定要注意类型的转换,这里推荐使用等同/不等同来进行判断,这样可以避免一些难以调试的bug。

事实上,很多新手会犯的错误都与JavaScript自身的灵活性有关系,比如由于JavaScript的弱类型特性,解释器会在运行时进行类型的转换,这样会隐藏一些出错的可能性,又比如是采取中括号运算符还是使用点运算符,以及相等还是等同等,JavaScript本身不会强制限制开发者,但是有一个好的编码标准仍然是非常必要的。jslint是一个用于静态检查JavaScript代码的工具,开发者可以在运行代码之前使用jslint检查潜在的错误(这些错误一般不会导致语法错误,但是可能在运行时出现难以跟踪的问题)。

相关文章
|
15天前
|
存储 JavaScript 前端开发
解释 JavaScript 中的作用域和作用域链的概念。
【4月更文挑战第4天】JavaScript作用域定义了变量和函数的可见范围,静态决定于编码时。每个函数作为对象拥有`scope`属性,关联运行期上下文集合。执行上下文在函数执行时创建,定义执行环境,每次调用函数都会生成独特上下文。作用域链是按层级组织的作用域集合,自内向外查找变量。变量查找遵循从当前执行上下文到全局上下文的顺序,若找不到则抛出异常。
20 6
|
2月前
|
JavaScript 前端开发 网络协议
​Node.js 教程(一) 基本概念与基本使用
​Node.js 教程(一) 基本概念与基本使用
|
3月前
|
JavaScript 前端开发 Java
JavaScript基础语法(运算符)
JavaScript基础语法(运算符)
32 0
|
26天前
|
JavaScript
js开发:请解释什么是ES6的扩展运算符(spread operator),并给出一个示例。
ES6的扩展运算符(...)用于可迭代对象展开,如数组和对象。在数组中,它能将一个数组的元素合并到另一个数组。例如:`[1, 2, 3, 4, 5]`。在对象中,它用于复制并合并属性,如`{a: 1, b: 2, c: 3}`。
12 3
|
1月前
|
Web App开发 JavaScript 前端开发
深入浅出:Node.js 在后端开发中的应用与实践
【2月更文挑战第13天】本文旨在探讨Node.js这一流行的后端技术如何在现代Web开发中被应用以及它背后的核心优势。通过深入分析Node.js的非阻塞I/O模型、事件驱动机制和单线程特性,我们将揭示其在处理高并发场景下的高效性能。同时,结合实际开发案例,本文将展示如何利用Node.js构建高性能、可扩展的后端服务,以及在实际项目中遇到的挑战和解决方案。此外,我们还将讨论Node.js生态系统中的重要工具和库,如Express.js、Koa.js等,它们如何帮助开发者快速搭建和部署应用。通过本文的探讨,读者将获得对Node.js在后端开发中应用的深入理解,以及如何有效利用这一技术来提升开发效率
|
1月前
|
JavaScript 前端开发 程序员
编程笔记 html5&css&js 074 Javascript 运算符
编程笔记 html5&css&js 074 Javascript 运算符
|
1月前
|
移动开发 前端开发 JavaScript
编程笔记 html5&css&js 002 一些基本概念
编程笔记 html5&css&js 002 一些基本概念
|
2月前
|
JavaScript 前端开发 UED
JavaScript中的事件委托机制及实践应用
事件委托是JavaScript中常用的性能优化技巧,通过将事件监听器绑定在父元素上,实现对子元素事件的统一管理,减少页面中事件处理函数的数量,提升页面性能。本文将介绍事件委托的原理和实践应用,帮助开发者更好地理解和运用这一技术。
|
2月前
|
Rust 前端开发 JavaScript
Rust与JavaScript的跨语言交互:探索与实践
本文旨在探讨Rust与JavaScript之间的跨语言交互方法。我们将深入了解WebAssembly(Wasm)的角色,以及它如何使得Rust与JavaScript能够在Web应用中和谐共处。此外,我们还将介绍Rust与JavaScript的集成方式,包括Rust编译到Wasm、使用wasm-bindgen进行Rust与JavaScript的绑定,并通过实际案例展示如何实现两者之间的交互。
|
2月前
|
存储 JavaScript 前端开发
JavaScript基础:了解核心概念和技能
JavaScript基础:了解核心概念和技能
55 1