《JavaScript高级程序设计》第7章 函数表达式
定义函数的方式有两种:函数声明和函数表达式
// 函数声明
function function_name(argument) {
// body...
} // 函数表达式
var function_name = function (argument) {
// body...
}
由于函数声明提升,在执行之前会先读取函数声明,所以调用的语句可以放在函数声明之前。但是函数表达式则不可以把调用的语句放在之前。
sayHi(); // 不会报错
function sayHi() {
console.log("Hi!");
} sayHi(); // 错误:函数不存在
var sayHi = function () {
console.log("Hi!");
}
7.1 递归
arguments.callee是一个指向正在执行的函数的指针,因此可以用它来实现对函数的递归调用。
function factorial(num) {
if (num <= 1) {
return 1;
} else {
// arguments.callee是一个指向正在执行的函数的指针
return num * arguments.callee(num - 1);
}
}
在编写递归函数时,使用arguments.callee总比使用函数名更保险(避免函数名突然被赋成null之类的导致出错)
但在严格模式下不可以访问arguments.callee,可以使用命名函数表达式来达成同样的效果。
var factorial = (function f(num) {
if (num <= 1) {
return 1;
} else {
return num * f(num - 1);
}
});
7.2 闭包
闭包是一个函数,这个函数可以访问另一个函数作用域中的变量。
function createComparison(propertyName) {
return function (object1, object2) {
// 以下两行代码访问了外部函数中的变量propertyName
// 即使这个函数被返回了,而且在其他地方被调用了,它仍然可以访问变量propertyName
var value1 = object1[propertyName];
var value2 = object2[propertyName];
if (value1 < value2) {
return -1;
} else if (value1 > value2) {
return 1;
} else {
return 0;
}
};
}
上面的例子中,propertyName是在createComparison函数的作用域中,但里面的匿名函数被返回了之后,就算在其他地方被调用,也能访问到propertyName。
我的理解:通常的函数在执行完毕之后,其局部活动对象(包括arguments和局部变量)就会被销毁。但是如果一个函数体内有闭包的话(也就是它里面有另一个函数,这个函数引用了它的局部活动对象),则在这个函数执行完毕之后,其局部活动对象也不会被销毁,因为它体内的闭包的作用域链仍然在引用这个活动对象。

7.2.1 闭包和变量
function createFunctions() {
var result = new Array();
for (var i = 0; i < 10; i++) {
result[i] = function () {
return i;
}
}
return result;
}
var iFun = createFunctions();
console.log(iFun[0]());
// 当我调用iFun的时候,createFunctions早就执行完毕了,这时候i的值为10,而由于闭包的缘故i没有被销毁
// 所以实际上iFun数组的所有函数引用的i都是同一个i,值都为10
function createFunctions() {
var result = new Array();
for (var i = 0; i < 10; i++) {
result[i] = function(num) { // B
return function() {
return num; // A
}
}(i);
}
return result;
}
var iFun = createFunctions();
console.log(iFun[0]());
// 这个与上一个函数的不同之处在于,当我调用iFun的时候,进的是A位置
// 就算是那个有num参数的匿名函数(B位置)已经执行完了,它的num保存的值没有变过,还是当时传进去的索引i
// 对于iFun数组的每一个函数,他们的外层B都是不同的(num不同),就能达成目的
7.2.2 关于this对象
this问题丢失的实质:闭包通常在另一个作用域中被调用,并且它有自己的this(通常来说是window,因为闭包被调用的作用域通常是全局作用域),会屏蔽掉外层作用域的this(比如说以下例子中的object)。
var object = {
name: "My Object",
getNameFunc: function() { // B
return function() {
return this.name; // 调用的时候已经在全局作用域下,屏蔽掉了B位置作用域的this(也就是object)
}
};
}
console.log(object.getNameFunc()()); // "The Window"
而that模式能修复这个问题的原因:因为闭包中没有自己的that呀,所以它只能往上找,找到外层作用域中的that了。
7.4 私有变量
任何在函数中定义的变量,都可以认为是私有变量,因为不能在函数外部访问这些变量。
但是通过闭包,我们可以创建用于访问私有变量的公有方法,这种公有方法叫做特权方法。
著作权归作者所有。
商业转载请联系作者获得授权,非商业转载请注明出处。
作者:DIYgod
链接:https://www.anotherhome.net/2073
来源:Anotherhome function MyObject() {
// 私有变量和私有函数,外部无法访问
var privateVariable = 'DIYgod';
function privateFunction() {
console.log('lalala');
}
// 特权方法
this.publicMethod = function () {
console.log(privateVariable);
privateFunction();
};
}
var o = new MyObject();
o.publicMethod(); // DIYgod lalala
o.privateFunction(); // Uncaught TypeError: o.privateFunction is not a function
《JavaScript高级程序设计》第7章 函数表达式的更多相关文章
- 《JavaScript高级程序设计》笔记:函数表达式(七)
递归 function factorial(num){ if(num<=1){ return 1; }else { return num * arguments.callee(num-1); } ...
- 《JavaScript》高级程序设计第7章 函数表达式
7.2 闭包 定义: 闭包是指有权访问另一个函数作用域中的变量的函数. 理解闭包: 作用域链: 当某个函数被调用时,会创建一个执行环境以及相应的作用域链. 作用域链中,外部函数的活动对象始终处于第二位 ...
- 读书笔记 - js高级程序设计 - 第七章 函数表达式
闭包 有权访问另一个函数作用域中的变量的函数 匿名函数 函数没有名字 少用闭包 由于闭包会携带包含它的函数的作用域,因此会比其它函数占用更多的内存.过度使用闭包可能会导致内存占用过多,我们建议读者 ...
- 《JavaScript高级程序设计》——第二章在HTML使用JavaScript
这章讲的是JavaScript在HTML中的使用,也就是<script>元素的属性.书中详细讲了async.defer.src和type四个<script>的属性. 下面是对第 ...
- js高级程序设计(七)函数表达式
定义函数的方式有两种:一种是函数声明,另一种就是函数表达式.函数声明的语法是这样的. function functionName(arg0, arg1, arg2) { //函数体 } Firefox ...
- JavaScript 高级程序设计 第5章引用类型 笔记
第五章 引用类型 一.object类型 1.创建方法: 1.使用new 操作符创建 var person=new object() Person.name=”Nicholasa” Porson.age ...
- 读书时间《JavaScript高级程序设计》三:函数,闭包,作用域
上一次看了第6章,面向对象.这里接着看第7章. 第7章:函数表达式 定义函数有两种方式:函数声明.函数表达式 //函数声明 function functionName(arg0,arg1,arg2){ ...
- JAVASCRIPT高程笔记-------第 七章 函数表达式
7.1递归 经典递归例子 function factorial(num){ if(num <= 1){ return 1; }else{ return num * factorial(num - ...
- JavaScript高级程序设计第20章JSON 笔记 (学习笔记)
第二十章 JSON 1.Json 可以表示三种类型的值: 1.简单值: 表示数值:5 表示字符串:“hello wrold”注表示字符串时必须使用双引号 2.对象: {“name”:“mi”,”ag ...
- JavaScript高级程序设计第14章表单脚本 (学习笔记)
第十四章 表单脚本 1.阻止默认表单提交 1.提交表单数据 1.使用type=submit提交按钮 2.使用submit():方法 注意:当用户点击提交按钮时,会触发submit事件,从而在这里我们有 ...
随机推荐
- git --fast-version-control
--distributed-is-the-new-centralized 读二进制文件 python 读doc xls 几乎所有的版本控制系统都以某种形式支持分支.使用分支意味着你可以把你的工作从 ...
- IO流入门-第二章-FileOutputStream
FileOutputStreamj基本用法和方法示例 /* java.io.OutputStream java.io.FileOutputStream 文件字节输出流 将计算机内存中的数据写入到硬盘文 ...
- python基础之类和对象、对象之间的交互、类名称空间与对象/实例名称空间
一 面向对象初识 Python要么是面向过程要么是面向对象. 概念及优缺点: 面向过程的程序设计的核心是过程,过程即解决问题的步骤,面向过程的设计就好比精心设计好一条流水线,考虑周全什么时候处理什么东 ...
- plotly绘制直方图示例
计算数值出现的次数“ import cufflinks as cf cf.go_offline() import numpy as np import pandas as pd set_slippag ...
- javascript把RGB指定颜色转换成十六进制颜色(Converting R,G,B values to HTML hex notation)
Prologue 看见一篇非常好的外国文章,Making annoying rainbows in javascript,事实上我当时非常想把它翻译下来的,可是对于一个连六级都没过的人确实有点难度,一 ...
- Tomcat 自定义默认网站目录
上面访问的网址为http://192.168.0.108:8080/memtest/meminfo.jsp 需求: 现在我想访问格式为http://192.168.0.108:8080/meminfo ...
- 基于docker 搭建Elasticsearch5.6.4 分布式集群
说明: 准备2台机器,我这里有192. 和 192.168.0.164 192.168.0.164 作为master 192.168.0.107 作为普通node 一.环境 .docker 环境 .E ...
- 001-ant design pro安装、目录结构、项目加载启动【原始、以及idea开发】
一.概述 1.1.脚手架概念 编程领域中的“脚手架(Scaffolding)”指的是能够快速搭建项目“骨架”的一类工具.例如大多数的React项目都有src,public,webpack配置文件等等, ...
- 000 初步使用Kotlin开发Android应用
Kotlin是Jetbrians公司开发的一款编程语言,基于jvm兼容Java. 要求 IDE:IDEA或者Android Studio(简称studio)对Kotlin语言有所了解,官方文档:htt ...
- 利用WebBrowser实现自动登入功能
公司内部改革,对考勤方面做出调整,要求实现办公自动化,在OA进行上下班考勤:作为程序员,突发奇想如何实现自动化考勤应用? 需求如下: 可设置考勤地址.用户信息.上下班时间: 根据设置的上下班时间,定时 ...