JavaScript基础笔记(五) 函数表达式
函数表达式
一、闭包
概念:闭包是指有权访问另一个函数作用域中变量的函数。
function createCompareFun(propertyName) {
return function (obj1, obj2) {
var a = obj1[propertyName];
var b = obj2[propertyName];
if (a < b) {
return -1;
}
if (a > b) {
return 1;
} else {
return 0;
}
}
}
//创建匿名函数
var compare = createCompareFun("age");
//调用该函数
var result = compare({age:3},{age:8});
console.log(result); //-1
如果一个闭包能访问的函数被执行完毕,作用域链(本质上是一个指向变量对象的指针列表)被销毁后,
其活动对象也不会被销毁,因为它们被闭包引用,知道闭包函数被销毁后,它们才被销毁。
由于闭包会携带包含它函数的作用域,所以会占用更多内存,建议能不使用闭包就不要使用闭包。
作用域链的这种配置机制引出了另外一个重要问题:闭包只能取得包含函数中任何一个变量的最后一个值。
function createFun() {
var result = new Array();
for (var i = 0; i < 10; i++) {
result[i] = function () {
return i;
}
}
return result;
};
function getResult(fun, array) {
for (var j = 0; j < fun.length; j++) {
array[j] = fun[j]();
}
return array;
}
var fun = createFun();
var array = new Array();
var a = getResult(fun,array);
console.log(a); //[ 10, 10, 10, 10, 10, 10, 10, 10, 10, 10 ]
注意该问题的本质:注意函数实际执行时i变量的值,定义时的值不能说明问题。
解决办法:
function createFun() {
var result = new Array();
for (var i = 0; i < 10; i++) {
result[i] = function (num) {
return function () {
return num;
};
}(i);
}
return result;
};
/*var b = function rightNow(a) {
return a;
}(5);
console.log(b);*/
闭包中this的问题:
var name = "Window";
var obj = {
name: "Object",
getFun: function () {
return function () {
return this.name;
};
}
};
console.log(obj.getFun()()); //Window //解决:
var obj2 = {
name: "Obj2",
getFun: function () {
console.log("Fun's name: "+this.name);
var v = this.name;
return function () {
return v;
}
}
}
console.log(obj2.getFun()());
var name = "Window";
var obj = {
name: "Object",
getFun: function () {
return this.name;
}
}; var a = obj.getFun;
console.log(a()); //Window
console.log(obj.getFun()); //Object
内存泄漏问题:
window.onload = function () {
function outer() {
var element = document.getElementById("testId");
element.onclick = function () {
//闭包的作用域链中保存了一个html元素
//只要匿名函数存在,element的引用数最少是1,因此它所占用的内存永远不会被回收。
console.log(element.id);
};
}
outer();
//解决
function outer2() {
var element = document.getElementById("testId");
var id = element.id;
element.onclick = function () {
console.log(id);
};
//闭包会引用包含函数的整个活动对象,闭包即使不直接引用element,
//包含函数的活动对象任然会保存一个引用,所以这步必须有
element = null;
}
}
二、模仿块级作用域
ECMAScript很操蛋地没有块级作用域:
function Afun() {
for (var i = 0; i < 10; i++){
console.log(i);
}
//在外面重新声明i,并没有什么卵用
var i;
console.log("Outer i is "+i);
}
Afun(); //Outer i is 10
function Afun() {
for (var i = 0; i < 10; i++){
console.log(i);
}
//
var i = 100;
console.log("Outer i is "+i);
}
Afun(); //Outer i is 100
可以用匿名函数来模仿块级作用域:
function Afun() {
//由于函数声明后不能直接跟员括号来执行它
//所以我们用()包裹函数声明把它转换一个指向函数的引用
(function () {
for (var i = 0; i < 10; i++){
console.log(i);
}
})();
console.log(i); //导致Error:i is not defined.
}
这种技术经常被用来限制在全局作用域中添加过多的变量和函数。
三、私有变量
私有变量:函数的参数、局部变量和在函数内部定义的函数。
共有方法 == 特权方法
//在构造函数定义特权方法
function MyObject() {
//私有变量
var v = 10;
//私有函数
function f() {
return false;
} //特权方法
this.pm = function () {
++v;
return v;
}
} var o = new MyObject();
//除了pm()没有任何途径能访问到真正的o.v
console.log(o.v); //undefined
o.v = 166;
//假的
console.log(o.v); //
console.log(o.pm()); //
一)静态私有变量
也可以通过在私有作用域中定义私有变量和函数,创建特权方法。
(function () {
var v = 10;
function f() {
return false;
}
//注意这里没有var,是全局的
//为了保证全局,也没有使用声明函数
MyObject = function () {
};
MyObject.prototype.pm = function () {
v++;
return f();
};
})();
console.log(MyObject);
console.log(new MyObject().pm()); //false
JavaScript基础笔记(五) 函数表达式的更多相关文章
- JavaScript基础笔记一
一.真假判断 真的:true.非零数字.非空字符串.非空对象 假的:false.数字零.空字符串.空对象.undefined 例: if(0){ alert(1) }else{ alert(2) } ...
- JavaScript基础笔记二
一.函数返回值1.什么是函数返回值 函数的执行结果2. 可以没有return // 没有return或者return后面为空则会返回undefined3.一个函数应该只返回一种类型的值 二.可变 ...
- JavaScript基础笔记集合(转)
JavaScript基础笔记集合 JavaScript基础笔记集合 js简介 js是脚本语言.浏览器是逐行的读取代码,而传统编程会在执行前进行编译 js存放的位置 html脚本必须放在&l ...
- javascript立即调用的函数表达式N种写法(第二篇)
原文:javascript立即调用的函数表达式N种写法(第二篇) 上一篇博客我谈到将函数声明转换为函数表达式最常见的一种写法是:通过括号()将匿名函数声明转换为函数表达式即(function(){}) ...
- 《JavaScript高级程序设计》笔记:函数表达式(七)
递归 function factorial(num){ if(num<=1){ return 1; }else { return num * arguments.callee(num-1); } ...
- JS学习笔记 (五) 函数进阶
1.函数基础 1.1 函数的基本概念 函数是一段JavaScript代码,只被定义一次,但是可以被调用或者执行许多次.函数是一种对象,可以设置属性,或调用方法. 函数中的参数分为实参和形参.其中,形参 ...
- 理解JavaScript的立即调用函数表达式(IIFE)
首先这是js的一种函数调用写法,叫立即执行函数表达式(IIFE,即immediately-invoked function expression).顾名思义IIFE可以让你的函数立即得到执行(废话). ...
- JavaScript基础笔记(三) 引用类型
引用类型 引用类型的值(对象)是引用类型的一个实例. 一.Object类型 创建Object实例: //方法一:通过new操作符创建 var a = new Object(); a.neme = &q ...
- JavaScript基础知识(函数)
函数的基础 函数: 把实现相同功能的代码放到一个函数体中,当想实现这个功能时,直接执行这个函数即可:减少了的冗余:高内聚,低耦合--> 函数的封装: 函数:引用数据类型: var a = 10; ...
随机推荐
- WEB测试总结
WEB测试总结:1.js文件session是否有缓存,如果没有缓存对服务器压力会很大:2.更改页面大小后,图表是否会再次向服务器请求数据:3.表单填写是否对html标识,script脚本做处理:(&l ...
- JSON数据写入和解析
如何写入JSON 需要第三方jar包,JSON包 //写入json数据 public static String sendJson() { JSONObject json = new JSONObje ...
- Python enumerate() 函数
描述 enumerate() 函数用于将一个可遍历的数据对象(如列表.元组或字符串)组合为一个索引序列,同时列出数据和数据下标,一般用在 for 循环当中. Python 2.3. 以上版本可用,2. ...
- 矩阵乘法的运算量计算(华为OJ)
题目地址: https://www.nowcoder.com/practice/15e41630514445719a942e004edc0a5b?tpId=37&&tqId=21293 ...
- python 进阶读书笔记1 -- 理解python一切皆对象
理解python一切皆对象: 1.所有的类都是由type创建的 2.所有的类的基类都是object 3.type是类,也是实例,type的基类是object,type对象是由type创建的 4.obj ...
- Kafka/Zookeeper集群的实现(二)
[root@kafkazk1 ~]# wget http://mirror.bit.edu.cn/apache/zookeeper/zookeeper-3.4.12/zookeeper-3.4.12. ...
- ncat
ncat 或者说 nc 是一款功能类似 cat 的工具,但是是用于网络的.它是一款拥有多种功能的 CLI 工具,可以用来在网络上读.写以及重定向数据. 它被设计成可以被脚本或其他程序调用的可靠的后端工 ...
- BZOJ3622 已经没有什么好害怕的了 动态规划 容斥原理 组合数学
原文链接https://www.cnblogs.com/zhouzhendong/p/9276479.html 题目传送门 - BZOJ3622 题意 给定两个序列 $a,b$ ,各包含 $n$ 个数 ...
- Linux安装Tomcat-Nginx-FastDFS-Redis-Solr-集群——【第八集之安装Nginx】
1,务必保证安装Nginx之前,所需的环境必须安装完备. gcc 安装nginx需要先将官网下载的源码进行编译,编译依赖gcc环境,如果没有gcc环境,需要安装gcc:yum install gcc ...
- Python replace() 和 re.sub() 字符串字符替换
Python replace() 和 re.sub() 字符串字符替换 replace() testStr = 'aa:bb[cc' testStr.replace(':','_') 每次只能替换一个 ...