新聞中心
看了很多介紹javascript面向?qū)ο蠹夹g(shù)的文章,很暈.為什么?不是因為寫得不好,而是因為太深奧.javascript中的對象還沒解釋清楚怎么回事,一上來就直奔主題,類/繼承/原型/私有變量。結(jié)果呢,看了大半天,有了一個大概的了解,細細一回味,好像什么都沒懂。

創(chuàng)新互聯(lián)于2013年創(chuàng)立,是專業(yè)互聯(lián)網(wǎng)技術(shù)服務公司,擁有項目成都網(wǎng)站建設、成都網(wǎng)站設計網(wǎng)站策劃,項目實施與項目整合能力。我們以讓每一個夢想脫穎而出為使命,1280元懷安做網(wǎng)站,已為上家服務,為懷安各地企業(yè)和個人服務,聯(lián)系電話:13518219792
這篇文章是參考<
作用域、閉包、模擬私有屬性
先來簡單說一下變量作用域,這些東西我們都很熟悉了,所以也不詳細介紹。
Js代碼
- var sco = "global"; //全局變量
- function t() {
- var sco = "local"; //函數(shù)內(nèi)部的局部變量
- alert(sco); //local 優(yōu)先調(diào)用局部變量
- }
- t(); //local
- alert(sco); //global 不能使用函數(shù)內(nèi)的局部變量
注意一點,在javascript中沒有塊級別的作用域,也就是說在java或c/c++中我們可以用"{}"來包圍一個塊,從而在其中定義塊內(nèi)的局部變量,在"{}"塊外部,這些變量不再起作用,同時,也可以在for循環(huán)等控制語句中定義局部的變量,但在javascript中沒有此項特性:
Js代碼
- function f(props) {
- for(var i=0; i<10; i++) {}
- alert(i); //10 雖然i定義在for循環(huán)的控制語句中,但在函數(shù)
- //的其他位置仍舊可以訪問該變量.
- if(props == "local") {
- var sco = "local";
- alert(sco);
- }
- alert(sco); //同樣,函數(shù)仍可引用if語句內(nèi)定義的變量
- }
- f("local"); //10 local local
在函數(shù)內(nèi)部定義局部變量時要格外小心:
Js代碼
- 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
前面兩個函數(shù)都很容易理解,關(guān)鍵是第三個:第一個alert語句并沒有把全局變量"global"顯示出來,而是undefined,這是因為在print3函數(shù)中,我們定義了sco局部變量(不管位置在何處),那么全局的sco屬性在函數(shù)內(nèi)部將不起作用,所以第一個alert中sco其實是局部sco變量,相當于:
Js代碼
- function print3() {
- var sco;
- alert(sco);
- sco = "local";
- alert(sco);
- }
從這個例子我們得出,在函數(shù)內(nèi)部定義局部變量時,最好是在開頭就把所需的變量定義好,以免出錯。
函數(shù)的作用域在定義函數(shù)的時候已經(jīng)確定了,例如:
Js代碼
- var scope = "global" //定義全局變量
- function print() {
- alert(scope);
- }
- function change() {
- var scope = "local"; //定義局部變量
- print(); //雖然是在change函數(shù)的作用域內(nèi)調(diào)用print函數(shù),
- //但是print函數(shù)執(zhí)行時仍舊按照它定義時的作用域起作用
- }
- change(); //golbal
閉包
閉包是擁有變量、代碼和作用域的表達式.在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).好像挺簡單.
但是閉包到底有什么作用呢?看一個例子。
我們想寫一個方法,每次都得到一個整數(shù),這個整數(shù)是每次加1的,沒有思索,馬上下筆:
Js代碼
- var i = 0;
- function getNext() {
- i++;
- return i;
- }
- alert(getNext()); //1
- alert(getNext()); //2
- alert(getNext()); //3
一直用getNext函數(shù)得到下一個整數(shù),而后不小心或者故意的將全局變量i的值設為0,然后再次調(diào)用getNext,你會發(fā)現(xiàn)又從1開始了........這時你會想到,要是把i設置成一個私有變量該多好,這樣只有在方法內(nèi)部才可能改變它,在函數(shù)之外就沒有辦法修改了.下面的代碼就是按照這個要求來做得,后面我們詳細討論。
為了解釋方便,我們就把下面的代碼稱為demo1.
Js代碼
- 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絕大多數(shù)都是指的在客戶端(瀏覽器)下,所以這里也不例外。在javascript解釋器啟動時,會首先創(chuàng)建一個全局的對象(global object),也就是"window"所引用的對象.然后我們定義的所有全局屬性和方法等都會成為這個對象的屬性.不同的函數(shù)和變量的作用域是不同的,因而構(gòu)成了一個作用域鏈(scope chain).
很顯然,在javascript解釋器啟動時,這個作用域鏈只有一個對象:window(Window Object,即global object).在demo1中,temp函數(shù)是一個全局函數(shù),因此temp()函數(shù)的作用域(scopr)對應的作用域鏈就是js解釋器啟動時的作用域鏈,只有一個window對象。
當temp執(zhí)行時,首先創(chuàng)建一個call對象(活動對象),然后把這個call對象添加到temp函數(shù)對應的作用域鏈的最前頭,這是,temp()函數(shù)對應的作用域鏈就包含了兩個對象:window對象和temp函數(shù)對應的call object(活動對象).然后呢,因為我們在temp函數(shù)里定義了變量i,定義了函數(shù)b(),這些都會成為call object的屬性。當然,在這之前會首先給call object對象添加arguments屬性,保存了temp()函數(shù)執(zhí)行時傳遞過來的參數(shù)。此時,整個的作用域鏈如下圖所示:
同理可以得出函數(shù)b()執(zhí)行時的整個作用域鏈:
注意在b()的作用域鏈中,b()函數(shù)對應的call object只有一個arguemnts屬性,并沒有i屬性,這是因為在b()的定義中,并沒有用var關(guān)鍵字來聲明i屬性,只有用var 關(guān)鍵字聲明的屬性才會添加到對應的call object上.
在函數(shù)執(zhí)行時,首先查找對應的call object有沒有需要的屬性,如果沒有,再往上一級查找,直到找到為止,如果找不到,那就是undefined了.
這樣我們再來看demo1的執(zhí)行情況。我們用getNext引用了temp函數(shù),而temp函數(shù)返回了函數(shù)b,這樣getNext函數(shù)其實就是b函數(shù)的引用。
執(zhí)行一次getNext,就執(zhí)行一次b()函數(shù)。因為函數(shù)b()的作用域依賴于函數(shù)temp,因此temp函數(shù)在內(nèi)存中會一直存在。函數(shù)b執(zhí)行時,首先查找i,在b對應的call object中沒有,于是往上一級找,在temp函數(shù)對應的call object中找到了,于是將其值加1,然后返回這個值。
這樣,只要getNext函數(shù)有效,那么b()函數(shù)就一直有效,同時,b()函數(shù)依賴的temp函數(shù)也不會消失,變量i也不會消失,而且這個變量在temp函數(shù)外部根本就訪問不到,只能在temp()函數(shù)內(nèi)部訪問(b當然可以了).
來看一個利用閉包來模擬私有屬性的例子:
Js代碼
- 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"私有屬性。
解釋閉包的確不是一件容易的事,在網(wǎng)上很多人也是利用例子來說明閉包。如果有地方說的不對,還請指正。
【編輯推薦】
- javascript面向?qū)ο蠹夹g(shù)基礎(一)
- javascript面向?qū)ο蠹夹g(shù)基礎(二)
- javascript面向?qū)ο蠹夹g(shù)基礎(三)
- javascript面向?qū)ο蠹夹g(shù)基礎(四)
- javascript面向?qū)ο蠹夹g(shù)基礎(五)
當前標題:javascript面向?qū)ο蠹夹g(shù)基礎(六)
網(wǎng)站地址:http://m.fisionsoft.com.cn/article/cdhcgig.html


咨詢
建站咨詢
