JavaScript 高级程序设计学习笔记一 《第8章 对象 类与面向对象编程》

一 创建有同样接口的多个对象

1.工厂模式

function createPerson(name,age){
 let o = new Object()
    o.age=age
    o.sayName=function(){
      console.log("this.name")
    }
}
let person1 = createPerson("li",29) 
let person2 = createPerson("zhang",30) 

缺点:没有解决对象的表示同意恩问题,不知道对象属于哪个类型

2.构造函数模式

function Person(name,age){
    this.age=age
    this.sayName=function(){
      console.log("this.name")
    }
    //this.sayName = new Function(console.log("this.name"))
}
let person1 = new Person("li",29) 
let person2 = new Person("zhang",30) 

//将方法转移到外部
function Person(name,age){
    this.age=age
}
function sayName=function(){
      console.log("this.name")
}
let person1 = new Person("li",29) 
let person2 = new Person("zhang",30) 

缺点:实例化多个对象时,实际上重复创建了多个相同的方法(sayName()),可以将方法转移到外部,但这相当于定义了全局方法,不能将自定义类型引用到代码很好的聚集起来

3.原型模式

/*无论如何,只要创建一个函数,就会按照特定的规则为这个函数创建一个prototype属性(指向
原型对象)。默认情况下,所有的原型对象自动或得一个名为constructor的属性,指回与之关联的构造函数*/
function Person(){}
Person.prototype.name="li"
Person.prototype.age=29
Person.prototype.sayName=function(){
      console.log("this.name")
}
//上面的代码比较冗余 用对象字面量来重写
//用字面量来重写 会覆盖默认的contructor属性 造成此属性丢失,所以要手动加回来
Person.prototype={
    constructor:Person, 
    name:"li",
    age:29,
    sayName:function(){
          console.log("this.name")
    }
}

缺点 (1)弱化了向构造函数传递初始化参数的能力 (2)所有实例共享了相同的属性,如果属性时引用类型则会相互影响

二 对象之间如何继承

1.原型链继承

function SuperType(){
 this.property=true;
}
SuperType.prototype.getSuperValue=function(){
 return this.property;
}
function SubType(){
 this.subproperty=false
}
//继承SuperType
SubType.prototype= new SuperType();
//新方法
SubType.prototype.getSubValue= function(){
 return this.subproperty;
}
//覆盖原有的方法
SubType.prototype.getSuperValue = function(){
    return false;
}

let instance = new SubType();

缺点 和前面用原型模式构造 对象一样,原型种包含引用值的时候对象之间会相互影响,同时子类在实例化的时候不能给父类传递参数

2 盗用构造函数 与 组合继承

function SuperType(){
 this.colors=["red","blue","greed"]
}

function SubType(){
//继承SuperType
 SuperType.call(this)
}
let instance1= new SubType();
instance1.colors.push("black");
console.log(instance1.colors) // "red,blue,green,black"
let instance2= new SubType();
console.log(instance2.colors) // "red,blue,green"

//盗用构造函数的缺点是必须在构造函数中定义方法,而且 无法重用父类的方法
//组合继承

function SuperType(name){
 this.name=name
 this.colors=["red","blue","greed"]
}
SuperType.prototype.sayName=function(){
      console.log("this.name")
}
function SubType(name,age){
//继承SuperType
 SuperType.call(this); // 第二次调用 SuperType()
this.age=age
}
SubType.prototype= new SuperType(); // 第一次调用 SuperType()
SubType.prototype.sayAge=function(){
 console.log(this.age)
}
let instance1= new SubType("li",29);
instance1.colors.push("black");
console.log(instance1.colors) // "red,blue,green,black"
instance1.sayName() // "li"
let instance2= new SubType("zhang",30);
console.log(instance2.colors) // "red,blue,green"
instance2.sayName() //"zhang"

缺点 两次调用父类的构造函数,有效率问题

3.寄生组合继承

function inheritPrototype(subType,superType){
 let prototype= object(superType.protoType); //复制父类原型
  prototype.constructor=subType;
  subType.prototype=prototype
}
function SuperType(name){
 this.name=name
 this.colors=["red","blue","greed"]
}
SuperType.prototype.sayName=function(){
      console.log("this.name")
}
function SubType(name,age){
//继承SuperType
 SuperType.call(this); // 第二次调用 SuperType()
 this.age=age
}
interitPrototype(SubType,SuperType);
SubType.prototype.sayAge = function(){
 console.log(this.age);
}

基本上避免了前面几种类型的缺点,是应用类型继承的最佳模式

你可能感兴趣的:(javascript)