基本语法
1.严格模式 "use strict"
作用
- 消除JS语法的一些不合理、不严谨、不安全的问题,减少怪异行为并保证代码运行安全
- 提高编译器解释器效率,增加运行速度
与标准模式的区别
- 隐式声明或定义变量:严格模式不能通过省略
var
关键字隐式声明变量。会报引用错误“ReferenceError:abc is not define” - 对象重名的属性:严格模式下对象不允许有重名的属性。
var obj = {a:1, b:2, a:3}
会报语法错误“SyntaxError”。 -
arguments.callee
:通常我们使用这个语法来实现匿名函数的递归,但在严格模式下是不允许的。会报错TypeError。 -
with
语句:严格模式下with语句是被禁用的。会报语法错误 SyntaxError 。
2.注释/* */
不可嵌套
类型系统
3.基本类型(标准类型)
主要介绍6种基本类型(Undefine、Null、Boolean、Number、String、Object)、原生类型及引用类型概念
原始数据类型:Undefined、Null、Boolean、String、Number
值存于栈内存(stack)中。占据的空间小,大小固定,频繁被使用。
引用数据类型:Object
值存于堆内存(heap)中。栈中只保留了一个指针,指向堆中存储位置的起始地址。占据空间大,大小不固定。
4.Undefined
出现场景
向其他数据类型转换
Boolean:false
Number:NaN
String:"undefined"
5.Null
出现场景
- null表示对象不存在 document.getElementById("notExitElement");
向其他数据类型转换
Boolean:false
Number:0
String:"null"
6.Boolean : true false
出现场景
- 条件语句导致系统执行的隐式类型转换
if(document.getElementById("notExitElement");){...}
- 字面量或变量定义:
true
,var a = true;
只有以下6个值会被转换为false:
undefined
null
false
0
NaN
""或''(空字符串)
注意:空数组[]
和空对象{}
,对应的布尔值都是true。
向其他数据类型转换
Number:1 0
String:"true" "false"
7.String
出现场景
"abbv" 'lll'
向其他数据类型转换
String: "" "123" "notEmpty"
Number: 0 123 NaN
Boolean:false true true
8.Number
出现场景
-
123
var a = 1;
向其他数据类型转换
Number: 0 123 NaN Infinity
Boolean:false true true false
String: "0" "123" "NaN" "Infinity"
十进制:没有前导0的数值。
八进制:有前缀0o或0O的数值,或者有前导0、且只用到0-7的七个阿拉伯数字的数值。
十六进制:有前缀0x或0X的数值。
二进制:有前缀0b或0B的数值。
9.Object
一组属性的集合。
出现场景
{a: "value-a", b: "value-b", ...}
向其他数据类型转换
Object: {}
Number: NaN
Boolean: true
String: "[object Object]"
类型识别
主要介绍typeof、Object.prototype.toString、constructor、instanceof等类型识别的方法。
10.typeof
对标准数据类型的识别
typeof 1 // number
typeof true // boolean
typeof "str" // string
typeof undefined // undefined
typeof null // object
typeof {} // object
也就是说,标准数据类型中,除了null被识别为object,其余类型都识别正确。
另外,需要注意typeof undefined
会得到 undefined
,利用这一点,typeof可用于检查一个没有声明的变量而不报错:
if(a) { ... }
//Uncaught ReferenceError: a is not defined
if(typeof a === 'undefined') { console.log('未定义也不报错') }
//未定义也不报错
对具体对象的识别
typeof function(){} // function
typeof [] // object
typeof new Date(); // object
typeof /\d/; // object
function Person() {};
typeof new Person; // object
因此,不能识别具体的对象类型,函数对象除外。
11.Object.prototype.toString
Object.prototype.toString.call(1); // "[object Number]"
所以我们写一个函数来简化调用,以及截取我们需要的部分:
function type(obj){
return Object.prototype.toString.call(obj).slice(8, -1).toLowerCase();
}
type(1) // number
type("str") // string
type(true) // boolean
type(undefined) //undefined
type(null) //null
type({}) //object
type([]) //array
type(new Date) // date
type(/\d/) // regexp
type(function(){}) // function
Object.prototype.toString可以准确识别出所有的标准类型
以及内置(build-in)对象类型
。
那么自定义类型
呢?
function Point(x, y){
this.x = x;
this.y = y;
}
type(new Point(1, 2)) //object
所以它无法识别出自定义类型
。
12.constructor
对象原型上的一个属性,它指向了构造器本身。
可以识别原始数据类型
(undefined
和null
除外,因为它们没有构造器),内置对象类型
和自定义对象类型
,尤其注意自定义对象类型(如:Person)。
//判断原始类型
"Jerry".constructor === String //true
(1).constructor === Number //true
true.constructor === Boolean //true
({}).constructor === Object //true
//判断内置对象
[].constructor === Array //true
//判断自定义对象 ヾ(o◕∀◕)ノ
function Person(name){
this.name = name;
}
new Person("Jerry").constructor === Person // true ヾ(o◕∀◕)ノ !!!
13.instanceof
//判断原始类型
1 instanceof Number // false
true instanceof boolean // false
//判断内置对象类型
[] instanceof Array // true
/\b/ instanceof RegExp // true
//判断自定义对象类型
function Person(name){
this.name = name;
}
new Person("miao") instanceof Person // true
不能判断原始类型
,可以判断内置对象类型
,可以判断自定义对象类型及父子类型
。
内置对象
分为两类:普通对象
与构造器对象
。其中,构造器对象可用于实例化普通对象。
普通对象只有自身的属性和方法。
构造器对象除了自身的属性和方法之外,还有原型对象prototype上的属性和方法,以及实例化出来的对象的属性和方法。
以下代码创建了一个Point构造器并实例化了一个p对象
function Point(x, y){
this.x = x;
this.y = y;
}
Point.prototype.move = function(x, y){
this.x += x;
this.y += y;
}
var p = new Point(1,1);
p.move(1,2);
普通对象p的原型链:
"__proto__"就是我们通常说的原型链属性,他有如下几个特点:
- "__proto__"是对象的一个内部隐藏属性。
- "__proto__"是对实例化该对象的构造器的
prototype属性
的一个引用,因此可以访问prototype的所有属性和方法。- 除了Object对象,每个对象都有一个"__proto__"属性,"__proto__"逐级增长形成一个链就是我们所说的原型链,原型链顶端是一个Object对象。
- 当开发者调用对象属性或方法时(比如"p.move(1,2)"),引擎首先会查找p对象的自身属性,如果自身属性中没有"move"方法,则会继续沿原型链逐级向上查找,直到找到该方法并调用。
- "__proto__"跟浏览器引擎实现相关,不同的引擎中名字和实现不尽相同(chrome、firefox中名称是"__proto__",并且可以被访问到,IE中无法访问)。基于代码兼容性、可读性等方面的考虑,不建议开发者显式访问"__proto__"属性或通过"__proto__"更改原型链上的属性和方法,可以通过更改构造器prototype对象来更改对象的__proto__属性。
构造器对象的原型链:
Point.prototype
其实就是个普通对象:
Point.prototype = {
move: function(){ ... },
constructor: function(){ ... }
}
构造器对象相对于普通对象有如下几个特点:
1.构造器对象原型链上 倒数第二个__proto__
是一个Function.prototype
对象引用,因此可以调用Function.prototype
的属性和方法。
2.构造器对象本身有一个prototype
属性, 有这个属性就表明该对象可以用来生成其他对象:用该构造器实例化对象时该prototype
会被实例对象的__proto__
所引用,即添加到实例对象的原型链上。
3.构造器对象本身是一个function
对象,因此会有name,length
等自身属性。
14.Object对象
对象的所有键名都是字符串,键名加不加引号都可以。
用于创建对象的下面三行语句是等价的:
var o1 = {};
var o2 = new Object();
var o3 = Object.create(Object.prototype);
String/Number/Boolean/Array/Date/Error
构造器都是Object子类对象。
自身的属性、方法:prototype,create,keys,getOwnPropertyNames,getOwnPropertyDescriptor,getPrototypeOf
原型对象prototype的属性和方法:constructor,toString,valueOf,hasOwnProperty,isPrototypeOf
生成的实例对象只有原型链属性__proto__,没有其他的属性,也就是Object
没有实例对象属性,方法
。相比较而言,Array
有实例对象属性length
,也就是说,Arrya类型的实例对象有length
属性。
几个重要方法:
Object.create(proto)
: 基于原型对象创建新对象,传入的是一个对象,会被作为创建出的对象的__proto__属性值。
Object.keys(obj)
:返回一个由obj自身所拥有的可枚举属性的属性名组成的数组。比如:
var obj = { a: 'aaa', b: 'bbb'};
Object.keys(obj); // ["a", "b"]
var arr = ['a', 'b', 'c'];
Object.keys(arr); // ["0", "1", "2"]
如果想要对象上不可枚举的属性也被列举出来,用Object.getOwnPropertyNames(obj)
方法。比如:
var arr = ['a', 'b', 'c'];
Object.getOwnPropertyNames(arr); // ["0", "1", "2", "length"]
Object.getOwnPropertyDescriptor(obj, 'prop')
可以读出对象自身属性的属性描述对象。
var obj = { prop1: 'hello'};
Object.getOwnPropertyDescriptor(obj, 'prop1');
//Object {value: "a", writable: true, enumerable: true, configurable: true}
Object.getPrototypeOf(obj)
获取对象的Prototype对象。
Object.prototype.valueOf
:返回指定对象的原始值
这个方法几乎不会手动调用,而是JavaScript自动去调用将一个对象转换成原始值(primitive value)。
默认情况下,每一个内置对象都会覆盖这个方法返回一个合理的值。如果对象没有原始值,这个方法就会返回对象自身。比如:
//有原始值
var num= new Number(3);
num; // {[[PrimitiveValue]]: 3}
num.valueOf(); // 3
num + 4; // 7 valueOf()被JavaScript隐式调用了
//没有原始值
var obj = {'a': 'aaa'}
obj.valueOf(); // {'a': 'aaa'}
假设你有一个对象类型的myNumberType,你想为它创建一个valueOf方法。下面的代码可以自定义一个valueOf方法:
function myNumberType (n){
this.number = n;
}
myNumberType.prototype.valueOf = function () {
return this.number;
}
var obj = new myNumberType(4);
obj + 3; // 7 自定义的valueOf方法被JavaScript自动调用了
Object.prototype.toString
:获取方法调用者的标准类型
默认情况下,toString() 方法被每个继承自Object的对象继承。如果此方法在自定义对象中未被覆盖,toString() 返回 "[object type]",其中type是对象类型。以下代码说明了这一点:
var obj = {a: 1};
obj.toString(); // "[object Object]"
很多内置对象重写了toString方法。包括Array,Boolean,Number,Date,String例如:
var a = new Number(1)
b = a.toString(); // "1"
//可用于进行进制转换
var num1 = 10;
num1.toString(16); // "a"
num1.toString(2); // "1010"
num1.toString(8); // "12"
var c = new String('abc')
d = c.toString(); // "abc"
var e = new Array(['a', 'b', 'c'])
f = e.toString(); // "a,b,c"
还可以用来检查对象类型:
var toString = Object.prototype.toString;
toString.call(new Date); // [object Date]
toString.call(new String); // [object String]
toString.call(Math); // [object Math]
toString.call(1); // [object Number]
toString.call('1'); // [object String]
//Since JavaScript 1.8.5
toString.call(undefined); // [object Undefined]
toString.call(null); // [object Null]
Object.prototype.hasOwnProperty
:判断一个属性是对象自身属性,还是原型上的属性。
var obj = Object.create({a:1});
obj.b = 2;
obj.hasOwnProperty('a'); //false
obj.hasOwnProperty('b'); //true
Object.property.isPrototypeOf
:判断当前对象是否为另一个对象的原型。
var father = { a: 'aaa', b: 'bbb'};
var child = Object.create(father);
father.isPrototypeOf(child); // true
delete
命令用于删除对象的属性,不管该属性是否存在,删除后都返回true
,只有当该属性的configurable: false
时,删除操作才返回false,也就是该属性不得删除。
var o = { p: 'Hello'}
delete o.p;
o; // Object{}
注意:delete
不得删除var定义的变量。
in
运算符用于检查对象是否包含某个属性(这里写的是属性的键名)。存在返回true
。可以用于判断全局变量是否存在。
var o = {p: 'hello'}
'p' in o; // true
'o' in window; // true
存在的问题是:in
运算符对于继承的属性也返回true
。
for ... in
循环用来遍历一个对象的全部属性。
var o = { a: 'hello', b: 'world'};
for( var key in o) {
console.log(key + ': ' + o[key]);
}
//a: hello
//b: world
注意:会遍历继承来的属性。所以一般不推荐使用。一般会用Object.keys(obj)
,可以只遍历对象本身的属性。
15.String, Number, Boolean
使用String()
和Number()
方法可以将各种类型的值强制转换为string
或number
类型。比如:
Number('222'); //222
Number(null); // 0
Number(undefined); // NaN
String(new Number(2)); // '2'
String(['a', 'b']); //'a,b'
但当转换的数值为对象时,转换规则为:
Number(obj)
:(简单地说值为NaN)
1.先调用对象自身的valueOf()方法,如果返回原始类型的值,那么直接对该值使用Number函数,不再进行后续步骤。
2.如果valueOf方法返回的值类型是对象,那么调用这个对象的toString()方法,如果返回的是原始类型的值,那么对这个值使用Number()方法,不再进行后续步骤。
3.如果toString返回的是对象,就报错。
var obj = {a: 1};
Number(obj); // NaN
//等同于
if(typeof obj.valueOf() === 'object') {
Number(obj.toString());
} else {
Number(obj.valueOf());
}
String(obj)
:
1.先调用对象自身的toString()方法,如果返回原始类型的值,那么直接对该值使用String函数,不再进行后续步骤。
2.如果toString方法返回的值类型是对象,那么调用这个对象的valueOf()方法,如果返回的是原始类型的值,那么对这个值使用String()方法,不再进行后续步骤。
3.如果valueOf返回的是对象,就报错。
var obj = {a: 1}
String(obj); // [object Object]
//等价于
if(typeof obj.toString() === 'object') {
String(obj.valueOf());
} else {
String(obj.toString());
}
String
构造器对象属性、方法:prototype, fromCharCode
原型对象属性、方法:constructor,indexOf,replace,slice,charCodeAt,toLowerCase
几个重要方法:
1.String.prototype.indexOf
:获取子字符串在字符串中位置索引(一次只查找一个)
语法:strObj.indexOf(searchvalue,fromindex)
var str = "abcdabcd";
var idx = str.indexOf("b"); // 1
var idx2 = str.indexOf("b", idx+1); //5
2.String.prototype.replace
:查找字符串替换成目标字符(一次只替换一个)
语法:strObj.replace(regexp/searchvalue, newSubStr/function)
var str = "1 plus 1 equal 3";
str = str.replace("1","2"); // "2 plus 1 equal 3"
str = str.replace(/\d+/g, "$& dollar"); //"2 dollar plus 1 dollar equal 3 dollar"
第二个参数是newSubStr
时,其中
$& 代表插入当前匹配的子串
$` 代表插入当前匹配的子串左边的内容
$' 代表插入当前匹配的子串右边的内容
$n 是当第一个参数为regexp对象,且n为小于100的非负整数时,插入regexp第n个括号匹配到的字符串
第二个参数是function
时,函数的返回值为替换的字符串,注意如果第一个参数为regexp且为全局匹配,那么每次匹配都会调用这个函数,把匹配到的值用函数的返回值替换。函数参数:
match:匹配的子串,相当于 $&
p1,p2,...: 如果第一个参数为regexp,对应第n个括号匹配到的字符串,相当于$1,$2
offset: 匹配到的字符串到原字符串中的偏移量,比如 bc 相对于 abcd ,offset 为1
string: 被匹配的原字符串
//精确的参数个数取决于第一个参数是否为regexp,以及有多少个括号分组
3.String.prototype.split
:按分割符将字符串分割成字符串数组
语法:strObj.split(separator, howmany)
var str = "1 plus 1 equal 3";
str.split(" "); // ["1", "plus", "1", "equal", "3"]
str.split(" ", 3); // ["1", "plus", "1"]
str.split(/\d+/); //["", " plus ", " equal ", ""]
4.str.slice(beginSlice[, endSlice])
提取一个字符串的一部分,并返回一个新的字符串
var str = 'abcdefg';
str.slice(1,3); // 'bc'
str; //'abcdefg'
5.str.substr(start[, length])
返回从start开始,length个数的字符串
var str = 'abcdefg';
str.substr(1,3); //'bcd'
str; //'abcdefg'
6.str.substring(start[, end])
返回从start开始到end(不包括end)结束的字符串。其中,start和end都大于0.
var str = 'abcdefg';
str.substring(1,3); //'bc'
str; //'abcdefg'
7.str.trim()
返回的是一个新字符串,删除了原字符串两端的空格
var str = ' abc ';
str.trim(); // 'abc'
str; // ' abc '
8.str.charAt(index)
从一个字符串中返回index位置的字符,indx默认是0
var str = 'abcdefg';
str.charAt(1); //'b'
9.str.concat(string2, string3[, ..., stringN])
返回新字符串,将原字符串与一个或多个字符串相连
var str = 'abc';
var str2 = '123';
str.concat(str2);// 'abc123'
强烈建议使用 赋值操作符(+, +=)代替 concat 方法。
10.str.includes(searchString[, position])
判断searchString是否包含在str中,返回布尔值。
var str = 'abc';
str.includes('ab'); //true
11.str.indexOf(searchValue[, fromIndex])
返回str中searchValue第一次出现的位置,没找到返回-1
"Blue Whale".indexOf("Blute"); // returns -1
"Blue Whale".indexOf("Whale", 0); // returns 5
12.str.lastIndexOf(searchValue[, fromIndex])
返回str中searchValue最后一次出现的位置,没找到返回-1
"abcabc".lastIndexOf("a") // returns 3
"abcabc".lastIndexOf("d") // returns -1
"abcabc".lastIndexOf("a",2) // returns 0
"cbacba".lastIndexOf("a",0) // returns -1
13.str.endsWith(searchString [, position]);
属于ES6.判断searchString是否是以另外一个给定的字符串结尾的,返回布尔值
var str = "To be, or not to be, that is the question.";
str.endsWith("question."); // true
str.endsWith("to be"); // false
str.endsWith("to be", 19); // true
str.endsWith("To be", 5); // true
14.str.match(regexp);
当一个字符串与一个正则表达式匹配时,match()方法检索匹配项。 返回一个包含了整个匹配结果以及任何括号捕获的匹配结果的 Array ;如果没有匹配项,则返回 null 。
如果正则表达式没有 g 标志,则 str.match() 会返回和 RegExp.exec() 相同的结果。
如果正则表达式包含 g 标志,则该方法返回一个 Array ,它包含所有匹配的子字符串而不是匹配对象。
var str = 'For more information, see Chapter 3.4.5.1';
var re = /see (chapter \d+(\.\d)*)/i;
var found = str.match(re);
console.log(found);
// logs [ 'see Chapter 3.4.5.1',
// 'Chapter 3.4.5.1',
// '.1',
// index: 22,
// input: 'For more information, see Chapter 3.4.5.1' ]
// 'see Chapter 3.4.5.1' 是整个匹配。
// 'Chapter 3.4.5.1' 被'(chapter \d+(\.\d)*)'捕获。
// '.1' 是被'(\.\d)'捕获的最后一个值。
// 'index' 属性(22) 是整个匹配从零开始的索引。
// 'input' 属性是被解析的原始字符串。
15.str.search(regexp)
执行正则表达式和String对象之间的一个搜索匹配。如果匹配成功,则 search() 返回正则表达式在字符串中首次匹配项的索引。否则,返回 -1。
下例记录了一个消息字符串,该字符串的内容取决于匹配是否成功。
function testinput(re, str){
var midstring;
if (str.search(re) != -1){
midstring = " contains ";
} else {
midstring = " does not contain ";
}
console.log (str + midstring + re);
}
16.str1.localeCompare(str2)
比较两个字符串,返回一个整数,如果小于0,表示第一个字符串小于第二个字符串;如果大于0,表示第一个字符串大于第二个字符串。
'apple'.localeCompare('banana'); //-1
该方法最大的特点就是会考虑语言的顺序。看下面的例子:
'B' > 'a' // false 因为Unicode编码中,大写字母都在小写字母前面:B的码点是66,而a的码点是97。
'B'.localeCompare('a'); //1 考虑了语言的自然顺序
Number
Number.prototype.toFixed()
用于将一个数转换为指定位数(有效范围为0到20)的小数,返回这个小数对应的字符串。
var a = 1.234
a.toFixed(2);//1.23
Number.prototype.toExponential()
用于将一个数转为科学计数法形式。参数为保留的小数位数,有效范围0-20。
(1234).toExponential() // "1.234e+3"
(1234).toExponential(1) // "1.2e+3"
(1234).toExponential(2) // "1.23e+3"
Number.prototype.toPrecision()
用于将一个数字转为指定位数的有效数字。参数为有效数字的位数,范围1-21。
(12.34).toPrecision(1) // "1e+1"
(12.34).toPrecision(2) // "12"
(12.34).toPrecision(4) // "12.34"
(12.34).toPrecision(5) // "12.340"
Boolean
记住:
Boolean(undefined) // false
Boolean(null) // false
Boolean(0) // false
Boolean('') // false
Boolean(NaN) // false
Boolean(1) // true
Boolean('false') // true
Boolean([]) // true
Boolean({}) // true
Boolean(function () {}) // true
Boolean(/foo/) // true
16.Array
构造器对象(自身)属性、方法:prototype,Array.isArray(obj)
原型对象属性、方法:constructor,
- 1.改变原数组: splice,push,pop[删除最后一个],shift[删除第一个],unshift[添加到第一个],reverse[反转],sort
- 2.不改变原数组: concat,slice,indexOf,lastIndexOf
- 3.对数组进行遍历的方法:every,filter,map,forEach,reduce
实例对象属性、方法:length
length
属性是可写的,如果设置length到小于数组当前长度,那么数组长度会缩短。
因此,将数组清空的一个有效方法,就是将数组的length设置为0。举例如下:
var arr = [1,2,3];
arr.length = 0;
arr; []
当length属性设为大于数组个数时,读取新增的位置都会返回undefined
。
注意:只要有length的对象就是类数组对象。典型的类似数组的对象是函数的arguments对象,以及大多数DOM元素集,还有字符串。
将类数组对象转化为对象的方法是:var arr = Array.prototype.slice.call(arrayLike);
。
几个常用对象方法:
- Array.prototype.sort:排序,并返回数组
语法: arr.sort([compareFunction])
如果compareFunction(a, b)返回值小于0,a在b前;返回值大于0,a在b后。
比较函数格式如下:
function compare(a, b) {
if (a is less than b by some ordering criterion) {
return -1;
}
if (a is greater than b by the ordering criterion) {
return 1;
}
// a must be equal to b
return 0;
}
希望比较数字而非字符串,比较函数可以简单的以 a 减 b,如下的函数将会将数组升序排列
function compareNumbers(a, b) {
return a - b;
}
var fruit = ['cherries', 'apples', 'bananas'];
fruit.sort();
// ['apples', 'bananas', 'cherries']
var scores = [1, 10, 21, 2];
scores.sort();
// [1, 10, 2, 21]
// 注意10在2之前,
// because '10' comes before '2' in Unicode code point order.
var things = ['word', 'Word', '1 Word', '2 Words'];
things.sort();
// ['1 Word', '2 Words', 'Word', 'word']
// 在Unicode中, 数字在大写字母之前,
// 大写字母在小写字母之前.
var numbers = [4, 2, 5, 1, 3];
numbers.sort(function(a, b) {
return a - b;
});
console.log(numbers);
// [1, 2, 3, 4, 5]
2.Array.prototype.splice(start[, deleteCount, item1, item2, ...])
:从数组中添加、删除或替换元素,并返回被删除的元素列表。会更改原数组。如果deleteCount
被省略,则其相当于(arr.length - start)
。
var arr = [1,2,3,4];
let deleteArr = arr.splice(1, 2, 'a', 'b');
deleteArr; //[2, 3]
arr; // [1, 'a', 'b', 4]
3.Array.prototype.forEach:遍历数组元素并调用回调函数
语法:arr.forEach(callback[,thisArg])
function callback(value, index, array){ ... }
var arr = ["a", ,"b"];
arr.forEach(function(value, index, array){ console.log(value);})
//"a"
//"b"
//空元素被跳过去了哦 (,• ₃ •,)
注意:forEach
方法无法中断执行,总会将所有成员遍历完,如果希望符合某种条件就中断执行,需要使用 for
循环。
4.arr.filter(callback)
返回一个新数组,包含通过了callback函数中的过滤条件的所有元素
语法:
let newArr = arr.filter(function(element, index, array){ ... })
function isBigEnough(value) {
return value >= 10;
}
var filtered = [12, 5, 8, 130, 44].filter(isBigEnough);
// filtered is [12, 130, 44]
5.arr.map(callback)
返回一个新数组,结果是该数组中的每个元素调用提供的callback函数之后的结果
语法:
let newArr = arr.map(function(element, index, array){ ... })
let numbers = [1, 5, 10, 15];
let roots = numbers.map(function(x, index, array) {
return x * 2;
});
// roots is now [2, 10, 20, 30]
// numbers is still [1, 5, 10, 15]
当需要返回值的时候,一般使用map
;不需要时,一般使用forEach
。
6.arr.every(callback)
测试数组的所有元素是否都通过了指定函数的测试,返回一个布尔值。
语法:
let passed = arr.every(function(element, index, array){ ... })
function isBigEnough(element, index, array) {
return (element >= 10);
}
var passed = [12, 5, 8, 130, 44].every(isBigEnough);
// passed is false
passed = [12, 54, 18, 130, 44].every(isBigEnough);
// passed is true
7.arr.concat(arr1[, arr2, ...])
合并两个或多个数组
let arr3 = arr1.concat(arr2);
8.arr.slice(begin, end)
返回一个从begin到end(不包括end)这之间的数组浅拷贝到一个新的数组对象。原始数组不会被修改。
var arr = [1,2,3,4];
let newArr = arr.slice(1,3);
arr; //[1,2,3,4];
newArr; //[2,3];
arr.slice(); // [1,2,3,4] 不加参数时返回原数组的拷贝
9.arr.indexOf(searchVal[, fromIndex = 0])
返回在数组中可以找到的给定元素的第一个索引,不存在返回-1
10.arr.lastIndexOf(searchVal[, fromIndex = arr.length - 1])
返回在数组中可以找到的给定元素的最后一个索引,不存在返回-1
11.arr.reduce()
从左到右依次处理数组的每个成员,最终累计为一个值。
方法的第一个参数是一个函数,这个函数接收4个参数:
1.累积变量:默认数组的第一个变量
2.当前变量:默认数组的第二个变量
3.当前位置(从0开始)
4.原数组
前两个参数是必须的。
求数组成员之和:
var arr = [1,2,3,4,5];
arr.reduce(function(x, y) {
console.log(x, y);
return x + y;
})
// 1 2
// 3 3
// 6 4
// 10 5
// 15
第一轮执行,x是数组的第一个成员,y是数组的第二个成员。从第二轮开始,x为上一轮的返回值,y为当前数组成员,直到遍历完所有成员,返回最后一轮计算后的x。
如果要对累积变量指定初值,可以把它放在reduce方法和reduceRight方法的第二个参数。
var arr = [1,2,3,4,5];
arr.reduce(function(x, y) {
console.log(x, y);
return x + y;
}, 10)
// 10 1
// 11 2
// 13 3
// 16 4
// 20 5
// 25
由于 reduce
方法依次处理每个元素,所以它实际上还可以用来搜索某个元素。比如,找出长度最长的数组元素:
function findLongest(entries) {
return entries.reduce(function (longest, entry){
return longest.length > entry.length ? longest : entry;
})
}
findLongest(['aaa', 'bbbb', 'cc'])
17.Function
采用function
命令声明函数时,整个函数会像变量声明一样被提升到代码头部。比如下面的代码不会报错:
f();
function f() { ... }
但是下面的代码却会报错:
f();
var f = function () { ... }
//相当于下面这样
var f;
f();
f = function(){ ... }
自身属性,方法:prototype
原型对象属性,方法:constructor,apply,call,bind
实例对象属性,方法:name
,length
,prototype
,arguments,caller
name
返回紧跟在 function
关键字之后的函数名。
length
返回函数定义中的参数个数。
toString()
返回函数的源码。f.toString()
几个常用对象方法:
1.Function.prototype.apply
:通过传入参数指定 函数调用者
和 函数参数(一个数组或类数组对象)
并执行该函数。
语法:funcObj.apply(thisArg[, argsArray])
thisArg
: 函数运行时指定的this值argsArray
: 一个数组或类数组对象
栗子:
(1) Object.prototype.toString.apply("123");// "[object String]"
(2) 使用apply可以允许你在本来需要遍历数组变量的任务中使用內建的函数,比如获得一个数组中最大的值:
var arr = [2, 3, 7, 1];
Math.max.apply(null, arr);// 7
//这等价于 Math.max(arr[0], arr[1], arr[2], arr[3])
(3) 还可以用于 使用任意一个对象(可以不是当前自定义对象的实例对象)
去调用自定义对象中定义的方法
function Point(x, y) {
this.x = x;
this.y = y;
}
Point.prototype.move = function(x1, y1) {
this.x += x1;
this.y += y1;
}
var p = new Point(0,0); //p是一个点
p.move(2,2); // p这个点的位置在x轴右移了2,y轴右移了2
//等价于
p.move.apply(p, [2,2]);
//现在我们有一个圆,原点位于(1,1),半径r=1
var c = {x: 1, y: 1, r: 1}
//要求x轴右移2,y轴右移1
p.move.apply(c, [2,1]);
2.Function.prototype.bind
:通过传入参数指定 函数调用者
和 函数参数(一个列表)
并返回该函数的引用而并不调用。
语法:funcObj.bind(thisArg[, args])
thisArg
: 函数运行时指定的this值args
: 一个列表(不是数组)
栗子:
(1)绑定函数调用的this值,避免错误的使用了将this指向全局作用域
this.x = 9;
var module = {
x: 81,
getX: function() { return this.x; }
};
module.getX(); // 返回 81
var retrieveX = module.getX;
retrieveX(); // 返回 9, 在这种情况下,"this"指向全局作用域
// 创建一个新函数,将"this"绑定到module对象
// 新手可能会被全局的x变量和module里的属性x所迷惑
var boundGetX = module.getX.bind(module);
boundGetX(); // 返回 81
(2)返回函数的引用而并不调用的另一个好处是:可以指定函数执行的时间。比如有的需求是,在过了一段时间后执行这个函数。
还是上面Point的例子,要求在1000ms以后移动圆:
var c = {x: 1, y: 1, r: 1}
//要求x轴右移2,y轴右移1
var circlemove = p.move.bind(c, 2, 1);
setTimeout(circlemove, 1000);
子类构造器
上面例子中,调用其他对象的方法
也可以通过继承的方法实现:
function Circle(x, y ,r) {
Point.apply(this, [x, y]);
this.radius = r;
}
//将Circle的原型对象指定为Point的实例对象,这样Circle.prototype上就有了move(x,y)方法
Circle.prototype = Object.create(Point.prototype);
//此时Circle.prototype.constructor属性还等于function Point(x,y),所以要手动改成function Circle(x,y,r)
Circle.prototype.constructor = Circle;
//还可以定义一些其他的方法
Circle.prototype.area = function () {
return Math.PI*this.radius*this.radius;
}
var c = new Circle(1,2,3);
c.move(2,2);//调用的是自己原型上从Point继承过来的方法
c.area();
构造器对象Circle的原型链:
实例对象c的原型链:
function作为普通函数,有3种调用方式:() , apply , call
函数参数的三个特点:
- 形参个数不一定等于实参个数
- 所有参数传递都是
值传递
,也就是说,参数传递都只是在栈内存中的操作。所以要特别小心引用类型的参数传递,可能会改变传进来的实参的值。 - 通过
参数类型检查
实现函数重载
18.RegExp、Date、Error
RegExp
构造方法:
- /pattern/flags
- new RegExp(pattarn[, flags]);
原型对象属性、方法:test
, exec
test
:使用正则表达式对字符串进行测试,并返回测试结果
语法:regObj.test(str)
var reg = /^abc/i;
reg.test('Abc123'); //true
reg.test('bc123');//false
Date
Date.now()
返回当前距离 1970年1月1日 00:00:00 UTC的毫秒数。
Date.parse()
解析日期字符串,返回距离待解析日期距离 1970年1月1日 00:00:00 UTC的毫秒数。
Date.parse('2017-10-10T14:48:00') // 1507618080000
Date.parse('2017-10-10') // 1507593600000
Date.parse('Aug 9, 1995') // 807897600000
Date.prototype.toString()
返回一个完整的日期字符串。
var d = new Date(2013, 0, 1);
d.toString()
// "Tue Jan 01 2013 00:00:00 GMT+0800 (CST)"
Date.prototype.toLocaleDateString()
返回一个字符串,代表日期的当地写法。
var d = new Date(2013, 0, 1);
d.toLocaleDateString()
// 中文版浏览器为"2013年1月1日"
// 英文版浏览器为"1/1/2013"
Date.prototype.toLocaleTimeString()
返回一个字符串,代表时间的当地写法。
var d = new Date(2013, 0, 1);
d.toLocaleTimeString()
// 中文版浏览器为"上午12:00:00"
// 英文版浏览器为"12:00:00 AM"
get
类方法:
getTime():返回距离1970年1月1日00:00:00的毫秒数,等同于valueOf方法。
getDate():返回实例对象对应每个月的几号(从1开始)。
getDay():返回星期几,星期日为0,星期一为1,以此类推。
getYear():返回距离1900的年数。
getFullYear():返回四位的年份。
getMonth():返回月份(0表示1月,11表示12月)。
getHours():返回小时(0-23)。
getMilliseconds():返回毫秒(0-999)。
getMinutes():返回分钟(0-59)。
getSeconds():返回秒(0-59)。
getTimezoneOffset():返回当前时间与UTC的时区差异,以分钟表示,返回结果考虑到了夏令时因素。
一个计算本年度还剩多少天的例子:
function leftDays() {
var today = new Date();
var endYear = new Date(today.getFullYear(), 11, 31, 23, 59, 59, 999);
var msPerDay = 24 * 60 * 60 * 1000;
return Math.round((endYear.getTime() - today.getTime()) / msPerDay);
}
var myDate = new Date();
var myDate = new Date(2017,3,1,7,1,1,100);
var year = myDate.getFullYear(); // 四位数字返回年份
vat month = myDate.getMonth(); // 0-11
var date = myDate.getDate(); // 1-31
var day = myDate.getDay(); //0-6
var h = myDate.getHours(); // 0-23
var m = myDate.getMinutes(); // 0-59
var s = myDate.getSeconds(); // 0-59
var ms = myDate.getMilliseconds(); //0-999
var allmMS = myDate.getTime(); //获取从1970至今的毫秒数 得到的值可以用于new Date(ms);
小练习:获取昨天的日期:
var today = new Date();
var todayms = today.getTime();
var oneDay = 24*60*60*1000;
var yesterday = new Date(todayms - oneDay);
Error
错误处理机制
Javascript的6种原生错误类型:
-
SyntaxError
解析代码时发生的语法错误 -
ReferenceError
引用不存在的变量时发生的错误 -
RangeError
当一个值超出有效范围时发生的错误 -
TypeError
变量或参数不是预期类型时发生的错误。比如new 123
-
URIError
URI相关函数的参数不正确时抛出的错误。主要涉及encodeURI()
,decodeURI()
,encodeURIComponent()
,decodeURIComponent()
,escape()
和unescape()
这六个函数。 -
EvalError
eval函数没有被正确执行时抛出。该错误类型已不再在ES5中出现了。
以上6种派生错误,连同原始的Error对象,都是构造函数。
19.标准内置对象中的非构造器对象之——Math,Json
Math对象是拥有一些属性和对象的单一对象,主要用于数字计算
常用方法:
Math.floor(num)
:向下取整
Math.ceil(num)
向上取整
Math.random()
返回0-1之间的一个伪随机数,一定小于1,但可能等于0。
用处:
1.任意范围内生成随机整数
function getRandomInt(min, max) {
return Math.floor(Math.random() * (max - min +1 )) + min;
}
2.返回指定长度的随机字符:
function random_str(length) {
var ALPHABET = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ';
ALPHABET += 'abcdefghijklmnopqrstuvwxyz';
ALPHABET += '0123456789-_';
var str = '', randIndex = '';
for(var i=0; i < length; i++) {
randIndex = Math.floor(Math.random() * ALPHABET.length);
str += ALPHABET.charAt(randIndex);
}
return str;
}
JSON对象主要用于存储和传递文本信息
常用方法:
JSON.parse(jsonStr)
:将JSON字符串解析
为JSON对象
JSON.stringify(jsonObj)
:将JSON对象序列化
为JSON字符串
20.标准内置对象中的非构造器对象之——全局对象
属性:NaN,Infinity,undefined
方法:parseInt,parseFloat,isNaN,isFinite,eval
处理URI的方法:encodeURIComponent,decodeURIComponent,encodeURI,decodeURI
构造器属性:Boolean,String,Number,Object,Function,Array,Date,Error...
对象属性:Math,JSON
(1)NaN
不等于任何值,包括它自己本身。
(2)parseInt(string[, radix])
用于将字符串转换为数字,参数:要转换的字符串,进制(默认十进制)
注意:如果第一个参数不是字符串,会将其先用String(param)
转换为字符串,所以parseInt
会将空字符串
,true
,null
转换为NaN
。
parseInt('1'); //1 * 10^0 = 1
parseInt('1', 16); //1 * 16^0 = 1
parseInt('1f'); // 1 * 10^0 = 1
parseInt('1f'); //1 * 16^1 + f * 16^0 = 16 + 15 = 31
parseInt('f1'); // NaN
将非字符串先转换为字符串还会导致一些意外的结果:
parseInt(0x11, 36) // 43
// 等同于
parseInt(String(0x11), 36)
parseInt('17', 36)
上面代码中,十六进制的0x11会被先转为十进制的17,再转为字符串。然后,再用36进制解读字符串17,最后返回结果43。
radix
取值在2-36之间,0、null、undefined都会被直接忽略,当做默认的10来处理:
parseInt('10', 1); // NaN
parseInt('10', 37); // NaN
parseInt('10', 0); // 10
parseInt('10', null); // 10
parseInt('10', undefined); // 10
parseInt('10', 2); // 2 ( 1*2^1 + 0*2^0 )
(3)parseFloat
用于将字符串转换为浮点数。注意,它与parseInt
一样,都会将空字符串
,true
,null
转换为NaN
:
parseFloat(true) // NaN
Number(true) // 1
parseFloat(null) // NaN
Number(null) // 0
parseFloat('') // NaN
Number('') // 0
parseFloat('123.45#') // 123.45
Number('123.45#') // NaN
(4)eval
计算某个字符串,并执行其中的Javascript代码。
语法:eval(string)
不建议使用,会有性能问题。
(5)encodeURIComponent
:用于将URI参数中的中文、特殊字符等作为URI的一部分进行编码
语法:encodeURIComponent(URIString)
栗子:
1.防止注入性攻击:var url = "http://www.xxx.com/index.html?name=" + encodeURIComponent(name);
2.依赖URL进行参数传递时,对于参数中有特殊字符(如"&","/")的要对其进行编码:
var url = "http://www.xxx.com/index.html?name=" + encodeURIComponent("Tom&Jerry") + "&src=" + encodeURIConponent("/assert/image/test.png");
表达式与运算符
介绍表达式、算术运算符、位运算符、布尔运算符、关系运算符、相等与全等、条件运算符。
21.表达式
JS短语,解释器可以执行它并生成一个值。
22.运算符
+
:加法运算符(既可以处理算术的加法,也可以处理字符串连接)
算法步骤:
- 如果运算子是对象先自动转成原始类型的值(即,先valueOf(),如果结果还是对象,就toString();Date类型则先toString())
- 两个运算子都为原始类型之后,如果其中一个为字符串,则两个都转为字符串,拼接。
- 否则,两个运算子都转为数值,进行加法运算。
[1, 2] + [3]
// "1,23"
// 等同于
String([1, 2]) + String([3])
// '1,2' + '3'
加法运算符以外的其他算术运算符(比如减法、除法和乘法),都不会发生重载。它们的规则是:所有运算子一律转为数值,再进行相应的数学运算。
1 - '2' // -1
1 * '2' // 2
1 / '2' // 0.5
===
:类型相同且值相等
var a = "123";
var oa = new String("123");
a === oa; // false 其中 a 为string值类型,oa为object对象类型
//undefined 与 null 都与自身严格相等
undefined === undefined // true
null === null // true
==
:判断操作符两边对象或值是否相等
规则用伪代码表示为:
function equal(a, b) {
if a、b类型相同
return a === b;
else a、b类型不同
return Number(a) === Number(b);
}
"99" == 99 //true
"abc" == new String("abc") // true
例外:
1.null == undefined //true
2.null与undefined进行 == 运算时不进行类型转换
0 == null //false
null == false // false
"undefined" == undefined // false
!!x
表示取x表达式运行后的Boolean
值
注意&&
和 ||
都有短路功能
且运算符的运算规则:如果第一个运算子的布尔值为true
,则返回第二个运算子的值(注意不是布尔值);如果第一个运算子的布尔值为false
,返回第一个运算子的值(不是布尔值)。或运算符的运算规则:如果第一个运算子的布尔值为
true
,则返回第一个运算子的值(注意不是布尔值),且不再对第二个运算子求值;如果第一个运算子的值为false
,则返回第二个运算子的值(不是布尔值)。
位运算符
用于直接对二进制位进行计算。
异或运算:^
。表示当两个二进制位不相同,则结果为1
,否则结果为0
.
常用于在不定义新变量的条件下互换两个数的值:
var a = 3;
var b = 7;
a^=b;
b^=a;
a^=b;
a; // 7
b; // 3
开关作用
位运算符可以用作设置对象属性的开关。
假设某个对象有4个开关,每个开关都有一个变量。那么可以设置一个四位的二进制数,用来表示四个开关
var FLAG_A = 1; // 0001
var FLAG_B = 2; // 0010
var FLAG_C = 4; // 0100
var FLAG_D = 8; // 1000
上面代码设置A、B、C、D四个开关,每个开关分别占有一个二进制位。
然后就可以用与运算检验当前是否打开了某个开关。
var flags = 5; // 0101
if(flags && FLAG_C) {
console.log('当前打开了C开关.');
}
// 当前打开了C开关.
逗号运算符,
用于对两个表达式求值,并返回后一个表达式的值。
var x = 0;
var y = (x++, 10);
x; // 1
y; // 10
运算符优先级: * / %
高于 + -
高于 && ||
高于 ? :
完整表格:
左结合与右结合
大部分运算符是从左到右计算的,少数是从右向左计算。最常见的是赋值运算符 =
和 三元条件运算符 ? :
var x = y = z = m;
var q = a ? b : c ? d : e ? f : g;
//其实等价于:
var x = (y = (z = m));
var q = a ? b : (c ? d : (e ? f : g));
JS语句
条件控制语句
循环控制语句
for in
:遍历对象的属性
function Car(id, type, color) {
this.id = id;
this.type = type;
this.color = color;
}
var benz = Car(12345, "benz", "black");
for(var key in benz) {
console.log(key + ": " + benz[key]);
}
//id: 12345
//type: benz
//color:black
//当这个对象有函数时遍历就会把原型对象上的方法也遍历出来
Car.prototype.start = function() {
console.log(this.type + "start");
}
//遍历的时候会多一项
//start: function(){ ... }
//但我们通常是不需要方法的,而方法通常都是存在于原型对象上,所以可以通过hasOwnProperty判断是否为对象本身属性:
for(var key in benz) {
if(benz.hasOwnProperty(key)) {
console.log(key + ":" + benz[key]);
}
}
异常处理语句
try catch finally
throw
with语句
通过暂时改变变量的作用域链(将with语句中的对象添加到作用域链的头部),来减少代码量。
(function(){
var a = Math.cos(3 * Math.PI) + Math.sin(Math.LN10);
});
//用with语句
(function(){
with(Math){
var a = cos(3 * PI) + sin(LN10);
}
})
原型链如图:
变量作用域
主要关注两个方面:
1. 变量的生命周期
和作用范围
。
2. 看到一个变量时,要能找到变量定义的位置。
变量作用域分为静态作用域和动态作用域:
静态作用域(词法作用域):在编译阶段就可以决定变量的引用,只跟程序定义的原始位置有关,与代码执行顺序无关。
动态作用域:在运行时才能决定变量的引用。一步一步执行代码,把执行到的变量和函数定义从下到上依次放到栈里面,用的时候选离自己最近的一个(比如有两个x的定义,选离自己最近的一个)。
JS使用静态作用域。ES5使用词法环境管理静态作用域。
词法环境
是什么? 是用来描述和管理静态作用域的一种数据结构。
组成?
- 环境记录(
record
)[形参、变量、函数等] - 对外部词法环境的引用(
outer
)
什么时候创建?
一段代码在执行之前,会先初始化创建一个词法环境。又因为JS没有块级作用域,只有全局作用域和函数作用域,所以在JS中只有全局代码
或者函数代码
开始执行前会先初始化词法环境。
哪些东西会被初始化到词法作用域中呢?
- 形参(一定要是这个函数显式定义的形参,而不是真实传进来的参数,即不是实参)
- 函数定义(除了这个函数的形参和函数体,需要特别注意:在初始化的时候会保存当前的作用域到函数对象中,即
scope:currentEnvironment
)这也是形成闭包的必要条件呀!●▽● - 变量定义(var a = xxx;)
词法环境是怎样构成的?
知道了词法环境的构成,就可以分析代码是如何执行的,知道里面的变量到底引用的是什么值。
几个关于词法环境的问题
1.形参、函数定义或变量定义名称冲突时,优先级:函数定义 > 形参 > 变量定义
2.arguments
这个对象其实也是在环境记录中的
3.函数表达式
是在执行到函数表达式这一句代码时才创建函数对象,而函数定义
是在代码初始化时就创建了函数对象,保存了当前作用域。
词法环境可能发生变化的栗子
with
var foo = "abc";
with({ foo: "bar" }) {
function f() { alert foo; }
(function() { alert foo; })(); //这就是一个函数表达式,只有在执行到这一句时才会创建函数对象
f();
}
注意函数定义与函数表达式的区别:
因为词法环境只会记录【形参,函数定义,变量定义】这三项。
而且JS中只有全局作用域
和函数作用域
,没有块级作用域
,所以所有不在函数作用域中定义的变量或者函数都是属于全局作用域的
。
而with会创建一个临时的作用域,叫with的词法环境,在with中用var定义一个变量与在with外一样,都是在代码初始化时就被放到词法环境中,而函数表达式则是在执行到那一句代码时再放进词法环境中。
所以函数定义f()的词法环境outer指向global,而函数表达式词法环境的outer指向with。
注意:这里仅仅拿with举例,我们不建议使用with,容易造成混淆。
try-catch
catch
的代码块与with
的代码块很相似,也是临时创建的词法环境。
左侧的 closure Environment
即为右边红框圈出来的函数表达式
的词法环境,outer指向catch Environment。
带名称的函数表达式(不常用)
(function A(){
A = 1;
alert(A);
})();
因为A是一个函数表达式,并不是一个函数定义,所以A不会定义到global Environment
中。
然后执行到这一句时,与之前的匿名函数表达式类似,创建一个该函数表达式的词法环境,把A这个函数表达式定义到这个词法环境中。而且在A的环境里定义的A函数不能被修改,所以A = 1
是没有用的,仍然会alert出 function A(){A=1;console.log(A);}
。
参考资料:wikipedia Scope
函数体内部声明的函数,作用域绑定函数体内部
function foo() {
var x = 1;
function bar() {
console.log(x);
}
return bar;
}
var x = 2;
var f = foo();
f() // 1
上面代码中,函数foo内部声明了一个函数bar,bar的作用域绑定foo。当我们在foo外部取出bar执行时,变量x指向的是foo内部的x,而不是foo外部的x。正是这种机制,构成了下文要讲解的“闭包”现象。
来自阮一峰的这篇文章
一个困扰我很久的问题:函数定义
和函数表达式
到底怎样区分?!
一直就没搞懂过这个问题,看了半天MDN上的文档,都是在说:
函数定义
与函数表达式
基本是一样滴
因为那些在等号右边的只能是表达式,所以它们是函数表达式
虽然,函数表达式
一般情况下没有名字,但是呢,它们也是可以有名字滴,有名字之后就与函数声明
的语法真的一样了,这还怎么区分啊摔!( ´Д`)
不信你自己看 MDN函数那一章 看的我都要哭了
直到我去翻了阮大大博客里讲函数的一篇, ⊙▽⊙ 里面讲立即调用的函数表达式(IIFE)一节里有这样一句话:
为了避免解析上的歧义,JavaScript引擎规定, 如果function关键字出现在行首,一律解释成语句。因此,JavaScript引擎看到行首是function关键字之后,认为这一段都是函数的定义。
所以这样句首不是function
的(function(){})();
就是函数表达式啦!下面这样也可以:
var i = function(){ return 10; }();
true && function(){ /* code */ }();
0, function(){ /* code */ }();
// 甚至这样也可以:
!function(){ /* code */ }();
~function(){ /* code */ }();
-function(){ /* code */ }();
+function(){ /* code */ }();
闭包
什么是闭包?
闭包是由函数
和它引用的外部词法环境
组合而成。
闭包允许函数访问其引用的外部词法环境中的变量(又称自由变量)。
看个栗子:下面的函数add()
最终return
的值是一个函数,且在add函数
外调用return出来的这个匿名函数
时可以引用add函数内部的环境
,使用add函数内部定义的变量i
。
function add(){
var i = 0;
return function() {
console.log(i++);
}
}
var f = add();
f(); // 0
f(); // 1
一般来说,一个函数执行完之后,内部的局部变量都会被释放掉,就不存在了。但是JS比较特殊,因为JS中允许函数作为返回值
,当函数作为返回值,这个函数就保存了对外部词法环境的引用,而相应的,外部词法环境中定义的变量就不会被释放掉,会一直保留在内存中。这就是JS的闭包。闭包使得它引用的函数
的内部环境一直存在,所以闭包可以看作是它引用的函数的内部环境
的一个接口。
广义上来说,所有的JS函数都可以称为闭包,因为所有的JS函数创建后都保存了当前的词法环境。
闭包的应用
闭包最大的作用有两个,一个是可以读取函数内部的变量,另一个是使这些变量始终保存在内存中。
1.使外部词法环境的变量始终保存在内存中
function createIncrementor(start) {
return function () {
return start++;
};
}
var inc = createIncrementor(5);
inc() // 5
inc() // 6
inc() // 7
start
存在于 createIncrementor
的词法环境中,通过闭包inc
被保留在内存中,因此可以累加。
2.保存变量现场
需求:为一组元素注册onclick事件,当点击其中某一个元素触发onclick事件时能将该元素的索引打印出来。
看下面这段代码是否可以实现?
var addHandlers = function(nodes) {
for(var i=0; i
jsfiddle自己试试
试过会发现:每次弹出的都是3,所以这样写是不能实现的。因为JS没有块级作用域,所以每次for循环不会形成单独的作用域,因此最内层的匿名function函数中所有的i共享它的外部词法环境addHandler函数中的同一个变量i。而当onclick被调用时,循环早已结束,所以i的值也就是循环结束后的nodes.length
。
想要更清晰一些,可以把for循环拆开看看:
var addHandler = function(nodes){
var i = 0;
nodes[i].onclick = function(){
alert(i);
}
//等价于
//nodes[0].onclick = function(){
// alert(i); 注意这里还是i,当onclick被调用,循环已经结束了,也就是addHandler函数执行到最后了,所以i是3
//}
i = 1;
nodes[i].onclick = function(){
alert(i);
}
i = 2;
nodes[i].onclick = function(){
alert(i);
}
i = 3;
}
addHandler(document.getElementsByTagName('a'));
这样总可以理解了吧。
如何改进?
既然问题出在每一次for循环没有能够形成单独的作用域来保存每一次for循环的变量i的值
,那么要解决的就是这个问题。
方法就是用刚刚学的闭包:在内层匿名函数外面与for循环之间加一个帮助函数helper,helper有一个形参i
,在每一次for循环时,都调用一次helper(i),传入本次循环的i,helper接收了这个形参
之后,把它存到自己的词法环境中,内部的匿名function函数就可以访问这个形参,再将这个匿名函数return出去,这就形成了一个闭包。这样就做到了每次循环都形成一个闭包
。因此就可以把传入的i的值给保存下来。将来当onclick事件被触发时,就会调用helper函数return出来的匿名函数,也就可以使用helper函数的内部词法环境,弹出当时存进去的那个i的值。
划重点:
每一次for循环,调用一次helper(i)函数;
每一次onclick事件被触发,调用helper内部return出来的匿名函数,这个匿名函数的outer指向helper函数内部。
var addHandlers = function(nodes) {
var helper = function(i) {
return function(){
alert(i);
}
}
for(var i=0; i
jsfiddle里置几试试
如果还是无法理解,那么我们再来把for循环拆开看看
var addHandler = function(nodes){
var helper = function(i){
return function(){
alert(i);
}
}
var i = 0;
nodes[i].onclick = helper(i); // 等价于 nodes[0].onclick = helper(0);
i = 1;
nodes[i].onclick = helper(i);// 等价于 nodes[1].onclick = helper(1);
i = 2;
nodes[i].onclick = helper(i);
i = 3;
}
addHandler(document.getElementsByTagName('a'));
那么,你能不能再想出一种使用闭包解决这个问题的写法呢?想想看 (。・ω・)ノ゙
参考写法:
var addHandler = function(nodes){
var helper2 = function(i) {
nodes[i].onclick = function(){
alert(i);
}
}
for(var i=0; i
注意,外层函数每次运行,都会生成一个新的闭包,而这个闭包又会保留外层函数的内部变量,所以内存消耗很大。因此不能滥用闭包,否则会造成网页的性能问题。
3.封装
共享函数,私有变量。
var observer = (function(){
var observerList = [];
return {
add: function(obj){
observerList.push(obj);
},
empty: function(obj){
observerList = [];
},
get: function(obj){
return observerList;
}
}
})
可以参考阮大大讲的闭包
面向对象
//Teacher constructor
function Teacher() {
this.courses = [];
this.job = 'teacher';
this.setName = function(name){
this.name = name;
}
this.addCourse = function(course){
this.courses.push(course);
}
}
var bill = new Teacher();
bill.setName("Bill");
bill.addCourse("math");
var tom = new Teacher();
tom.setName("Tom");
tom.addCourse("physics");
这样完全使用构造函数来构造实例对象,他们的存储是这样的:
这样一来,每个实例对象都会存储一遍所有的属性和方法,对于所有的实例对象来说,都耗用了一遍内存,这是很浪费的。实际上我们可以看到属性job
和方法setName
,addCourse
其实是所有实例对象可以共用的,那么如何实现呢?这就引出了原型。
原型
原型是构造器对象的一个属性,叫prototype
,JS中每个构造器对象都有这样一个属性。这个属性的值就是实例对象的原型,用于实现原型继承,让同一个构造器创建出的多个实例对象共享同一个原型
,这些实例对象就有了共同的属性和方法。
用原型改写上面的Teacher构造器:
//Teacher prototype
function Teacher(){
this.courses = [];
}
Teacher.prototype = {
job: 'teacher',
setName: function(name) {
this.name = name;
},
addCourses: function(course){
this.courses = course;
}
}
var bill = new Teacher();
bill.setName("Bill");
bill.addCourse("math");
var tom = new Teacher();
tom.setName("Tom");
tom.addCourse("physics");
原型链如图所示:
可以看到实例对象bill
与tom
都有一个隐式的指针指向了Teacher.prototype
原型链
上面的原型链并不是完整的,我们来看完整的原型链的构成:
我们知道,在JS中,函数对象是用于创建自定义构造器对象
的构造器对象,所以所有的自定义构造器对象都是Function的一个实例对象;又因为Function也有一个prototype属性,所以我们自定义的Teacher也有一个隐式的__proto__
属性指向Function.prototype
,而这其中有一些引擎自身实现了的一些属性和方法,比如toString() , apply()
等方法。
而又因为Teacher.prototype
的对象可以由new Object()
来创建,它是Object
对象的一个实例,所以Teacher.prototype
有一个隐式的__proto__
属性指向Object.prototype
。
它们之间其实构成了一个链式的关系:tom
是以Teacher.prototype
为原型的,Teacher.prototype
又是以Object.prototype
为原型的。这样就形成了一个原型链。
JS中的属性查找、属性修改和属性删除都是通过原型链实现的。
属性查找
先在对象本身找,找不到就到对象的原型上找。
属性修改
以tom对象为例:
tom.name = 'Tom Green';
因为tom对象本身有name
这个属性,所以直接修改。
tom.job = 'assitant';
tom对象
本身没有job
这个属性,那么就为tom
对象创建一个job
属性,同时把值赋给这个属性。注意这里即使tom对象的原型
上有job
这个属性,也不会直接修改原型上的属性。
如果想要修改原型对象上的属性,那么这样做:
Teacher.prototype.job = "assistant";
属性删除
与修改属性类似,delete也只能删除对象自身的属性,不能删除原型上的属性。eg:
delete tom.job;
如何判断属性是否来自对象本身:
tom.hasOwnProperty('job');
来源于对象本身就返回 true
,来源于原型或不存在则返回 false
。
使用ES5中的原型继承构造一个有原型的实例对象: Object.create(proto[, propertiesObject])
(而不是使用构造器constructor
方法)
可以直接从一个原型对象
创建一个新的对象,同时为这个对象指定一个原型。而不需要通过 new Teacher();
来完成。
var teacher = {
job: "teacher",
courses: [],
setName: function(name) {
this.name = name;
},
addCourses: function(course) {
this.courses.push(course);
},
}
var bill = Object.create(teacher);
tom.setName('Bill')
这样我们就不用通过构造器,直接创建一个以teacher
对象为原型的bill实例对象。并没有使用前面说的constructor,prototype
等方式。这是ES5中提供的一种新的原型继承的方式。原型链:
JS面向对象
全局变量很容易冲突,所以需要做信息隐藏。最好的方式就是封装:将一部分信息开放出来,另外一部分隐藏起来。
封装
Java中有private, protected, public
关键字来定义属性和方法的访问权限。但JS中并没有,那么JS中是如何实现封装的呢?
按照我们原来的写法,像下图这样是没有实现封装的:
通过JS语言的特性做类似的模拟。像下图这样是可以实现封装的:
模拟private
:在function A(){}内部通过var 定义了一个属性_config
,在外部无法被直接访问和修改,然后提供了一个函数this.getConfig()
来让我们可以在外部获取这个属性。这样一来就模拟了private
的特性。
模拟protected
:对于protected
和public
,在JS中无法做到本质上的区分。可以使用一些人为约束的规则来区分,比如:使用下划线开头来定义protected
的属性和方法。这样虽然实际上是都可以调用的,但是在代码的书写上是可以区分出来的。
继承
类继承
(function(){
function ClassA(){}
ClassA.classMethod = function(){}
ClassA.prototype.api = function(){}
function ClassB(){
ClassA.apply(this, arguments);
}
//给ClassB的原型赋值了一个ClassA类型的实例对象
ClassB.prototype = new ClassA();
//上述这样会导致ClassB.prototype的constructor也变成ClassA所以我们要手动改成ClassB
ClassB.prototype.constructor = ClassB;
ClassB.prototype.api = function() {
ClassA.prototype.api.apply(this, arguments);
}
var b = new ClassB();
b.api();
})();
原型链:
原型继承
类继承可以说是对其他语言的类继承的一种模拟,而原型继承就是JS固有的特性了。
(function(){
//先定义一个原型
var proto = {
action1: function(){
},
action2: function(){
}
}
//使用Object.create基于已有对象创建一个新的对象
var obj = Object.create(proto);
})();
编程风格
- 缩进:2个空格。
- 循环和判断的代码体总是使用大括号。
- 圆括号。表示函数调用时,函数名与左括号间不加空格;表示函数定义时,函数名与与左括号间不加空格;其他情况,前面位置的语法元素与左括号之间,都加空格。
- 行尾的分号建议不要省略。
- 全局变量尽量减少使用。不得不用时建议用大写字母。
- 变量声明。因为JS有变量声明提升,所以尽量将所有变量声明在代码块头部。
-
new
命令。用new
命令从构造函数生成一个新对象时,如果忘了加上new
,新对象内部this
关键字就会指向全局对象,导致所有绑定在this
上的变量都变成全局变量。因此,建议使用Object.create()
命令替代new
命令。 - with语句尽量不要使用。
- 不要使用相等
==
运算符,只使用严格相等===
运算符。 - 语句的合并。建议不要将不同目的的语句合并到同一行。
- 尽量用
+=
代替++
。比如:用x += 1;
代替++x;
。 -
switch ... case
每一个case
后都要加break
语句,这样造成代码冗长。而且,使得代码结构混乱。建议避免使用,改写成对象结构。
function doAction(action) {
var actions = {
'hack': function() {
return 'hack';
},
'slash': function() {
return 'slash';
},
...
}
}