我们这样来定义类型:类型是值的内部特征,它定义了值的行为,以使其区别于其他值
JS有其中内置类型:null、undefined、boolean、number、string、object、symbol(ES6新增),除了对象之外,其余统称为“基本类型”
可以使用typeof运算符来查看值的类型,但是null是一个例外
typeof null === "object" // true
在函数对象中,length属性是其声明的参数的个数,数组也是对象,是object的一个“子类型”
function a(b,c){
...
}
a.length // 2
JS中的变量是没有类型的,只有值才有。变量可以随时持有任何类型的值
typeof运算符总是会返回一个字符串
已在作用域中声明但还没有赋值的变量,是undefined。相反,还没有在作用域中声明过的变量,是undeclared的
var a ;
a // undefined
b // ReferenceError: b is not defined
对于这类情况的处理让人抓狂,如果对于b可以返回undeclared的话,情况会好很多
JS有 七 种 内 置 类 型:null、undefined、boolean、number、string、object 和symbol,可以使用 typeof 运算符来查看。
变量没有类型,但它们持有的值有类型。类型定义了值的行为特征。
很 多 开 发 人 员 将 undefined 和 undeclared 混 为 一 谈, 但 在JS中 它 们 是 两 码 事。undefined 是值的一种。undeclared 则表示变量还没有被声明过。遗憾的是,JavaScript 却将它们混为一谈,在我们试图访问 “undeclared” 变量时这样报错:ReferenceError: a is not defined, 并 且 typeof 对 undefined 和 undeclared 变 量 都 返 回"undefined"。
然而,通过 typeof 的安全防范机制(阻止报错)来检查 undeclared 变量,有时是个不错的办法。
在JS中,数组可以容纳任何类型的值,可以是字符串、数组、对象,甚至是其他数组
使用delete运算符可以将单元从数组中删除,但是删除后,数组的length属性并不会发生变化
var a = [ ];
a[0] = 1;
a["foobar"] = 2;
a.length; // 1
a["foobar"]; // 2
a.foobar; // 2
数组通过数字进行索引,但是有趣的是他们也是对象,所以他们也可以包含字符串键值和属性
可以使用数组工具函数(indexOf(…)、concat(…)、forEach(…))将类数组转换位真正的数组,也可以使用Array.from()
const arrayLike = { 0: 'a', 1: 'b', 2: 'c', length: 3 };
const arr = Array.from(arrayLike);
console.log(arr); // ['a', 'b', 'c']
const arrayLike = { 0: 'a', 1: 'b', 2: 'c', length: 3 };
const arr = Array.prototype.slice.call(arrayLike);
console.log(arr); // ['a', 'b', 'c']
字符串与数组类似,,他们都是类数组,都有length属性以及indexOf(…)和concat(…)方法,但是在JS中字符串是不可变的,而数组是可变的,在进行字符串反转时,我们可以如下来做:
var c = a
.split("") // 将a的值转换为字符数组
.reverse() // 将数组中的字符进行倒转
.join("") // 将字符串中的字符拼接回字符串
JS只有一种数值类型,包括“整数”和带小数的十进制数
JS中的数字类型是基于IEEE754标准来实现的,该标准通常也被称为“浮点数”,JS使用的是“双精度”格式(64位二进制)
特别大和特别小的数字默认用指数格式显示,与toExponential()函数(用于将数字转换为指数形式的字符串)的输出结果一致
const num = 123.456789;
console.log(num.toExponential()); // 输出:1.23456789e+2
console.log(num.toExponential(2)); // 输出:1.23e+2
console.log(num.toExponential(6)); // 输出:1.234568e+2
tofixed(…)方法可指定小数部分的显示位数
var a = 42.59;
a.toFixed( 0 ); // "43"
a.toFixed( 1 ); // "42.6"
a.toFixed( 2 ); // "42.59"
a.toFixed( 3 ); // "42.590"
a.toFixed( 4 ); // "42.5900"
toPrecision(…) 方法用来指定有效数位的显示位数
var a = 42.59;
a.toPrecision( 1 ); // "4e+1"
a.toPrecision( 2 ); // "43"
a.toPrecision( 3 ); // "42.6"
a.toPrecision( 4 ); // "42.59"
a.toPrecision( 5 ); // "42.590"
a.toPrecision( 6 ); // "42.5900"
上边的方法不仅适用于数字变量,也适用于数字字面量。不过对于.运算符需要给予特别注意,因为它是一个有效数字字符,会被优先识别为数字字面量的一部分,然后才是对象属性访问运算符
// 无效语法:
42.toFixed( 3 ); // SyntaxError
// 下面的语法都有效:
(42).toFixed( 3 ); // "42.000"
0.42.toFixed( 3 ); // "0.420"
42..toFixed( 3 ); // "42.000"
在JS中0.1+0.2!=0.3,因为他是浮点数,我们可以设置一个误差范围值,该值定义在Number.EPSILON中,可以使用它来比较两个数字是否相等
function numbersCloseEnoughToEqual(n1,n2) {
return Math.abs( n1 - n2 ) < Number.EPSILON;
}
var a = 0.1 + 0.2;
var b = 0.3;
numbersCloseEnoughToEqual( a, b ); // true
numbersCloseEnoughToEqual( 0.0000001, 0.0000002 ); // false
如果要检测一个值是否为整数,可以使用ES6中的Number.isInteger(…)方法
Number.isInteger( 42 ); // true
Number.isInteger( 42.000 ); // true
Number.isInteger( 42.3 ); // false
null是一个特殊关键字,不是标识符,不能将其当做变量来使用,尽管 null 可以被赋值给变量,但它本身并不是变量,也不应该被当做变量来使用。当我们需要表示一个空的对象引用时,才将 null 赋值给变量,并使用该变量来表达空值的概念
undefined是一个标识符,可以被当做变量来使用和赋值
void运算符,void___ 没有返回值,因此返回结果是undefined,void并不改变表达式的结果,只是让表达式不返回值,如果要将代码中的值设为undefined,就可以使用void
var a = 42;
console.log( void a, a ); // undefined 42
NaN意指“不是一个数字”,是一个特殊值,它和自身不相等,是唯一一个非自反的值,因此无法对其进行比较,那么应该如何判断它,可以使用isNaN,该方法意思是“参数是否不是NaN,也不是数字”
var a = 2 / "foo";
isNaN( a ); // true
我们也可以使用下边的方法
var a = 2 / 'foo';
var b = 'foo';
console.log(Number.isNaN(a)); // true
console.log(Number.isNaN(b)); // false
无穷数Infinity
在JS中var a = 1 / 0
的结果为Infinity,如果该式中一个数为负数,结果即为-Infinity
JS中,零值分为+0和-0,为什么我们需要-0,因为某些程序中的数据需要以级数形式来表示(比如动画帧的移动速度)
在ES6中,新加入了一个工具方法Object.is(…)来判断两个值是否相等,但是能使用“”和“=”的时候就尽量不要使Object.is(…),因为前者的效率更高也更为通用
在许多语言中,复制和参数传递可以通过值复制或者引用复制来完成。引用就像一种特殊的指针,是来指向变量的指针,但是在JS中没有指针,引用的工作机制也不尽相同。在JS中变量不可能成为指向另一个变量的引用。JS引用指向的是值。如果一个值有10个引用,这些引用指向的都是同一个值,它们之间没有引用/指向关系。
简单值(基本类型值)总是通过值复制的方式来赋值/传递,包括null、undefined、字符串、数组、布尔和ES6中的symbol,复合值—对象和函数,则总是通过引用复制的方式来赋值/传递。
引用指向的是值本身而非变量,所以一个引用无法更改另一个引用的指向
function foo(x) {
x.push( 4 );
x; // [1,2,3,4]
// 然后
x = [4,5,6];
x.push( 7 );
x; // [4,5,6,7]
}
var a = [1,2,3];
foo( a );
a; // 是[1,2,3,4],不是[4,5,6,7]
我们不能通过引用 x 来更改引用 a 的指向,只能更改 a 和 x 共同指向的值。
如果要将a的值变为[4,5,6,7],必须更改x指向的数组,而不是为x赋值一个新数组
function foo(x) {
x.push( 4 );
x; // [1,2,3,4]
// 然后
x.length = 0; // 清空数组
x.push( 4, 5, 6, 7 );
x; // [4,5,6,7]
}
var a = [1,2,3];
foo( a );
a; // 是[4,5,6,7],不是[1,2,3,4]
我们无法自行决定使用值复制还是引用复制,一切由值的类型来决定。
JS中的数组是通过数字索引的一组任意类型的值。字符串和数组类似,但是它们的行为特征不同,在将字符作为数组来处理时需要特别小心。JavaScript 中的数字包括“整数”和“浮点型”。
基本类型中定义了几个特殊的值。null 类型只有一个值 null,undefined 类型也只有一个值 undefined。所有变量在赋值之前默认值都是 undefined。void 运算符返回 undefined。
数 字 类 型 有 几 个 特 殊 值, 包 括 NaN( 意 指“not a number”, 更 确 切 地 说 是“invalidnumber”)、+Infinity、-Infinity 和 -0。
简单标量基本类型值(字符串和数字等)通过值复制来赋值 / 传递,而复合值(对象等)通过引用复制来赋值 / 传递。JavaScript 中的引用和其他语言中的引用 / 指针不同,它们不能指向别的变量 / 引用,只能指向值。