while (x < 10) { // “x < 10” 是循环条件
// do stuff
x++;
}
可以被转化成一个递归函数和对其的调用:
function loop(x) {
if (x >= 10) // “x >= 10” 是退出条件(等同于 “!(x < 10)”)
return;
// 做些什么
loop(x + 1); // 递归调用
}
loop(0);
不过,有些算法并不能简单的用迭代来实现。例如,获取树结构中所有的节点时,使用递归实现要容易得多:
function walkTree(node) {
if (node == null) //
return;
// do something with node
for (var i = 0; i < node.childNodes.length; i++) {
walkTree(node.childNodes[i]);
}
}
跟loop
函数相比,这里每个递归调用都产生了更多的递归。
将递归算法转换为非递归算法是可能的,不过逻辑上通常会更加复杂,而且需要使用堆栈。事实上,递归函数就使用了堆栈:函数堆栈。
这种类似堆栈的行为可以在下例中看到:
function foo(i) {
if (i < 0)
return;
console.log(‘begin:’ + i);
foo(i - 1);
console.log(‘end:’ + i);
}
foo(3);
// 输出:
// begin:3
// begin:2
// begin:1
// begin:0
// end:0
// end:1
// end:2
// end:3
嵌套函数和闭包
你可以在一个函数里面嵌套另外一个函数。嵌套(内部)函数对其容器(外部)函数是私有的。它自身也形成了一个闭包。一个闭包是一个可以自己拥有独立的环境与变量的表达式(通常是函数)。
既然嵌套函数是一个闭包,就意味着一个嵌套函数可以”继承“容器函数的参数和变量。换句话说,内部函数包含外部函数的作用域。
可以总结如下:
-
内部函数只可以在外部函数中访问。
-
内部函数形成了一个闭包:它可以访问外部函数的参数和变量,但是外部函数却不能使用它的参数和变量。
下面的例子展示了嵌套函数:
function addSquares(a, b) {
function square(x) {
return x * x;
}
return square(a) + square(b);
}
a = addSquares(2, 3); // returns 13
b = addSquares(3, 4); // returns 25
c = addSquares(4, 5); // returns 41
由于内部函数形成了闭包,因此你可以调用外部函数并为外部函数和内部函数指定参数:
function outside(x) {
function inside(y) {
return x + y;
}
return inside;
}
fn_inside = outside(3); // 可以这样想:给一个函数,使它的值加3
result = fn_inside(5); // returns 8
result1 = outside(3)(5); // returns 8
保存变量
注意到上例中 inside
被返回时 x
是怎么被保留下来的。一个闭包必须保存它可见作用域中所有参数和变量。因为每一次调用传入的参数都可能不同,每一次对外部函数的调用实际上重新创建了一遍这个闭包。只有当返回的 inside
没有再被引用时,内存才会被释放。
这与在其他对象中存储引用没什么不同,但是通常不太明显,因为并不能直接设置引用,也不能检查它们。
多层嵌套函数
函数可以被多层嵌套。例如,函数A可以包含函数B,函数B可以再包含函数C。B和C都形成了闭包,所以B可以访问A,C可以访问B和A。因此,闭包可以包含多个作用域;他们递归式的包含了所有包含它的函数作用域。这个称之为作用_域链_。(稍后会详细解释)
思考一下下面的例子:
function A(x) {
function B(y) {
function C(z) {
console.log(x + y + z);
}
C(3);
}
B(2);
}
A(1); // logs 6 (1 + 2 + 3)
在这个例子里面,C可以访问B的y和A的x。这是因为:
-
B形成了一个包含A的闭包,B可以访问A的参数和变量
-
C形成了一个包含B的闭包
-
B包含A,所以C也包含A,C可以访问B和A的参数和变量。换言之,C用这个顺序链接了B和A的作用域
反过来却不是这样。A不能访问C,因为A看不到B中的参数和变量,C是B中的一个变量,所以C是B私有的。
命名冲突
当同一个闭包作用域下两个参数或者变量同名时,就会产生命名冲突。更近的作用域有更高的优先权,所以最近的优先级最高,最远的优先级最低。这就是作用域链。链的第一个元素就是最里面的作用域,最后一个元素便是最外层的作用域。
看以下的例子:
function outside() {
var x = 5;
function inside(x) {
return x * 2;
}
return inside;
}
outside()(10); // returns 20 instead of 10
命名冲突发生在return x
上,inside
的参数x
和outside
变量x
发生了冲突。这里的作用链域是{inside
, outside
, 全局对象}。因此inside
的x
具有最高优先权,返回了20(inside
的x
)而不是10(outside
的x
)。
闭包是 JavaScript 中最强大的特性之一。JavaScript 允许函数嵌套,并且内部函数可以访问定义在外部函数中的所有变量和函数,以及外部函数能访问的所有变量和函数。
但是,外部函数却不能够访问定义在内部函数中的变量和函数。这给内部函数的变量提供了一定的安全性。
此外,由于内部函数可以访问外部函数的作用域,因此当内部函数生存周期大于外部函数时,外部函数中定义的变量和函数的生存周期将比内部函数执行时间长。当内部函数以某一种方式被任何一个外部函数作用域访问时,一个闭包就产生了。
var pet = function(name) { //外部函数定义了一个变量"name"
var getName = function() {
//内部函数可以访问 外部函数定义的"name"
return name;
}
//返回这个内部函数,从而将其暴露在外部函数作用域
return getName;
};
myPet = pet(“Vivie”);
myPet(); // 返回结果 “Vivie”
实际上可能会比上面的代码复杂的多。在下面这种情形中,返回了一个包含可以操作外部函数的内部变量方法的对象。
var createPet = function(name) {
var sex;
return {
setName: function(newName) {
name = newName;
},
getName: function() {
return name;
},
getSex: function() {
return sex;
},
setSex: function(newSex) {
if(typeof newSex == “string”
&& (newSex.toLowerCase() == “male” || newSex.toLowerCase() == “female”)) {
sex = newSex;
}
}
}
}
var pet = createPet(“Vivie”);
pet.getName(); // Vivie
pet.setName(“Oliver”);
pet.setSex(“male”);
pet.getSex(); // male
pet.getName(); // Oliver
在上面的代码中,外部函数的name
变量对内嵌函数来说是可取得的,而除了通过内嵌函数本身,没有其它任何方法可以取得内嵌的变量。内嵌函数的内嵌变量就像内嵌函数的保险柜。它们会为内嵌函数保留“稳定”——而又安全——的数据参与运行。而这些内嵌函数甚至不会被分配给一个变量,或者不必一定要有名字。
var getCode = (function(){
var secureCode = “0]Eal(eh&2”; // A code we do not want outsiders to be able to modify…
return function () {
return secureCode;
};
})();
getCode(); // Returns the secret code
尽管有上述优点,使用闭包时仍然要小心避免一些陷阱。如果一个闭包的函数定义了一个和外部函数的某个变量名称相同的变量,那么这个闭包将无法引用外部函数的这个变量。
var createPet = function(name) { // Outer function defines a variable called “name”
return {
setName: function(name) { // Enclosed function also defines a variable called “name”
name = name; // ??? How do we access the “name” defined by the outer function ???
}
}
}
函数的实际参数会被保存在一个类似数组的arguments对象中。在函数内,你可以按如下方式找出传入的参数:
arguments[i]
其中i
是参数的序数编号(译注:数组索引),以0开始。所以第一个传来的参数会是arguments[0]
。参数的数量由arguments.length
表示。
使用arguments对象,你可以处理比声明的更多的参数来调用函数。这在你事先不知道会需要将多少参数传递给函数时十分有用。你可以用arguments.length
来获得实际传递给函数的参数的数量,然后用arguments
对象来取得每个参数。
例如,设想有一个用来连接字符串的函数。唯一事先确定的参数是在连接后的字符串中用来分隔各个连接部分的字符(译注:比如例子里的分号“;”)。该函数定义如下:
function myConcat(separator) {
var result = ‘’; // 把值初始化成一个字符串,这样就可以用来保存字符串了!!
var i;
// iterate through arguments
for (i = 1; i < arguments.length; i++) {
result += arguments[i] + separator;
}
return result;
}
你可以给这个函数传递任意数量的参数,它会将各个参数连接成一个字符串“列表”:
// returns "red, orange, blue, "
myConcat(", ", “red”, “orange”, “blue”);
// returns "elephant; giraffe; lion; cheetah; "
myConcat("; ", “elephant”, “giraffe”, “lion”, “cheetah”);
// returns "sage. basil. oregano. pepper. parsley. "
myConcat(". ", “sage”, “basil”, “oregano”, “pepper”, “parsley”);
提示:arguments
变量只是 *”*类数组对象“,并不是一个数组。称其为类数组对象是说它有一个索引编号和length
属性。尽管如此,它并不拥有全部的Array对象的操作方法。
从ECMAScript 6开始,有两个新的类型的参数:默认参数,剩余参数。
默认参数
在JavaScript中,函数参数的默认值是undefined
。然而,在某些情况下设置不同的默认值是有用的。这时默认参数可以提供帮助。
在过去,用于设定默认参数的一般策略是在函数的主体中测试参数值是否为undefined
,如果是则赋予这个参数一个默认值。如果在下面的例子中,调用函数时没有实参传递给b
,那么它的值就是undefined
,于是计算a*b
得到、函数返回的是 NaN
。但是,在下面的例子中,这个已经被第二行获取处理:
function multiply(a, b) {
b = (typeof b !== ‘undefined’) ? b : 1;
return a*b;
}
multiply(5); // 5
使用默认参数,在函数体的检查就不再需要了。现在,你可以在函数头简单地把1设定为b
的默认值:
function multiply(a, b = 1) {
return a*b;
}
multiply(5); // 5
剩余参数
剩余参数语法允许将不确定数量的参数表示为数组。在下面的例子中,使用剩余参数收集从第二个到最后参数。然后,我们将这个数组的每一个数与第一个参数相乘。这个例子是使用了一个箭头函数,这将在下一节介绍。
function multiply(multiplier, …theArgs) {
return theArgs.map(x => multiplier * x);
}
var arr = multiply(2, 1, 2, 3);
console.log(arr); // [2, 4, 6]
箭头函数表达式也称胖箭头函数)相比函数表达式具有较短的语法并以词法的方式绑定 this
。箭头函数总是匿名的。
有两个因素会影响引入箭头函数:更简洁的函数和 this
。
更简洁的函数
在一些函数模式中,更简洁的函数很受欢迎。对比一下:
var a = [
“Hydrogen”,
“Helium”,
“Lithium”,
“Beryllium”
];
var a2 = a.map(function(s){ return s.length });
console.log(a2); // logs [ 8, 6, 7, 9 ]
var a3 = a.map( s => s.length );
console.log(a3); // logs [ 8, 6, 7, 9 ]
this
的词法
在箭头函数出现之前,每一个新函数都重新定义了自己的 [this]值(在构造函数中是一个新的对象;在严格模式下是未定义的;在作为“对象方法”调用的函数中指向这个对象;等等)。以面向对象的编程风格,这样着实有点恼人。
function Person() {
// 构造函数Person()将this
定义为自身
this.age = 0;
setInterval(function growUp() {
// 在非严格模式下,growUp()函数将this
定义为“全局对象”,
// 这与Person()定义的this
不同,
// 所以下面的语句不会起到预期的效果。
this.age++;
}, 1000);
}
var p = new Person();
在ECMAScript 3/5里,通过把this
的值赋值给一个变量可以修复这个问题。
function Person() {
var self = this; // 有的人习惯用that
而不是self
,
// 无论你选择哪一种方式,请保持前后代码的一致性
self.age = 0;
setInterval(function growUp() {
// 以下语句可以实现预期的功能
self.age++;
}, 1000);
}
另外,创建一个约束函数可以使得 this
值被正确传递给 growUp()
函数。
箭头函数捕捉闭包上下文的this
值,所以下面的代码工作正常。
function Person(){
this.age = 0;
setInterval(() => {
this.age++; // 这里的this
正确地指向person对象
}, 1000);
}
var p = new Person();
JavaScript语言有好些个顶级的内建函数:
eval()
eval()
方法会对一串字符串形式的JavaScript代码字符求值。
uneval()
uneval()
方法创建的一个Object
的源代码的字符串表示。
isFinite()
isFinite()
函数判断传入的值是否是有限的数值。 如果需要的话,其参数首先被转换为一个数值。
isNaN()
isNaN()
函数判断一个值是否是NaN
。注意:isNaN
函数内部的强制转换规则
十分有趣; 另一个可供选择的是ECMAScript 6 中定义Number.isNaN()
, 或者使用 typeof
来判断数值类型。
parseFloat()
parseFloat()
函数解析字符串参数,并返回一个浮点数。
parseInt()
parseInt()
函数解析字符串参数,并返回指定的基数(基础数学中的数制)的整数。
decodeURI()
decodeURI()
函数对先前经过encodeURI
函数或者其他类似方法编码过的字符串进行解码。
decodeURIComponent()
decodeURIComponent()
方法对先前经过encodeURIComponent
函数或者其他类似方法编码过的字符串进行解码。
encodeURI()
**encodeURI()**
方法通过用以一个,两个,三个或四个转义序列表示字符的UTF-8编码替换统一资源标识符(URI)的某些字符来进行编码(每个字符对应四个转义序列,这四个序列组了两个”替代“字符)。
encodeURIComponent()
encodeURIComponent()
方法通过用以一个,两个,三个或四个转义序列表示字符的UTF-8编码替换统一资源标识符(URI)的每个字符来进行编码(每个字符对应四个转义序列,这四个序列组了两个”替代“字符)。
escape()
已废弃的 **escape()**
方法计算生成一个新的字符串,其中的某些字符已被替换为十六进制转义序列。使用 encodeURI
或者encodeURIComponent
替代本方法。
unescape()
已废弃的 **unescape()**
方法计算生成一个新的字符串,其中的十六进制转义序列将被其表示的字符替换。上述的转义序列就像escape
里介绍的一样。因为 unescape
已经废弃,建议使用decodeURI()
或者decodeURIComponent
替代本方法。
最后
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助。
因此收集整理了一份《2024年Web前端开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点!不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门!
如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
deURIComponent`替代本方法。
unescape()
已废弃的 **unescape()**
方法计算生成一个新的字符串,其中的十六进制转义序列将被其表示的字符替换。上述的转义序列就像escape
里介绍的一样。因为 unescape
已经废弃,建议使用decodeURI()
或者decodeURIComponent
替代本方法。
最后
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长,自己不成体系的自学效果低效漫长且无助。
因此收集整理了一份《2024年Web前端开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
[外链图片转存中…(img-b35SjsaT-1715528391957)]
[外链图片转存中…(img-RgcQfHNM-1715528391958)]
[外链图片转存中…(img-f6I54GbV-1715528391958)]
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点!不论你是刚入门Android开发的新手,还是希望在技术上不断提升的资深开发者,这些资料都将为你打开新的学习之门!
如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!