`
q272156430
  • 浏览: 275830 次
  • 性别: Icon_minigender_1
  • 来自: 成都
社区版块
存档分类
最新评论

javaScript 面象对象 6

阅读更多

作用域、闭包、模拟私有属性
先来简单说一下变量作用域,这些东西我们都很熟悉了,所以也不详细介绍。

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  

 

 在函数内部定义局部变量时要格外小心:

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);   
}  

 

 从这个例子我们得出,在函数内部定义局部变量时,最好是在开头就把所需的变量定义好,以免出错。

函数的作用域在定义函数的时候已经确定了,例如:

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  

 

一直用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  

 

因为我们平时所说的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    

 

定义了一个Person类,有两个私有属性name,age,分别定义对应的get/set方法。
虽然可以显示的设置p1的name、age属性,但是这种显示的设置,并不会改变我们
最初设计时模拟出来的"name/age"私有属性。

解释闭包的确不是一件容易的事,在网上很多人也是利用例子来说明闭包。如果有地方说的不对,还请指正。
下面是另一篇解释javascript闭包的文章,一块儿参考吧。

分享到:
评论

相关推荐

    javascript面象对象编程

    JavaScript中的面向对象编程(Object-Oriented Programming, OOP)是一种设计模式,它允许开发者创建具有封装、继承和多态性的复杂结构。在JavaScript中,虽然没有像Java或C++那样的类关键字,但它通过函数和原型...

    JavaScript面象对象设计

    ### JavaScript面向对象设计详解 #### 一、面向对象的基本概念 面向对象编程(Object-Oriented Programming,简称 OOP)是一种编程范式,其核心思想是将程序中的数据和操作这些数据的方法组织在一起,形成一个整体...

    Javascript面象对象成员、共享成员变量实验

    在探讨JavaScript面向对象编程时,成员变量和共享成员变量是非常重要的概念。JavaScript中的对象可以视为具有属性(成员变量)和方法(成员函数)的实例。我们通过构造函数或者直接使用对象字面量来创建对象,并且...

    经典Javacript的面向对象设计例子

    javascript面象对象设计经典例子

    js面向对象游戏开发,飞机大战

    在JavaScript的世界里,面向对象编程(Object-Oriented Programming,OOP)是一种常见的设计模式,尤其在游戏开发中,它能帮助我们构建复杂且可扩展的代码结构。本项目"js面向对象游戏开发,飞机大战"是利用...

    js面向对象.pdf

    ### JavaScript面向对象编程详解 #### 6.1 JavaScript中支持面向对象的基础 ##### 6.1.1 使用定义函数的方式定义类 在面向对象编程(OOP)中,“类”是核心概念之一,它代表着一类事物的抽象。在JavaScript中,可以...

    ArcGIS for JavaScript 车辆轨迹回放

    在GIS领域,空间数据通常以几何对象(如点、线、面)的形式存在,而时间戳则与这些几何对象关联,记录了每个位置点的时间信息。在车辆轨迹中,每一条轨迹可以看作是一系列有序的地理坐标点,每个点都有对应的时间...

    Javascript添加要素

    在这些场景中,"要素"通常指的是地图上的点、线和面对象,它们用于表示地理实体。本教程将深入探讨如何利用JavaScript实现这一功能。 首先,让我们了解“点”要素。在JavaScript中,一个点要素通常表示一个精确的...

    HTMLTest(js面向对象)

    HTMLTest(js面向对象),javascript也有面象对象的概念,有封装,有继承。事件模型

    判断点是否在面内

    2. 创建面对象:接着,我们需要构建Polygon对象来表示多边形。这通常涉及提供一个包含多个坐标对的数组,这些坐标对定义了多边形的边界。 ```javascript var polygonCoordinates = [[...], [...], ...]; // 多边形...

    Jquery UI插件

    如果你想学习javascript面象对象编程,这个源代码可以让你学习学习。

    点,线,面对象转成GeoJson格式存储起来

    它包括了点(Point)、线(LineString)、面(Polygon)等几何类型,以及Feature对象,用于包含属性数据。GeoJSON的特点是结构清晰,易于解析,支持多种GIS操作。 2. **SuperMap .NET库** SuperMap是一个全面的GIS...

    63.(leaflet篇)leaflet+turf获取geojson面的中心点与质心.rar

    在本教程中,我们将深入探讨如何使用Leaflet和Turf.js来获取GeoJSON几何对象(如面)的中心点和质心。 首先,了解GeoJSON是一种开放的数据格式,常用于存储地理空间数据,包括点、线、面等多种类型。GeoJSON对象由...

    arcgis server javascript api 点线面buffer

    ### ArcGIS Server JavaScript API 实现点、线、面的 Buffer 分析 #### 一、概述 ArcGIS Server JavaScript API 是一款强大的工具集,允许开发者在 Web 应用程序中集成地理信息系统 (GIS) 功能。它支持多种 GIS ...

    理解Javascript

    01_理解内存分配在正式开始之前,我想先说两句,理解javascript系列博文是通过带领大家分析javascript执行时的内存分配情况,来解释javascript原理,具体会涵盖javascript预加载,闭包原理,面象对象,执行模型,...

    GeoJson规范 (2).pdf

    6. MultiPolygon:多面是一个面对象的数组,表示多个面域的位置。 7. GeometryCollection:几何集合是一个geometry对象的数组,表示多种几何体的集合。 Feature Objects 要素对象是GeoJSON对象的子集,包含一个...

    GIS通过js判断点是否在面对象中

    而面对象(多边形)是由一系列线段(边)围成的闭合区域,可以代表地理上的区域,如行政区、湖泊、森林等。 在JavaScript中,我们可以利用库如Leaflet、OpenLayers或Mapbox GL JS来处理GIS相关的任务。这些库提供了...

Global site tag (gtag.js) - Google Analytics