自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

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

開(kāi)發(fā) 前端
好多JAVASCRIPT的文章,對(duì)于初學(xué)者來(lái)說(shuō),太深?yuàn)W,難理解。本系列會(huì)從基礎(chǔ)開(kāi)始講起,今天介紹這一系列的最后一篇,作用域、閉包和模擬私有屬性,一起來(lái)看。

看了很多介紹javascript面向?qū)ο蠹夹g(shù)的文章,很暈.為什么?不是因?yàn)閷懙貌缓?而是因?yàn)樘願(yuàn)W.javascript中的對(duì)象還沒(méi)解釋清楚怎么回事,一上來(lái)就直奔主題,類/繼承/原型/私有變量。結(jié)果呢,看了大半天,有了一個(gè)大概的了解,細(xì)細(xì)一回味,好像什么都沒(méi)懂。

這篇文章是參考<<javascript-the definitive guide,5th edition>>第7,8,9章而寫成的,我也會(huì)盡量按照原書(shū)的結(jié)構(gòu)來(lái)說(shuō)明javascript的面向?qū)ο蠹夹g(shù)(對(duì)象/數(shù)組->函數(shù)-->類/構(gòu)造函數(shù)/原型).對(duì)一些我自己也拿捏不準(zhǔn)的地方,我會(huì)附上原文的英文語(yǔ)句,供大家參考.

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

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

Js代碼

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

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

Js代碼

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

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

Js代碼

  1. var sco = "global";  
  2. function print1() {  
  3. alert(sco); //global  
  4. }  
  5. function print2() {  
  6. var sco = "local";  
  7. alert(sco); //local  
  8. }  
  9. function print3() {  
  10. alert(sco); //undefined  
  11. var sco = "local";   
  12. alert(sco); local  
  13. }  
  14. print1(); //global  
  15. print2(); //local  
  16. print3(); //undefined local  

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

Js代碼

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

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

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

Js代碼 

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

閉包

閉包是擁有變量、代碼和作用域的表達(dá)式.在javascript中,函數(shù)就是變量、代碼和函數(shù)的作用域的組合體,因此所有的函數(shù)都是閉包(JavaScript functions are a combination of code to be executed and the scope in which toexecute them. This combination of code and scope is known as a closure in the computer science literature.All JavaScript functions are closures).好像挺簡(jiǎn)單.

但是閉包到底有什么作用呢?看一個(gè)例子。

我們想寫一個(gè)方法,每次都得到一個(gè)整數(shù),這個(gè)整數(shù)是每次加1的,沒(méi)有思索,馬上下筆:

Js代碼 

  1. var i = 0;  
  2. function getNext() {  
  3. i++;  
  4. return i;  
  5. }  
  6. alert(getNext()); //1  
  7. alert(getNext()); //2  
  8. alert(getNext()); //3  

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

為了解釋方便,我們就把下面的代碼稱為demo1.

Js代碼

  1. function temp() {  
  2. var i = 0;  
  3. function b() {  
  4. return ++i;  
  5. }  
  6. return b;  
  7. }  
  8. var getNext = temp();  
  9. alert(getNext()); //1  
  10. alert(getNext()); //2  
  11. alert(getNext()); //3  
  12. alert(getNext()); //4  

因?yàn)槲覀兤綍r(shí)所說(shuō)的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í)傳遞過(guò)來(lái)的參數(shù)。此時(shí),整個(gè)的作用域鏈如下圖所示:

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

注意在b()的作用域鏈中,b()函數(shù)對(duì)應(yīng)的call object只有一個(gè)arguemnts屬性,并沒(méi)有i屬性,這是因?yàn)樵赽()的定義中,并沒(méi)有用var關(guān)鍵字來(lái)聲明i屬性,只有用var 關(guān)鍵字聲明的屬性才會(huì)添加到對(duì)應(yīng)的call object上.

在函數(shù)執(zhí)行時(shí),首先查找對(duì)應(yīng)的call object有沒(méi)有需要的屬性,如果沒(méi)有,再往上一級(jí)查找,直到找到為止,如果找不到,那就是undefined了.

這樣我們?cè)賮?lái)看demo1的執(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中沒(méi)有,于是往上一級(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ù)外部根本就訪問(wèn)不到,只能在temp()函數(shù)內(nèi)部訪問(wèn)(b當(dāng)然可以了).

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

Js代碼 

  1. function 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. var p1 = new Person("sdcyst",3);   
  8. alert(p1.getName()); //sdcyst   
  9. alert(p1.name); //undefined 因?yàn)镻erson('類')沒(méi)有name屬性   
  10. p1.name = "mypara" //顯示的給p1添加name屬性   
  11. alert(p1.getName()); //sdcyst 但是并不會(huì)改變getName方法的返回值   
  12. alert(p1.name); //mypara 顯示出p1對(duì)象的name屬性   
  13. p1.setName("sss"); //改變私有的"name"屬性  
  14. alert(p1.getName()); //sss   
  15. alert(p1.name); //仍舊為mypara  

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

解釋閉包的確不是一件容易的事,在網(wǎng)上很多人也是利用例子來(lái)說(shuō)明閉包。如果有地方說(shuō)的不對(duì),還請(qǐng)指正。

【編輯推薦】

  1. javascript面向?qū)ο蠹夹g(shù)基礎(chǔ)(一)
  2. javascript面向?qū)ο蠹夹g(shù)基礎(chǔ)(二)
  3. javascript面向?qū)ο蠹夹g(shù)基礎(chǔ)(三)
  4. javascript面向?qū)ο蠹夹g(shù)基礎(chǔ)(四)
  5. javascript面向?qū)ο蠹夹g(shù)基礎(chǔ)(五)
責(zé)任編輯:于鐵 來(lái)源: iteye.com
相關(guān)推薦

2011-05-13 11:05:52

javascript

2011-05-13 09:58:46

javascript

2011-05-13 10:51:25

javascript

2011-05-13 11:17:18

javascript

2011-05-13 11:27:59

javascript

2009-06-10 22:06:29

JavaScript面向?qū)ο?/a>

2020-10-20 08:35:34

JS基礎(chǔ)進(jìn)階

2017-04-21 09:07:39

JavaScript對(duì)象編程

2012-01-17 09:34:52

JavaScript

2013-08-21 17:20:49

.NET面向?qū)ο?/a>

2021-10-21 18:47:37

JavaScript面向對(duì)象

2012-02-27 09:30:22

JavaScript

2009-01-04 09:08:30

面向?qū)ο?/a>繼承接口

2011-05-25 10:21:44

Javascript

2011-05-25 10:59:26

Javascript繼承

2023-10-25 13:42:19

Java面向?qū)ο?/a>

2024-05-10 09:28:57

Python面向?qū)ο?/a>代碼

2010-10-08 09:13:15

oop模式JavaScript

2010-06-18 17:49:34

UML面向?qū)ο蠹夹g(shù)

2010-06-17 18:17:36

UML面向?qū)ο蠹夹g(shù)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)