最愛Java

          書山有路勤為徑,學(xué)海無涯苦作舟

          javascript面向?qū)ο蠹夹g(shù)基礎(chǔ)(六)

                  本文轉(zhuǎn)載于javaeye(http://www.javaeye.com/wiki/Object_Oriented_JavaScript/1279-javascript-object-oriented-technology-one),只進(jìn)行了重新排版以便收藏。
                  文中所有英文語句(程序語句除外),都引自<<javascript-the definitive guide,5th edition>>。

          ------------------------------------------------------------------------------------
          作用域、閉包、模擬私有屬性

              先來簡單說一下變量作用域,這些東西我們都很熟悉了,所以也不詳細(xì)介紹。

          1var sco = "global";  //全局變量
          2function t() 
          3    var sco = "local";  //函數(shù)內(nèi)部的局部變量
          4    alert(sco);         //local 優(yōu)先調(diào)用局部變量
          5}

          6t();             //local
          7alert(sco);       //global  不能使用函數(shù)內(nèi)的局部變量

          注意一點(diǎn),在javascript中沒有塊級(jí)別的作用域,也就是說在java或c/c++中我們可以用"{}"來包圍一個(gè)塊,從而在其中定義塊內(nèi)的局部變量,在"{}"塊外部,這些變量不再起作用,同時(shí),也可以在for循環(huán)等控制語句中定義局部的變量,但在javascript中沒有此項(xiàng)特性:

           1function f(props) {
           2    for(var i=0; i<10; i++{}
           3    alert(i);         //10  雖然i定義在for循環(huán)的控制語句中,但在函數(shù)
           4                      //的其他位置仍舊可以訪問該變量.
           5    if(props == "local"{
           6        var sco = "local";
           7    alert(sco); 
           8    }

           9    alert(sco);       //同樣,函數(shù)仍可引用if語句內(nèi)定義的變量
          10}

          11f("local");      //10  local   local

              在函數(shù)內(nèi)部定義局部變量時(shí)要格外小心:

           1var sco = "global";
           2function print1() {
           3    alert(sco);   //global
           4}

           5function print2() {
           6    var sco = "local";
           7    alert(sco);   //local
           8}

           9function print3() {
          10    alert(sco);   //undefined
          11    var sco = "local"
          12    alert(sco);   local
          13}

          14
          15print1();  //global
          16print2();  //local
          17print3();  //undefined  local

                  前面兩個(gè)函數(shù)都很容易理解,關(guān)鍵是第三個(gè):第一個(gè)alert語句并沒有把全局變量"global"顯示出來,而是undefined,這是因?yàn)樵趐rint3函數(shù)中,我們定義了sco局部變量(不管位置在何處),那么全局的sco屬性在函數(shù)內(nèi)部將不起作用,所以第一個(gè)alert中sco其實(shí)是局部sco變量,相當(dāng)于:

          1function print3() {
          2    var sco;
          3    alert(sco);
          4    sco = "local";
          5    alert(sco);
          6}

                  從這個(gè)例子我們得出,在函數(shù)內(nèi)部定義局部變量時(shí),最好是在開頭就把所需的變量定義好,以免出錯(cuò)。

                  函數(shù)的作用域在定義函數(shù)的時(shí)候已經(jīng)確定了,例如:

           1var scope = "global"   //定義全局變量
           2function print() {
           3    alert(scope);
           4}

           5function change() {
           6    var scope = "local";  //定義局部變量
           7    print();              //雖然是在change函數(shù)的作用域內(nèi)調(diào)用print函數(shù),
           8                          //但是print函數(shù)執(zhí)行時(shí)仍舊按照它定義時(shí)的作用域起作用
           9}

          10change();    //golbal

          閉包
                  閉包是擁有變量、代碼和作用域的表達(dá)式.在javascript中,函數(shù)就是變量、代碼和函數(shù)的作用域的組合體,因此所有的函數(shù)都是閉包(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).好像挺簡單。但是閉包到底有什么作用呢?看一個(gè)例子。
                  我們想寫一個(gè)方法,每次都得到一個(gè)整數(shù),這個(gè)整數(shù)是每次加1的,沒有思索,馬上下筆:

          1var i = 0;
          2function getNext() {
          3    i++;
          4    return i;
          5}

          6alert(getNext()); //1
          7alert(getNext()); //2
          8alert(getNext()); //3

                  一直用getNext函數(shù)得到下一個(gè)整數(shù),而后不小心或者故意的將全局變量i的值設(shè)為0,然后再次調(diào)用getNext,你會(huì)發(fā)現(xiàn)又從1開始了........這時(shí)你會(huì)想到,要是把i設(shè)置成一個(gè)私有變量該多好,這樣只有在方法內(nèi)部才可能改變它,在函數(shù)之外就沒有辦法修改了。下面的代碼就是按照這個(gè)要求來做得,后面我們?cè)敿?xì)討論。
              為了解釋方便,我們就把下面的代碼稱為demo1。

           1function temp() {
           2    var i = 0;
           3    function b() {
           4        return ++i;
           5    }

           6    return b;
           7}

           8var getNext = temp();
           9alert(getNext());    //1
          10alert(getNext());    //2
          11alert(getNext());    //3
          12alert(getNext());    //4
              
                  因?yàn)槲覀兤綍r(shí)所說的javascript絕大多數(shù)都是指的在客戶端(瀏覽器)下,所以這里也不例外。
                  在javascript解釋器啟動(dòng)時(shí),會(huì)首先創(chuàng)建一個(gè)全局的對(duì)象(global object),也就是"window"所引用的對(duì)象。然后我們定義的所有全局屬性和方法等都會(huì)成為這個(gè)對(duì)象的屬性。不同的函數(shù)和變量的作用域是不同的,因而構(gòu)成了一個(gè)作用域鏈(scope chain)。很顯然,在javascript解釋器啟動(dòng)時(shí),這個(gè)作用域鏈只有一個(gè)對(duì)象:window(Window Object,即global object)。
                  在demo1中,temp函數(shù)是一個(gè)全局函數(shù),因此temp()函數(shù)的作用域(scopr)對(duì)應(yīng)的作用域鏈就是js解釋器啟動(dòng)時(shí)的作用域鏈,只有一個(gè)window對(duì)象。
                  當(dāng)temp執(zhí)行時(shí),首先創(chuàng)建一個(gè)call對(duì)象(活動(dòng)對(duì)象),然后把這個(gè)call對(duì)象添加到temp函數(shù)對(duì)應(yīng)的作用域鏈的最前頭,這是,temp()函數(shù)對(duì)應(yīng)的作用域鏈就包含了兩個(gè)對(duì)象:window對(duì)象和temp函數(shù)對(duì)應(yīng)的call object(活動(dòng)對(duì)象).然后呢,因?yàn)槲覀冊(cè)趖emp函數(shù)里定義了變量i,定義了函數(shù)b(),這些都會(huì)成為call object的屬性。當(dāng)然,在這之前會(huì)首先給call object對(duì)象添加arguments屬性,保存了temp()函數(shù)執(zhí)行時(shí)傳遞過來的參數(shù)。此時(shí),整個(gè)的作用域鏈如下圖所示:


          同理可以得出函數(shù)b()執(zhí)行時(shí)的整個(gè)作用域鏈:


                  注意在b()的作用域鏈中,b()函數(shù)對(duì)應(yīng)的call object只有一個(gè)arguemnts屬性,并沒有i屬性,這是因?yàn)樵赽()的定義中,并沒有用var關(guān)鍵字來聲明i屬性,只有用var 關(guān)鍵字聲明的屬性才會(huì)添加到對(duì)應(yīng)的call object上。在函數(shù)執(zhí)行時(shí),首先查找對(duì)應(yīng)的call object有沒有需要的屬性,如果沒有,再往上一級(jí)查找,直到找到為止,如果找不到,那就是undefined了。

                  這樣我們?cè)賮砜磀emo1的執(zhí)行情況。我們用getNext引用了temp函數(shù),而temp函數(shù)返回了函數(shù)b,這樣getNext函數(shù)其實(shí)就是b函數(shù)的引用。執(zhí)行一次getNext,就執(zhí)行一次b()函數(shù)。因?yàn)楹瘮?shù)b()的作用域依賴于函數(shù)temp,因此temp函數(shù)在內(nèi)存中會(huì)一直存在。函數(shù)b執(zhí)行時(shí),首先查找i,在b對(duì)應(yīng)的call object中沒有,于是往上一級(jí)找,在temp函數(shù)對(duì)應(yīng)的call object中找到了,于是將其值加1,然后返回這個(gè)值。這樣,只要getNext函數(shù)有效,那么b()函數(shù)就一直有效,同時(shí),b()函數(shù)依賴的temp函數(shù)也不會(huì)消失,變量i也不會(huì)消失,而且這個(gè)變量在temp函數(shù)外部根本就訪問不到,只能在temp()函數(shù)內(nèi)部訪問(b當(dāng)然可以了)。

          來看一個(gè)利用閉包來模擬私有屬性的例子:

           1function Person(name, age) {  
           2    this.getName = function() return name; };  
           3    this.setName = function(newName) { name = newName; };  
           4    this.getAge = function() return age; };  
           5    this.setAge = function(newAge) { age = newAge; };  
           6}
            
           7  
           8var p1 = new Person("sdcyst",3);  
           9alert(p1.getName());  //sdcyst  
          10alert(p1.name);       //undefined   因?yàn)镻erson('類')沒有name屬性  
          11p1.name = "mypara"    //顯示的給p1添加name屬性  
          12alert(p1.getName());  //sdcyst     但是并不會(huì)改變getName方法的返回值  
          13alert(p1.name);       //mypara     顯示出p1對(duì)象的name屬性  
          14p1.setName("sss");    //改變私有的"name"屬性
          15alert(p1.getName());  //sss  
          16alert(p1.name);       //仍舊為mypara  

          定義了一個(gè)Person類,有兩個(gè)私有屬性name,age,分別定義對(duì)應(yīng)的get/set方法。雖然可以顯示的設(shè)置p1的name、age屬性,但是這種顯示的設(shè)置,并不會(huì)改變我們最初設(shè)計(jì)時(shí)模擬出來的"name/age"私有屬性。

          解釋閉包的確不是一件容易的事,在網(wǎng)上很多人也是利用例子來說明閉包。如果有地方說的不對(duì),還請(qǐng)指正。
          下面是另一篇解釋javascript閉包的文章,一塊兒參考吧。
          http://softbbs.pconline.com.cn/9497825.html

          posted on 2009-10-20 11:47 Brian 閱讀(277) 評(píng)論(0)  編輯  收藏 所屬分類: JScript

          公告


          導(dǎo)航

          <2009年10月>
          27282930123
          45678910
          11121314151617
          18192021222324
          25262728293031
          1234567

          統(tǒng)計(jì)

          常用鏈接

          留言簿(4)

          隨筆分類

          隨筆檔案

          收藏夾

          搜索

          最新評(píng)論

          閱讀排行榜

          評(píng)論排行榜

          主站蜘蛛池模板: 靖西县| 开阳县| 永和县| 新宁县| 修武县| 新丰县| 洪洞县| 嵊州市| 博兴县| 连江县| 荆州市| 海盐县| 昌乐县| 友谊县| 西昌市| 崇仁县| 察隅县| 芜湖市| 昌宁县| 泰宁县| 绥阳县| 定兴县| 平江县| 班玛县| 石狮市| 桐梓县| 平利县| 永吉县| 巴东县| 广安市| 香格里拉县| 都昌县| 古蔺县| 镇雄县| 永兴县| 简阳市| 定南县| 绥中县| 乌拉特前旗| 如东县| 麻阳|