- 相關(guān)推薦
明白javascript封裝
封裝可以被定義為對對象的內部數據表現形式和實(shí)現細節進(jìn)行隱藏。通過(guò)封裝可以強制實(shí)施信息隱藏。
在JavaScript中,并沒(méi)有顯示的聲明私有成員的關(guān)鍵字等。所以要想實(shí)現封裝/信息隱藏就需要從另外的思路出發(fā)。我們可以使用閉包的概念來(lái)創(chuàng )建只允許從對象內部訪(fǎng)問(wèn)的方法和屬性,來(lái)達到封裝的要求。
基本方式
一般來(lái)說(shuō),我們學(xué)用的有三種方法來(lái)達到封裝的目的。
使用this.XXX來(lái)聲明一個(gè)變量,然后再聲明getXXX、setXXX等取值、賦值的方法。
使用this._XXX來(lái)聲明一個(gè)變量,然后再聲明getXXX、setXXX等取值、賦值的方法。
利用“函數作用域”這一個(gè)概念來(lái)做。
1. 門(mén)戶(hù)大開(kāi)型
var Book = function(isbn,title,author){ this.setIsbn(isbn); this.setTitle(title); this.setAuthor(author); }; Book.prototype = { setIsbn: function(isbn){ this.isbn = isbn; }, getIsbn: function(){ return this.isbn; }, setTitle: function(title){ this.title = title; }, getTitle: function(){ return this.title; }, setAuthor: function(author){ this.author = author; }, getAuthor: function(){ return this.author; } };
使用這種方法實(shí)現的封裝,雖然實(shí)現了取值器與賦值器以保護私有屬性。但是在實(shí)際使用中,私有屬性依然可以從外部訪(fǎng)問(wèn),所以從根本上講,沒(méi)有實(shí)現封裝。
2. 用命名規范進(jìn)行區別
var Book = function(isbn,title,author){ this.setIsbn(isbn); this.setTitle(title); this.setAuthor(author); }; Book.prototype = { setIsbn: function(isbn){ this._isbn = isbn; }, getIsbn: function(){ return this._isbn; }, setTitle: function(title){ this._title = title; }, getTitle: function(){ return this._title; }, setAuthor: function(author){ this._author = author; }, getAuthor: function(){ return this._author; } };
使用這種方法與第一種類(lèi)似,區別在于使用不同的命名來(lái)保護私有屬性的使用。但是,從實(shí)際應用來(lái)說(shuō)其仍然沒(méi)有實(shí)現封裝。
3. 使用函數作用域
var Book = function(newIsbn,newTitle,newAuthor){ var isbn,title,author; this.setIsbn=function(newIsbn){ isbn = newIsbn; }; this.getIsbn=function(){ return isbn; }; this.setTitle=function(newTitle){ title = newTitle; }; this.getTitle=function(){ return title; }; this.setIsbn=function(newAuthor){ author = newAuthor; }; this.getIsbn=function(){ return author; }; }
由于在JavaScript的函數中聲明的變量是有作用域的,所以使用這種方法可以避免在外部直接訪(fǎng)問(wèn)私有屬性;具_到封裝所要求的內容。
這里要注意的是,我們在函數的內部,可以使用this.XXX以及var來(lái)聲明變量。區別是使用this.XXX聲明的變量在外部是可以訪(fǎng)問(wèn)的。使用var聲明的變量,由于受到函數作用域的保護,在函數的外部是無(wú)法直接訪(fǎng)問(wèn)的。
4. 使用函數作用域的變形
var Book = (function(){ // ...其他靜態(tài)方法 return function(newIsbn,newTitle,newAuthor){ var isbn,title,author; this.setIsbn=function(newIsbn){ isbn = newIsbn; }; this.getIsbn=function(){ return isbn; }; this.setTitle=function(newTitle){ title = newTitle; }; this.getTitle=function(){ return title; }; this.setIsbn=function(newAuthor){ author = newAuthor; }; this.getIsbn=function(){ return author; }; }; })();
這種方法是直接返回一個(gè)構造器的執行。且這里的構造器是一個(gè)內嵌函數。
這種方法的優(yōu)點(diǎn)是“在內存中只會(huì )存在一份。因為其他靜態(tài)方法被聲明在構造器之外,所以它們不是特權方法!
判斷一個(gè)方法是否應該被設計為靜態(tài)方法的原則是“這個(gè)方法是否會(huì )訪(fǎng)問(wèn)私有屬性”。如果它不需要,那么將其設計為靜態(tài)方法會(huì )更有效率,因為它只會(huì )被創(chuàng )建一份。
常量
我們可以使用“只有取值器,沒(méi)有賦值器”的方式來(lái)實(shí)現常量。
// 1. var Book = function(){ var constants = ["key1": "1","key2": "2","key3": "3"]; this.getConstant = function(key){ return constants[key]; }; }; Book.getConstant("key1"); // 2. var Book = (function(){ var constants = ["key1": "1","key2": "2","key3": "3"]; var con = function(){}; con.getConstant = function(name){ return constants[name]; }; return con; })(); Book.getConstant("key1");
利弊
1、利處
封裝保護了內部數據的完整性;
封裝使對象的重構更輕松;
弱化模塊間的耦合,提高對象的可重用性;
有助于避免命名空間沖突;
……
2、弊處
私用方法很難測試;
必須與復雜的作用域鏈打交道,使錯誤調度更困難;
容易形成過(guò)度封裝;
JavaScript并不原生支持封裝,所以在JavaScript中實(shí)現封裝存在復雜性的問(wèn)題;
【明白javascript封裝】相關(guān)文章:
對javascript的理解08-08
常用的JavaScript模式09-22
Javascript的this用法簡(jiǎn)述08-15
JavaScript學(xué)習筆記08-24
JavaScript 基礎教學(xué)09-29
JavaScript的課堂講解09-03
JavaScript常用方法匯總10-25
JavaScript數組常用方法介紹09-04