在JavaScript中,建立一個具有自訂屬性和方法的物件有很多種模式,下面一一進行介紹。
1、直接建立模式。這是最簡單也是最直接的模式,首先建立一個引用類型的對象,然後為其新增自訂屬性和方法。範例程式碼如下:
1 var person = new Object();
2 person.name = "Sam";
3 person.age = 16;
4 person.speak = function(){
5 alert(this. name + "is " + this.age + "years old");
6 }
7 person.speak();
可以看到,上面創建了一個Object類型的對象,然後為其添加了name和age屬性以及一個speak方法。直接建立模式雖然簡單,但其缺點是顯而易見的:當我們需要建立許多相同的物件時,每次都要重複編寫程式碼。為了解決這個問題,我們可以將創建物件的過程進行封裝,於是便有了下面的工廠模式。
2、工廠模式。工廠模式是程式設計中常用的設計模式,它主要是將創建物件的流程進行了封裝,範例程式碼如下:
1 function createPerson(name, age){
2 var person = new Object() ;
3 person.name = name;
4 person.age = age;
5 person.speak = function(){
6 alert(this.name + "is " + this.age + "yyears old");🎟 7 }
8 return person;
9 }
10 var person1 = createPerson("Sam", 16);
11 var person2 = createPerson("Jack", 18);
3、建構函數模式。構造函數和普通函數沒有任何區別。任何普通函數都可以當作建構函數,只要使用new運算元即可;任何建構函數也都可以當作普通函數來呼叫。只不過在JavaScript中,有一個約定,就是用來作為建構函數的函數名稱需要一個首字母大寫。範例程式碼如下:
1 function Person(name, age){
2 this.name = name;
3 this.age = age;
4 this.speak = function(){
5 "is " + this.age + "years old");
6 }
7 }
8 var person1 = new Person("Sam", 16);
9 var person2 = new Person("Jack", 18);
解決這個缺點的一個比較簡單的方法就是將函數的聲明放到構造函數的外面,即:
1 function name, age){
2 this.name = name;
3 this.age = age;
4 this.speak = speak;
5 }
6 + this.age + "years old");
8 }
9 var person1 = new Person("Sam", 16);
10 var person2 = new Person("Jack", 18);
11 alert(person1. speak == person2.speak); // true
問題解決了,但這種方法又帶來了新的問題。首先,函數speak是在全域作用域中宣告的,但它卻只能被用於Person建構函數,放在全域作用域中有被誤用的風險;其次,如果一個自訂型別有很多的方法,則需要宣告很多的全域函數,這不僅會導致全域作用域的污染,也不利於程式碼的封裝。那麼,有沒有辦法能讓自訂類型的方法成為與類別綁定的,又不污染全域作用域呢?答案是使用原型模式。
4、原型模式。在我們宣告一個新的函數後,該函數(在JavaScript中,函數也是物件)就會擁有一個prototype的屬性。 prototype是一個對象,表示會被該函數所建立的所有物件擁有的公共屬性和方法。範例程式碼如下:
1 function Person(){}
2 Person.prototype.name="Sam";
3 Person.prototype.age=16;
4 Person.prototype.speak = function(5){5 alert(this.name + "is " + this.age + "years old");
6 }
7 var person1 = new Person();
8 person1.speak();
9 var person2 = new Person() ;
10 alert(person1.speak == person2.speak); // true
5、組合模式。範例程式碼如下:
1 function Person(name, age){
2 this.name = name;
3 this.age = age;
4 }
5 Person.prototype。 (this.name + "is " + this.age + "years old");
7 }
8 var person1 = new Person();
9 person1.speak();
10 var person2 = new Person();
11 alert(person1.speak == person2.speak); // true
不難發現,組合模式實現了我們所有的需求,這也是目前應用得比較廣泛的一種模式。有物件導向程式設計經驗的開發人員可能會覺得將prototype的聲明放在建構函式外面有點彆扭,那麼能否將其放到建構函式裡去呢?答案是肯定的,使用動態組合模式即可。
2 this.name = name;
3 this.age = age;
4 if (Person.prototype.speak == "undefined"{
Person.prototype.speak = function(){
6 alert(this.name + "is " + this.age + "years old");
7 }
8 }
9 }