- 浏览: 36559 次
- 性别:
- 来自: 上海
-
最新评论
原文:javascript面向对象技术基础(六)
作用域、闭包、模拟私有属性
先来简单说一下变量作用域,这些东西我们都很熟悉了,所以也不详细介绍。
- var sco = "global" ; //全局变量
- function t() {
- var sco = "local" ; //函数内部的局部变量
- alert(sco); //local 优先调用局部变量
- }
- t(); //local
- alert(sco); //global 不能使用函数内的局部变量
var sco = "global"; //全局变量 function t() { var sco = "local"; //函数内部的局部变量 alert(sco); //local 优先调用局部变量 } t(); //local alert(sco); //global 不能使用函数内的局部变量
注意一点,在javascript中没有块级别的作用域,也就是说在java或c/c++中我们可以用"{}"来包围一个块,从而在其中定义块内的 局部变量,在"{}"块外部,这些变量不再起作用,同时,也可以在for循环等控制语句中定义局部的变量,但在javascript中没有此项特性:
- function f(props) {
- for ( var i=0; i<10; i++) {}
- alert(i); //10 虽然i定义在for循环的控制语句中,但在函数
- //的其他位置仍旧可以访问该变量.
- if (props == "local" ) {
- var sco = "local" ;
- alert(sco);
- }
- alert(sco); //同样,函数仍可引用if语句内定义的变量
- }
- f("local" ); //10 local local
function f(props) { for(var i=0; i<10; i++) {} alert(i); //10 虽然i定义在for循环的控制语句中,但在函数 //的其他位置仍旧可以访问该变量. if(props == "local") { var sco = "local"; alert(sco); } alert(sco); //同样,函数仍可引用if语句内定义的变量 } f("local"); //10 local local
在函数内部定义局部变量时要格外小心:
- var sco = "global" ;
- function print1() {
- alert(sco); //global
- }
- function print2() {
- var sco = "local" ;
- alert(sco); //local
- }
- function print3() {
- alert(sco); //undefined
- var sco = "local" ;
- alert(sco); local
- }
- print1(); //global
- print2(); //local
- print3(); //undefined local
var sco = "global"; function print1() { alert(sco); //global } function print2() { var sco = "local"; alert(sco); //local } function print3() { alert(sco); //undefined var sco = "local"; alert(sco); local } print1(); //global print2(); //local print3(); //undefined local
前面两个函数都很容易理解,关键是第三个:第一个alert语句并没有把全局变量"global"显示出来,而是undefined,这是因为在 print3函数中,我们定义了sco局部变量(不管位置在何处),那么全局的sco属性在函数内部将不起作用,所以第一个alert中sco其实是局部 sco变量,相当于:
- function print3() {
- var sco;
- alert(sco);
- sco = "local" ;
- alert(sco);
- }
function print3() { var sco; alert(sco); sco = "local"; alert(sco); }
从这个例子我们得出,在函数内部定义局部变量时,最好是在开头就把所需的变量定义好,以免出错。
函数的作用域在定义函数的时候已经确定了,例如:
- var scope = "global" //定义全局变量
- function print() {
- alert(scope);
- }
- function change() {
- var scope = "local" ; //定义局部变量
- print(); //虽然是在change函数的作用域内调用print函数,
- //但是print函数执行时仍旧按照它定义时的作用域起作用
- }
- change(); //golbal
var scope = "global" //定义全局变量 function print() { alert(scope); } function change() { var scope = "local"; //定义局部变量 print(); //虽然是在change函数的作用域内调用print函数, //但是print函数执行时仍旧按照它定义时的作用域起作用 } change(); //golbal
闭包
闭包是拥有变量、代码和作用域的表达式.在javascript中,函数就是变量、代码和函数的作用域的组合体,因此所有的函数都是闭包
(JavaScript functions are a combination of code to be executed and the
scope in which to execute them. This combination of code and scope is
known as a closure in the computer science literature.All JavaScript
functions are closures).好像挺简单.但是闭包到底有什么作用呢?看一个例子。
我们想写一个方法,每次都得到一个整数,这个整数是每次加1的,没有思索,马上下笔:
- var i = 0;
- function getNext() {
- i++;
- return i;
- }
- alert(getNext()); //1
- alert(getNext()); //2
- alert(getNext()); //3
var i = 0; function getNext() { i++; return i; } alert(getNext()); //1 alert(getNext()); //2 alert(getNext()); //3
一直用getNext函数得到下一个整数,而后不小心或者故意的将全局变量i的值设为0,然后再次调用getNext,你会发现又从1开始
了........这时你会想到,要是把i设置成一个私有变量该多好,这样只有在方法内部才可能改变它,在函数之外就没有办法修改了.下面的代码就是按照
这个要求来做得,后面我们详细讨论。
为了解释方便,我们就把下面的代码称为demo1.
- function temp() {
- var i = 0;
- function b() {
- return ++i;
- }
- return b;
- }
- var getNext = temp();
- alert(getNext()); //1
- alert(getNext()); //2
- alert(getNext()); //3
- alert(getNext()); //4
function temp() { var i = 0; function b() { return ++i; } return b; } var getNext = temp(); alert(getNext()); //1 alert(getNext()); //2 alert(getNext()); //3 alert(getNext()); //4
因为我们平时所说的javascript绝大多数都是指的在客户端(浏览器)下,所以这里也不例外。在javascript解释器启动时,会首先创 建一个全局的对象(global object),也就是"window"所引用的对象.然后我们定义的所有全局属性和方法等都会成为这个对象的属性.不同的函数和变量的作用域是不同的, 因而构成了一个作用域链(scope chain).很显然,在javascript解释器启动时,这个作用域链只有一个对象:window(Window Object,即global object).
在demo1中,temp函数是一个全局函数,因此temp()函数的作用域(scopr)对应的作用域链就是js解释器启动时的作用域链,只有一个
window对象。当temp执行时,首先创建一个call对象(活动对象),然后把这个call对象添加到temp函数对应的作用域链的最前头,这
是,temp()函数对应的作用域链就包含了两个对象:window对象和temp函数对应的call
object(活动对象).然后呢,因为我们在temp函数里定义了变量i,定义了函数b(),这些都会成为call
object的属性。当然,在这之前会首先给call
object对象添加arguments属性,保存了temp()函数执行时传递过来的参数。此时,整个的作用域链如下图所示:
同理可以得出函数b()执行时的整个作用域链:
注意在b()的作用域链中,b()函数对应的call object只有一个arguemnts属性,并没有i属性,这是因为在b()的定义中,并没有用var关键字来
声明i属性,只有用var 关键字声明的属性才会添加到对应的call object上.在函数执行时,首先查找对应的call object有没有需要的属性,如果没有,再往上一级查找,直到找到为止,如果找不到,那就是undefined了.
这样我们再来看demo1的执行情况。我们用getNext引用了temp函数,而temp函数返回了函数b,这样getNext函数其实就是b函数的引
用。执行一次getNext,就执行一次b()函数。因为函数b()的作用域依赖于函数temp,因此temp函数在内存中会一直存在。函数b执行时,首
先查找i,在b对应的call object中没有,于是往上一级找,在temp函数对应的call
object中找到了,于是将其值加1,然后返回这个值。这样,只要getNext函数有效,那么b()函数就一直有效,同时,b()函数依赖的temp
函数也不会消失,变量i也不会消失,而且这个变量在temp函数外部根本就访问不到,只能在temp()函数内部访问(b当然可以了).
来看一个利用闭包来模拟私有属性的例子:
- function Person(name, age) {
- this .getName = function () { return name; };
- this .setName = function (newName) { name = newName; };
- this .getAge = function () { return age; };
- this .setAge = function (newAge) { age = newAge; };
- }
- var p1 = new Person( "sdcyst" ,3);
- alert(p1.getName()); //sdcyst
- alert(p1.name); //undefined 因为Person('类')没有name属性
- p1.name = "mypara" //显示的给p1添加name属性
- alert(p1.getName()); //sdcyst 但是并不会改变getName方法的返回值
- alert(p1.name); //mypara 显示出p1对象的name属性
- p1.setName("sss" ); //改变私有的"name"属性
- alert(p1.getName()); //sss
- alert(p1.name); //仍旧为mypara
function Person(name, age) { this.getName = function() { return name; }; this.setName = function(newName) { name = newName; }; this.getAge = function() { return age; }; this.setAge = function(newAge) { age = newAge; }; } var p1 = new Person("sdcyst",3); alert(p1.getName()); //sdcyst alert(p1.name); //undefined 因为Person('类')没有name属性 p1.name = "mypara" //显示的给p1添加name属性 alert(p1.getName()); //sdcyst 但是并不会改变getName方法的返回值 alert(p1.name); //mypara 显示出p1对象的name属性 p1.setName("sss"); //改变私有的"name"属性 alert(p1.getName()); //sss alert(p1.name); //仍旧为mypara
定义了一个Person类,有两个私有属性name,age,分别定义对应的get/set方法。
虽然可以显示的设置p1的name、age属性,但是这种显示的设置,并不会改变我们
最初设计时模拟出来的"name/age"私有属性。
解释闭包的确不是一件容易的事,在网上很多人也是利用例子来说明闭包。如果有地方说的不对,还请指正。
下面是另一篇解释javascript闭包的文章,一块儿参考吧。
http://softbbs.pconline.com.cn/9497825.html
发表评论
-
javascript 获取单选按钮的值(转载)
2010-10-14 10:40 1028原文:javascript 获取单选按钮的值 ... -
javascript面向对象技术基础(五)(转载)
2010-10-14 10:37 675原文:javascript面向对象技术基础(五) 类变 ... -
javascript面向对象技术基础(四)(转载)
2010-10-14 10:36 609原文:javascript面向对象技术基础(四) 类、 ... -
javascript面向对象技术基础(三)
2010-10-14 10:35 726原文:javascript面向对象技术基础(三) ... -
javascript面向对象技术基础(二)(转载)
2010-10-14 10:34 752原文:javascript面向对象技术基础(二) 数组 ... -
javascript技术基础1(转载)
2010-10-14 10:31 1054javascript面向对象技术基础(一) ...
相关推荐
JavaScript面向对象技术是该编程语言非常核心的一个部分,它允许开发者使用对象来组织代码,并模拟现实世界中的实体。在JavaScript中,几乎所有的东西都可以被视为对象,包括基本数据类型如数字和字符串。对象包含...
本书全面覆盖了面向对象编程的基础理论,并结合JavaScript的具体实现进行讲解。 - **第一章:JavaScript简介**:介绍JavaScript的发展历程、特点及其与Web浏览器的关系。 - **第二章:面向对象编程简介**:解释OOP...
JavaScript 面向对象技术是编程中的核心概念,它允许我们以更加接近人类思维的方式组织代码,通过模拟现实世界中的对象来实现复杂的功能。本文将深入探讨JavaScript中的面向对象技术,包括对象、数组、函数、类、...
JavaScript面向对象编程指南
本文介绍了JavaScript面向对象编程的基本概念和技术细节,包括变量和对象的基础用法、函数的作用以及如何通过封装和继承来构建复杂的对象层次结构。JavaScript的独特之处在于它的灵活性和动态性,这使得它成为了一种...
面向对象编程的基础包括类、对象、继承和多态等概念。类是对象的模板或蓝图,定义了一组属性(数据成员)和方法(函数)。对象则是类的实例,具备类所定义的属性和行为。JavaScript虽然没有传统的类定义,但可以通过...
### JavaScript面向对象进阶知识点详解 #### 8.1 面向对象编程的基本特性 在探讨面向对象的JavaScript之前,我们首先需要了解面向对象编程(Object-Oriented Programming, OOP)的基本特性:封装性、抽象性、继承...
JavaScript面向对象编程指南.pdf
1. 基础对象创建和属性操作:理解如何创建对象,包括字面量语法和构造函数,以及访问和修改对象的属性。 2. 构造函数与原型链:详细解释构造函数的工作原理,以及如何利用原型实现继承。 3. `new` 关键字的作用:...
《JavaScript面向对象精要》这本书不仅介绍了JavaScript面向对象的基础概念,还深入探讨了其实现机制及其在实际开发中的应用。对于希望提高自己JavaScript技能水平的开发者来说,本书是一本不可多得的好书。通过学习...
本文将详细介绍JavaScript中的面向对象编程概念和技术,包括类定义、对象创建、继承、封装等内容。 #### 二、类定义和对象创建 ##### 2.1 类定义 在JavaScript中,“类”这一概念并不像在Java或C#这样的强类型...
- **现代JavaScript与面向对象编程**:现代JavaScript程序设计中融入了许多新的概念和技术,这些技术让JavaScript应用程序与以往相比有了显著的不同。面向对象编程是现代JavaScript的一个重要组成部分。 #### 二、...
面向对象编程的基础知识及其在JavaScript中的运用;数据类型、操作符以及流程控制语句;函数、闭包、对象和原型等概念,以代码重用为目的的继承模式;BOM、DOM、浏览器事件、AJAX和JSON;如何实现JavaScript中缺失的...
面向对象编程(Object-Oriented Programming,OOP)是JavaScript中的一个重要概念,它允许开发者以更加模块化、可复用的方式组织代码。下面将详细探讨JavaScript面向对象编程的基本原理、特性以及实际应用。 1. **...
javascript面向对象编程指南 2nd英文版,英文名:Object-Oriented JavaScript。 What you will learn from this book The basics of object-oriented programming, and how to apply it in the JavaScript ...
JavaScript面向对象编程指南是完整的扫描版...
JavaScript面向对象编程指南完整版是扫描的....
在本文中,我们将深入探讨 JavaScript 的面向对象基础,主要包括类的定义、实例化以及对象属性和方法的引用。 首先,让我们了解如何在 JavaScript 中定义一个类。在 JavaScript 中,类通常通过函数来实现。例如,...