函数表达式

一、闭包

概念:闭包是指有权访问另一个函数作用域中变量的函数。

        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基础笔记(五) 函数表达式的更多相关文章

  1. JavaScript基础笔记一

    一.真假判断 真的:true.非零数字.非空字符串.非空对象 假的:false.数字零.空字符串.空对象.undefined 例: if(0){ alert(1) }else{ alert(2) } ...

  2. JavaScript基础笔记二

    一.函数返回值1.什么是函数返回值    函数的执行结果2. 可以没有return // 没有return或者return后面为空则会返回undefined3.一个函数应该只返回一种类型的值 二.可变 ...

  3. JavaScript基础笔记集合(转)

    JavaScript基础笔记集合   JavaScript基础笔记集合   js简介 js是脚本语言.浏览器是逐行的读取代码,而传统编程会在执行前进行编译   js存放的位置 html脚本必须放在&l ...

  4. javascript立即调用的函数表达式N种写法(第二篇)

    原文:javascript立即调用的函数表达式N种写法(第二篇) 上一篇博客我谈到将函数声明转换为函数表达式最常见的一种写法是:通过括号()将匿名函数声明转换为函数表达式即(function(){}) ...

  5. 《JavaScript高级程序设计》笔记:函数表达式(七)

    递归 function factorial(num){ if(num<=1){ return 1; }else { return num * arguments.callee(num-1); } ...

  6. JS学习笔记 (五) 函数进阶

    1.函数基础 1.1 函数的基本概念 函数是一段JavaScript代码,只被定义一次,但是可以被调用或者执行许多次.函数是一种对象,可以设置属性,或调用方法. 函数中的参数分为实参和形参.其中,形参 ...

  7. 理解JavaScript的立即调用函数表达式(IIFE)

    首先这是js的一种函数调用写法,叫立即执行函数表达式(IIFE,即immediately-invoked function expression).顾名思义IIFE可以让你的函数立即得到执行(废话). ...

  8. JavaScript基础笔记(三) 引用类型

    引用类型 引用类型的值(对象)是引用类型的一个实例. 一.Object类型 创建Object实例: //方法一:通过new操作符创建 var a = new Object(); a.neme = &q ...

  9. JavaScript基础知识(函数)

    函数的基础 函数: 把实现相同功能的代码放到一个函数体中,当想实现这个功能时,直接执行这个函数即可:减少了的冗余:高内聚,低耦合--> 函数的封装: 函数:引用数据类型: var a = 10; ...

随机推荐

  1. Centos6.10部署TeamViewer

    1.在官网下载支持Linux系统的包,建议下载TeamViewer12的包,官网URL:https://www.teamviewer.com/cn/download/linux/ 2.将下载的软件包导 ...

  2. SSM文件下载

    SSM框架文件下载比文件上传稍微麻烦一点,但这次还是写成最简朴的形式,哈哈~如下 参考:http://blog.csdn.net/lcx556224523/article/details/702076 ...

  3. 论文阅读笔记二十五:Spatial Pyramid Pooling in Deep Convolutional Networks for Visual Recognition(SPPNet CVPR2014)

    论文源址:https://arxiv.org/abs/1406.4729 tensorflow相关代码:https://github.com/peace195/sppnet 摘要 深度卷积网络需要输入 ...

  4. python 给对象绑定属性和方法和__slots__的使用

    # 以c语言为主是静态语言,运行之前先编译,在运行的过程中不允许编辑代码# 在运行的过程中,可以改变,可以添加属性,就是属于动态语言(python) # python动态的添加属性以及方法class ...

  5. 012-Python-paramiko和IO多路复用

    1.IO 多路复用 1.监听多个socket变化 2.socket服务端 IO多路复用+socket 来实现web服务器: a.服务端优先运行 b.浏览器:http://.......com 浏览器连 ...

  6. 腾讯应用宝Android 应用加固(乐固)操作说明(转)

    此处引用腾讯云对加固的优点说明如下: 为什么应用需要加固? 若应用不做任何安全防护,极易被病毒植入.广告替换.支付渠道篡改.钓鱼.信息劫持等,严重侵害开发者的利益. 应用进行安全防护,防止应用分发后, ...

  7. python之集合set

    1.测试 # python2和python3方法列表相同 ops23 = ['add', 'clear', 'copy', 'difference', 'difference_update', 'di ...

  8. spring security实现动态配置url权限的两种方法

    缘起 标准的RABC, 权限需要支持动态配置,spring security默认是在代码里约定好权限,真实的业务场景通常需要可以支持动态配置角色访问权限,即在运行时去配置url对应的访问角色. 基于s ...

  9. Coolpy网络部署说明(局域网)

    本文将介绍Coolpy第一种方案的网络部署方法.以方便大家学习如何让coolpy设备部署到相应的应用场景中.本例将以水星MW310R无线路由器作为演示路由器. 1.局域网部署 即coolpy设备=&g ...

  10. js网页下载csv格式的表格

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...