第1章面向對象的Javascript

面向對象的Javascript

1.1動態類型與鴨子類型

var duck = {
    duckSinging : function(){
        console.log('嘎嘎嘎');
    }
}
var chicken = {
    duckSinging : function(){
        console.log('咯咯咯');
    }
}
var choir = []    //合唱團
var joinChoir = function(animal){
    if(animal && typeof animal.duckSinging =='function'){
        choir.push(animal);
        console.log('恭喜加入合唱團');
        console.log('合唱團已有成員:' + choir.length);
    }
}
joinChoir(duck);
joinChoir(chicken);

1.2多態

1.2.1一段“多態”的javascript代碼

var makeSoud = function(animal){
    if(animal instanceof Duck){
        console.log('嘎嘎嘎');
    }else if (animal instanceof Chicken){
        console.log('咯咯咯');
    }
}
var Duck = function(){}
var Chicken = function(){}

makeSoud(new Duck());
makeSoud(new Chicken());

加狗又要改makeSoud代碼

1.2.2對象的多態性

var makeSoud = function(animal){
    animal.sound();
};
var Duck = function(){};
Duck.prototype.sound = function(){
    console.log('嘎嘎噶');
};
var Chicken = function(){};
Chicken.prototype.sound = function(){
    console.log('咯咯咯');
}
makeSoud(new Duck());
makeSoud(new Chicken());
var Dog = function(){};
Dog.prototype.sound = function(){
    console.log('汪汪汪');
}
makeSoud(new Dog());

1.2.3 類型檢查與多態

java

1.2.4 使用繼承得到多態效果

1.2.5 Javascript的多態

不需要java類似的向上轉型

1.2.6 多態在面向對象程序設計中的作用

var googleMap = {
    show:function(){
        console.log('開始渲染谷歌地圖');
    }
};


var baiduMap = {
    show:function(){
        console.log('開始渲染百度地圖');
    }
}
var renderMap = function(type){
    if(type=='google'){
        googleMap.show();
    }else if(type=='baidu'){
        baiduMap.show();
    }
};

renderMap('google');
renderMap('baidu');

把相同的部分抽象出來

var googleMap = {
    show:function(){
        console.log('開始渲染谷歌地圖');
    }
};


var baiduMap = {
    show:function(){
        console.log('開始渲染百度地圖');
    }
}
var renderMap = function(map){
    if(map.show instanceof Function){
        map.show();
    }
}

renderMap(googleMap);
renderMap(baiduMap);

實際開發中借助適配器模式

1.2.7 設計模式與多態

1.3 封裝

1.3.1 封裝數據

var myObject = (function(){
    var _name = 'sven';
    return {
        getName:function(){
            return _name;
        }
    }
})();

console.log(myObject.getName()); //sven
console.log(myObject._name); //undefined

1.3.2 封裝實現

封裝不僅僅是隱藏數據,還包括隱藏細節 設計細節 對象的類型 等

1.3.3 封裝類型

封裝類型是通過抽象類和接口來進行的 Javascript不區分類型

1.3.4 封裝變化

“可復用面向對象軟件基礎”

1.4 原型模式與基于原型繼承的Javascript對象系統

原型編程的思想中,類并不是必需的,對象未必需要從類中創建而來,一個對象是通過克隆另外一個對象所得到的

1.4.1 使用克隆的原型模式

var Plane = function(){
    this.blood == 100;
    this.attackLevel = 1;
    this.defenseLevel = 1;
};
var plane = new Plane();
plane.blood = 500;
plane.attackLevel = 10;


var clonePlane = Object.create(plane);
console.log(clonePlane.attackLevel);  //10
/*不支持Object.create方法的瀏覽器中*/
Object.create = Object.create||function(obj) {
        var F = function () {
        };
        F.prototype = obj;
        return new F();
    }

1.4.2 克隆是創建對象的手段

1.4.3 體驗lo語言

1.4.4 原型編程范型的一些規則

  • 所有的數據都是對象
  • 要得到一個對象,不是通過實例化類,而是找到一個對象作為原型并克隆它。

1.4.5 Javascript中的原型繼承

  • 所有的數據都是對象
  • 要得到一個對象,不是通過實例化類,而是找到一個對象作為原型并克隆它。
  • 對象會記住它的原型。
  • 如果對象無法響應某個請求,它會把這個請求委托給它的對象系統的。

1.所以的數據都是對象

undefined number boolean string function object

Javascript設計本意是除了undefined之外,一切應是對象,為了實現這一目標number boolean string這幾種基本類型也可以通過“包裝類”的方式變成對象類型數據來處理。

Javascript中的根對象是Object.prototype對象->空對象

var obj1 = new Object();
var obj2 = new Object();
console.log(Object.getPrototyprOf(obj1) === Object.prototype);//ture
console.log(Object.getPrototyprOf(obj2) === Object.prototype);//ture

2.要得到一個對象,不是通過實例化類,而是找到一個對象的原型并克隆它

function Person(name){
this.name = name;
}
Person.prototype.getName = function(){
    return this.name;
}
var a = new Person('sven');
console.log(a.name);  //sven
console.log(a.getName()); //sven
console.log(Object.getPrototypeOf(a) === Person.prototype);  //ture

Person并不是類,而是函數構造器,Javascript的函數既可以作為普通函數被調用,也可以做為構造器被調用。當使用new運算符來調用函數時,此時的函數就是一個構造器。用new運算符來創建對象的過程,實際上也只是先克隆Object.prototype對象,再進行一些額外的操作。

function Person(name){
    this.name = name;
}
Person.prototype.getName = function(){
    return this.name;
}
var objectFactory = function(){
    var obj = new Object(),  //從Object.prototype上克隆一個空的對象
        Constructor = [].shift.call(arguments); //取得外部傳人的構造期,此例是Person
    obj._proto_ = Constructor.prototype;
    var ret = Constructor.apply(obj,arguments); //借外部傳人的構造器給obj設置屬性。
    return typeof ret === 'object' ? ret : obj;
};

var a = objectFactory(Person,'sven');

console.log(a.name);  //sven
console.log(a.getName()); //sven
console.log(Object.getPrototypeOf(a) === Person.prototype);  //ture

Javascript給對象提供一個名為proto的隱藏屬性,默認指向它的構造器的原型對象。即{Constructor}.prototype.在一些瀏覽器中,proto被公開出來。
proto就是對象跟“對象構造器的原型”聯系起來的紐帶

var a = new Object();
console.log(a.__proto__ === Object.prototype);

4如果無法響應某個請求,它會把這個請求委托給它的構造器的原型

var obj = {name:'sven'};

var A = function(){};
A.prototype = obj;

var a = new A();
console.log(a.name); //sven

繼承

var A = function(){};
A.prototype = {name:'sven'};

var B = function(){};
B.prototype = new A();

var b = new B();
console.log(b.name);  //輸出sven

1.4.6 原型繼承未來

Object.create(null)可以創建出沒有原型的對象

es6

class Animal{
    constructor(name){
        this.name = name;
    }
    getName(){
        return this.name;
    }
}

class Dog extends Animal{
    constructor(name){
        super(name);
    }
    speak(){
        return'woof';
    }
}
var dog = new Dog('scamp');
console.log(dog.getName()+'says' dog.speak());
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容