成人无码视频,亚洲精品久久久久av无码,午夜精品久久久久久毛片,亚洲 中文字幕 日韩 无码

資訊專欄INFORMATION COLUMN

Javascript面向?qū)ο蟮某绦蛟O(shè)計

lbool / 1127人閱讀

摘要:原型模式就是通過調(diào)用構(gòu)造函數(shù)而創(chuàng)建的那個對象實例的原型對象。我們來看下面一個例子來理解這句話與構(gòu)造函數(shù)模式不同的是,新對象的這些屬性和方法是由所有實例共享的。原型對象的問題首先,它省略了為構(gòu)造函數(shù)傳遞初始化參數(shù)這一環(huán)節(jié)。

最近因為在給一個小同學(xué)做學(xué)習(xí)計劃,所以也記錄一些知識點,便于后面的同學(xué)的學(xué)習(xí)交流。這篇文章是關(guān)于Javascript的面向?qū)ο蟮某绦蛟O(shè)計,主要從三個方面來介紹,1. 理解對象屬性; 2. 理解并創(chuàng)建對象; 3. 理解繼承

一、理解對象屬性
首先我們來理解Javascript對象是什么?在Javascript中,萬物皆對象。其中創(chuàng)建自定義對象的最簡單的方式就是創(chuàng)建一個Object的實例,如下:

var person = new Object();
person.age = 29;

// 對象字面量的形式:
var person = {
     age: 29
};

ECMAScript中有兩種屬性:數(shù)據(jù)屬性和訪問器屬性。

數(shù)據(jù)屬性:
其中數(shù)據(jù)屬性有四個描述其行為的特性:
Configurable: 表示能都通過delete刪除屬性從而重新定義屬性。
Enumerable: 表示能否通過for in 循環(huán)返回屬性。
Writable: 表示能否修改屬性的值。
Value: 包含這個屬性的數(shù)據(jù)值。
要修改屬性默認的配置,必須使用Object.defineProperty(), 這個方法接收三個參數(shù):屬性所在的對象,屬性的名字和一個描述性對象。
比如:

var person = {};
Object.defineProperty(person, ’name’, {
      writable: false,
      value: ’Nicholas"
});

alert(person.name); //Nicholas
person.name = ‘Greg’;
alert(person.name); //Nicholas

訪問器屬性:
訪問器屬性包含一對setter和getter函數(shù)。包含如下4個特性:
Configurable:能否被delete刪除屬性重新定義。默認值:true
Enumerable:能否被for-in枚舉。默認值:true
Get:讀取屬性值。默認值:undefined
Set:寫入屬性值。默認值:undefined

var dog = {
    _age: 2,
    weight: 10
}

Object.defineProperty(dog, "age", {
    get: function () {
        return this._age
    },
    set: function (newVal) {
        this._age = newVal
        this.weight += 1
    }
})

知道了對象的屬性,那么我們創(chuàng)建對象的方式是什么呢?

二、創(chuàng)建對象的方式

創(chuàng)建對象的方式通常有下面幾種方式:
1、工廠模式
我們舉個例子:

function createPerson(name, age, job) {
    var o = new Object();
    o.name = name;
    o.age = age;
    o.job = job;
    o.sayName = function () {
        alert(this.name);
   };
   Return o;
}

var person = createPerson(‘Greg’, 27, ‘Doctor’);

工廠模式雖然解決了創(chuàng)建多個相似對象的問題,但卻沒有解決問題識別的問題(即怎樣知道一個對象的類型)

2、構(gòu)造函數(shù)模式
我們舉個例子:
**function Person(name, age, job) {

this.name = name;
this.age = age;
this.job = job;
this.sayName =  function () {
    alert(this.name);

};
}
var person = new Person(‘Greg’, 27, ‘Doctor’);**

構(gòu)造函數(shù)始終都應(yīng)該以一個大寫字母開頭,而非構(gòu)造函數(shù)則應(yīng)該以一個小寫字母開頭。這里要提的一個屬性是Constructor, 每個new 出來的實例都有一個Constructor(構(gòu)造函數(shù))屬性,該屬性指向構(gòu)造函數(shù)。
對象的Constructor屬性最初是用來標識對象類型的。但是,提到檢測對象類型,還是instanceof操作符更好可靠一些。
alert(person1 instanceof Object); //true

構(gòu)造函數(shù)的問題問題就是,每個方法都要在每個實例上重新創(chuàng)建一遍,當然,可以把函數(shù)定義轉(zhuǎn)移到構(gòu)造函數(shù)外部來解決這個問題,如下實例:

**function Person(name, age, job) {

this.name = name;
this.age = age;
this.job = job;
this.sayName = sayName;

}
function sayName() {

alert(this.name);

}
var person = new Person(‘Greg’, 27, ‘Doctor’);**

那么這里的問題就是:在全局作用域中定義的函數(shù)實際上被只能被某個對象調(diào)用,這讓全局作用域有點名不副實。而更讓人無法接受的是:如果獨享需要定義很多方法,那么就要定義很多個全局函數(shù)。下面我們來看看原型模式是不能解決這個問題。

3、原型模式
prototype就是通過調(diào)用構(gòu)造函數(shù)而創(chuàng)建的那個對象實例的原型對象。我們來看下面一個例子來理解這句話:

function Person() {
}
Person.prototype.name = “Nicholas”;
Person.prototype.age = 29;
Person.prototype.job = “Software Engineer”;
Person.prototype.sayName = function() {
    alert(this.name);
};
var person1 = new Person();
person1.sayName(); // “Nicholas”

與構(gòu)造函數(shù)模式不同的是,新對象的這些屬性和方法是由所有實例共享的。
針對這個特性,我們要注意的幾個點就是。
第一,如果實例的屬性與方法與原型的屬性和方法同名,那誰的優(yōu)先級高呢?
當然是創(chuàng)建出來的實例的屬性和方法的優(yōu)先級高。
第二,實例的屬性與方法的修改會影響原型同名的屬性和方法嗎?
不會
這里提一下hasOwnProperty(), 使用hasOwnProperty()方法可以檢測一個屬性是存在于實例中,還是存在原型中。
第三,如何判斷某個屬性是存在原型中?

function hasPrototypeProperty(object, name) {
    return !Object.hasOwnProperty(name) && (name in object);
}

hasOwnProperty()只在屬性存在于實例中才返回true, 因此只要in操作符返回true而hasOwnProperty()返回false, 就可以確定屬性是原型中的屬性、這里說下Constructor, 每創(chuàng)建一個函數(shù),就會同時創(chuàng)建它的Prototype對象,這個對象也會自動獲得Constructor屬性。
我們來看一個例子:

function Person() {

}

Person.prototype = {
    constructor: Person,
    name: “Nichloas”,
    age: 29,
    job: “Software Engineer”,
    sayName: function () {
         alert(this.name);
    }  
};

以上代碼特意包含了一個Constructor屬性,并將它的值設(shè)置為Person, 從而確保了通過該屬性能夠訪問到適當?shù)闹?。但是,以這種方式重設(shè)Constructor屬性會導(dǎo)致它的Enumerable特性被設(shè)置為true, 默認情況下原生的Constructor屬性是不可枚舉的。

Object.defineProperty(Person.prototype, “Constructor”, {
    enumerable: false,
    value: Person
});

原型對象的問題:
首先,它省略了為構(gòu)造函數(shù)傳遞初始化參數(shù)這一環(huán)節(jié)。
然后,原型中所有屬性是被很多實例共享的,對于包含引用類型值的屬性來說,問題比較突出。所以,使用最多的方式使用構(gòu)造函數(shù)和原型模式

4、組合使用構(gòu)造函數(shù)模式和原型模式

function Person(name, age, job) {
     this.name = name;
     this.age = age;
     this.job = job;
     this.friends = [“Shelby”, “Court”];
} 

Person.prototype = {
     constructor: Person,
     sayName: function () {
          alert(this.name);
     }
};

var person = new Person(‘Greg’, 27, ‘Doctor’);

知道了創(chuàng)建對象的方式,那么在Javascript中我們?nèi)绾蝸砝^承對象呢?

三、繼承
1、原型鏈
構(gòu)造函數(shù),原型和實例的關(guān)系:每一個構(gòu)造函數(shù)都也有一個原型對象,原型對象都包含一個指向構(gòu)造函數(shù)的指針,而實例都包含一個指向原型對象的內(nèi)部指針。那么,假如我們讓原型對象等于另一個類型的實例,結(jié)果會怎么樣呢?顯然,此時的原型對象將包含一個指向另一個原型的指針,相應(yīng)的,另一個原型中也包含也包含著一個指向一個構(gòu)造函數(shù)的指針,假如另一個原型又是另一個類型的實例,那么上訴關(guān)系依然成立,我們來看下面一個例子.

function SuperType() {
    this.property = true;
}

SuerType.prototype.getSuperValue = function () {
     return this.property;
};

function SubType() {
    This.subProperty = false;
}

// 繼承了SuperType
SubType.prototype = new SuperType();

SubType.prototype.getSubValue = function () {
    return this.subProperty();
};

var instance = new SubType();
alert(instance.getSuperValue); // true
這個例子中,SubType繼承了SuperType,而SuperType繼承了Object。如果要確定原型和實例的關(guān)系,可以用instanceOf操作符與isPropertyOF()方法。
如 instance instanceOf object; // true
Object.prototype isPropertype(instance); // true

但是通過原型鏈實現(xiàn)繼承時,不能使用對象字面量創(chuàng)建原型方法。

function SuperType() {
    this.property = true;
}

SuerType.prototype.getSuperValue = function () {
     return this.property;
};

function SubType() {
    This.subProperty = false;
}

// 繼承了SuperType
SubType.prototype = new SuperType();

// 使用字面量添加新方法,會導(dǎo)致上一行代碼無效
SubType.prototype = {
      getSubValue: function () {
            return this.subproperty;
       },
      someOtherMethod: function () {
            return false;
      }
};

var instance = new SubType();
alert(instance.getSuperValue); // error

接下來,我們來說下原型鏈繼承的問題:
第一,最主要的問題來自包含引用類型值的原型。包含引用類型值的原型屬性會被所有實例共享,而這也正是為什么要在構(gòu)造函數(shù)中,而不是在原型對象中定義屬性的原因。
第二,在創(chuàng)建子類型的實例時候,不能向超類型的構(gòu)造函數(shù)中傳遞參數(shù)。

2、借用構(gòu)造函數(shù)

function SuperType(name) {
       this.name = name;
}

function SubType() {
    // 繼承了SuperType, 同時還傳遞了參數(shù)
    SuperType.call(this, “Nicholas”);
    this.age = 29;
}

var instance = new SubType();
alert(instance.name);  // “Nicholas” 

借用構(gòu)造函數(shù)的問題,方法都在構(gòu)造函數(shù)中定義,因此函數(shù)復(fù)用就無從談起了。而且,在超類型的原型中定義的方法,對于子類型而言也是不可見的,結(jié)果所有類型都只能使用構(gòu)造函數(shù)模式,考慮到這些問題,借用構(gòu)造函數(shù)的技術(shù)也是很少多帶帶使用的。

3、組合繼承
組合繼承,指的是將原型鏈和借用構(gòu)造函數(shù)的技術(shù)組合到一塊,從而發(fā)揮二者之長的一種繼承模式。其背后的思路是使用原型鏈實現(xiàn)對原型屬性和放大的績效,而通過借用構(gòu)造函數(shù)來實現(xiàn)對實例屬性的繼承。

function SuperType(name) {
    this.name = name;
    this.colors = [“red”, “blue”, “green"]
} 

SuperType.prototype.sayName = function () {
     alert[this.name];
};

function SubType(name, age) {
     // 繼承屬性
     SuperType.call(this, name);
     this.age = age;
} 

// 繼承方法
SubType.prototype = new SuperType();

SubType.prototype.sayAge = function () {
    alert(this.age);
};


var instance = new SubType(“Nicholas”, 29);
instance.colors.push(“black”);
alert(instance.colors); // red, blue, green, black

組合繼承避免了原型鏈和借用構(gòu)造函數(shù)的缺陷,融合了它們的優(yōu)點,成為Javascript中最常見的繼承模式。

4、原型式繼承

function object(o) {
    function F() {}
    F.prototype = o;
    return new F();
]

var person = {
    Name: “Nicholas”,
    friends: [“Shelby”, “Court”, “Van"]
};

var anotherPerson = object(person);
anotherPerson.name = “Greg”;

ECMAScipt5通過新增Object.create()方法規(guī)范化了原型式繼承。
接著上面的例子:

 var anotherPerson = Object.create(person);
anotherPerson.name = “Greg”;

5、寄生式繼承
寄生式繼承是與原型式繼承緊密相關(guān)的一種思路,它與工廠模式類似,即創(chuàng)建一個僅用于封裝過程的函數(shù),該函數(shù)在內(nèi)部以某種方法來增強對象,最后再像真地是它做了所有工作一樣返回對象。
例如:

function createAnother(original) {
     var clone = object(original);
     clone.sayHi = function () {
           alert(‘hi’);
     };
     return clone; 
}

var person = {
    name: ’Nochplas’,
    friends: [’Shelby’, ‘Court’, ‘Van"]
};

var anotherPerson = createAnother(person);

6、組合寄生式繼承

組合繼承是Javascript最常用的繼承模式,不過,組合繼承最大的問題就是無論什么情況下,都會調(diào)用兩次超類型構(gòu)建函數(shù):一次是在創(chuàng)建子類型原型的時候,另一次是在子類型構(gòu)造函數(shù)內(nèi)部。

function SuperType(name) {
    this.name = name;
    this.colors = [“red”, “blue”, “green"]
} 

SuperType.prototype.sayName = function () {
     alert[this.name];
};

function SubType(name, age) {. 
     // 繼承屬性
     SuperType.call(this, name);   // 第二次調(diào)用 SuperType()
     this.age = age;
} 

// 繼承方法
SubType.prototype = new SuperType(); // 第一次調(diào)用 SuperType()

SubType.prototype.sayAge = function () {
    alert(this.age);
};

所謂寄生組合式繼承,即通過借用構(gòu)造函數(shù)來繼承屬性,通過原型鏈的混成形式來繼承方法。不必為了制定子類型的原型而調(diào)用超類型的構(gòu)造函數(shù),我們所需要的無非就是超類型原型的一個副本而已。本質(zhì)上,就是使用寄生式繼承來繼承超類型的原型,然后再將結(jié)果制定給子類型的原型。如下:

function inheritPrototype (subType, superType) {
   var prototype = object (superType.prototype); // 前面的原型式繼承object方法
   prototype.constructor = subtype;
   subtype.prototype = prototype;
}


function SuperType(name) {
    this.name = name;
    this.colors = [“red”, “blue”, “green"]
} 

SuperType.prototype.sayName = function () {
     alert[this.name];
};

function SubType(name, age) {. 
     // 繼承屬性
     SuperType.call(this, name);  
     this.age = age;
} 

inheritPrototype(SubType, SuperType);

SubType.prototype.sayAge = function () {
    alert(this.age);
};


文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://m.hztianpu.com/yun/109243.html

相關(guān)文章

  • SegmentFault 技術(shù)周刊 Vol.32 - 七夕將至,你對象”還好嗎?

    摘要:很多情況下,通常一個人類,即創(chuàng)建了一個具體的對象。對象就是數(shù)據(jù),對象本身不包含方法。類是相似對象的描述,稱為類的定義,是該類對象的藍圖或原型。在中,對象通過對類的實體化形成的對象。一類的對象抽取出來。注意中,對象一定是通過類的實例化來的。 showImg(https://segmentfault.com/img/bVTJ3H?w=900&h=385); 馬上就要到七夕了,離年底老媽老爸...

    李昌杰 評論0 收藏0
  • SegmentFault 技術(shù)周刊 Vol.32 - 七夕將至,你對象”還好嗎?

    摘要:很多情況下,通常一個人類,即創(chuàng)建了一個具體的對象。對象就是數(shù)據(jù),對象本身不包含方法。類是相似對象的描述,稱為類的定義,是該類對象的藍圖或原型。在中,對象通過對類的實體化形成的對象。一類的對象抽取出來。注意中,對象一定是通過類的實例化來的。 showImg(https://segmentfault.com/img/bVTJ3H?w=900&h=385); 馬上就要到七夕了,離年底老媽老爸...

    Lyux 評論0 收藏0
  • SegmentFault 技術(shù)周刊 Vol.32 - 七夕將至,你對象”還好嗎?

    摘要:很多情況下,通常一個人類,即創(chuàng)建了一個具體的對象。對象就是數(shù)據(jù),對象本身不包含方法。類是相似對象的描述,稱為類的定義,是該類對象的藍圖或原型。在中,對象通過對類的實體化形成的對象。一類的對象抽取出來。注意中,對象一定是通過類的實例化來的。 showImg(https://segmentfault.com/img/bVTJ3H?w=900&h=385); 馬上就要到七夕了,離年底老媽老爸...

    AaronYuan 評論0 收藏0
  • 面向對象 JavaScript

    摘要:是完全的面向?qū)ο笳Z言,它們通過類的形式組織函數(shù)和變量,使之不能脫離對象存在。而在基于原型的面向?qū)ο蠓绞街校瑢ο髣t是依靠構(gòu)造器利用原型構(gòu)造出來的。 JavaScript 函數(shù)式腳本語言特性以及其看似隨意的編寫風(fēng)格,導(dǎo)致長期以來人們對這一門語言的誤解,即認為 JavaScript 不是一門面向?qū)ο蟮恼Z言,或者只是部分具備一些面向?qū)ο蟮奶卣?。本文將回歸面向?qū)ο蟊疽猓瑥膶φZ言感悟的角度闡述為什...

    novo 評論0 收藏0
  • javascript 面向對象版塊之理解對象

    摘要:用代碼可以這樣描述安全到達國外面向過程既然說了面向?qū)ο?,那么與之對應(yīng)的就是面向過程。小結(jié)在這篇文章中,介紹了什么是面向?qū)ο蠛兔嫦蜻^程,以及中對象的含義。 這是 javascript 面向?qū)ο蟀鎵K的第一篇文章,主要講解對面向?qū)ο笏枷氲囊粋€理解。先說說什么是對象,其實這個還真的不好說。我們可以把自己當成一個對象,或者過年的時候相親,找對象,那么你未來的老婆也是一個對象。我們就要一些屬性,比...

    lovXin 評論0 收藏0
  • JS對象(1)重新認識面向對象

    摘要:對象重新認識面向?qū)ο竺嫦驅(qū)ο髲脑O(shè)計模式上看,對象是計算機抽象現(xiàn)實世界的一種方式。除了字面式聲明方式之外,允許通過構(gòu)造器創(chuàng)建對象。每個構(gòu)造器實際上是一個函數(shù)對象該函數(shù)對象含有一個屬性用于實現(xiàn)基于原型的繼承和共享屬性。 title: JS對象(1)重新認識面向?qū)ο? date: 2016-10-05 tags: JavaScript 0x00 面向?qū)ο?從設(shè)計模式上看,對象是...

    superw 評論0 收藏0

發(fā)表評論

0條評論

閱讀需要支付1元查看
<