资讯专栏INFORMATION COLUMN

【JavaScript】【对象】创建对象、对象继承的实用方式及理解

scola666 / 2178人阅读

摘要:创建对象对象继承实际上是一回事我们所需要的实例对象通过构造函数获得私有属性通过原型链获得共享的属性。原型式继承和寄生式继承用于创建与已有对象类似的实例对象。

本文约定:不特殊声明的情况下,属性代指属性或方法

创建对象、对象继承实际上是一回事:我们所需要的实例对象通过构造函数获得私有属性通过原型链获得共享的属性。什么是好的方式?私有属性通过构造函数的方式获得(不考虑实例中自定义私有属性)且不需要重写,共享属性通过原型链找到且不需要重复创建

普适的方式 组合使用构造函数模式和原型模式创建对象
function HNU_student(name) {
    this.name = name;
    this.sayName = function() {
        return this.name;
    };
}
HNU_student.prototype = {
    school: "HNU",
    saySchool: function() {
        return this.school;
    }
};
Object.defineProperty(HNU_student, "constructor", {value: HNU_student});

var hiyohoo = new HNU_student("xujian");

通过字面量的方式会重写prototype,且原型的constructor指向了Object,必要的情况下需要重新定义constructor,使用Object.defineProperty()方法可以将constructorenumerable等特性变为初始默认的false

寄生组合式继承
function object(o) {
    function F() {};
    F.prototype = o;
    return new F();
}
function inheritPrototype(child, parent) {
    var prototype = object(parent.prototype);
    prototype.constructor = child;
    child.prototype = prototype;
}

function HNU_student(name) {
    this.name = name;
    this.sayName = function() {
        return this.name;
    };
}
HNU_student.prototype.school = "HNU";
HNU_student.prototype.saySchool = function() {
    return this.school;
};

function Student_2011(name, number) {
    HNU_student.call(this, name);
    this.number = number;
    this.sayNumber = function() {
        return this.number;
    }
}
inheritPrototype(Student_2011, HNU_student);
Student_2011.prototype.graduationTime = 2015;
Student_2011.prototype.sayGraduationTime = function() {
    return this.graduationTime;
};

var hiyohoo = new Student_2011("xujian", 20110803203);

object()的作用:将作为参数传入的对象变成实例的原型,该对象的属性被所有实例共享。

共享属性:inheritPrototype(Student_2011, HNU_student);,子构造函数原型成为超构造函数原型的一个实例,超构造函数原型中的属性共享给子构造函数。
私有属性:HNU_student.call(this, name);,通过子构造函数创建实例时调用超构造函数创建私有属性。

创建对象的其他方式 动态原型模式
function HNU_student(name) {
    this.name = name;
    this.sayName = function() {
        return this.name;
    };

    if (!HNU_student.prototype.school) {
        HNU_student.prototype.school = "HNU";
        HNU_student.prototype.saySchool = function() {
            return this.school;
        };
    }
}

var hiyohoo = new HNU_student("xujian");

将定义在原型中的共享属性放入构造函数中,使用判断语句,在第一次调用构造函数创建实例时,初始化原型共享属性。

寄生构造函数模式
function SpecialArray() {
    var values = new Array();
    values.push.apply(values, arguments);
    values.toPipedString = function() {
        return this.join("|");
    };

    return values;
}

var colors = new SpecialArray("red", "black", "white");

用于为原生构造函数添加特殊的属性。

对象继承的其他方式 组合继承
function HNU_student(name) {
    this.name = name;
    this.sayName = function() {
        return this.name;
    };
}
HNU_student.prototype.school = "HNU";
HNU_student.prototype.saySchool = function() {
    return this.school;
};

function Student_2011(name, number) {
    HNU_student.call(this, name);
    this.number = number;
    this.sayNumber = function() {
        return this.number;
    };
}
Student_2011.prototype = new HNU_student();    //重写原型一定要放在所有原型属性定义之前
Student_2011.prototype.constructor = Student_2011;
Student_2011.prototype.graduationTime = 2015;
Student_2011.prototype.sayGraduationTime = function() {
    return this.graduationTime;
}

var hiyohoo = new Student_2011("xujian", 20110803203);

共享属性:Student_2011.prototype = new HNU_student();,子构造函数的原型就指向了超构造函数的原型,实例通过原型链找到所有共享的属性。
私有属性:HNU_student.call(this, name);,通过子构造函数创建实例时调用超构造函数创建私有属性。

缺陷:超构造函数被调用了两遍。Student_2011.prototype = new HNU_student();的同时,在子构造函数原型中创建了超构造函数定义的私有属性,这些原型中的私有属性被实例中的同名属性覆盖屏蔽。

原型式继承、寄生式继承
function object(o) {
    function F() {}
    F.prototype = o;
    return new F();
}

var student1 = {
    school: "HNU",
    saySchool: function() {
        return this.school;
    }
};

var student2 = object(student1);

Object.creat()ECMAScript 5新增的方法,接受两个参数:一是作为原型的原对象,二是重写或新增属性的对象,作用与自定义的object()相同。

var student1 = {
    name: "xujian",
    school: "HNU"
};
var student2 = Object.create(student1, {
    name: {
        value: "huangjing"
    }
});

寄生式继承在原型式继承的基础上添加了额外的属性用来增强对象。

function object(o) {
    function F() {}
    F.prototype = o;
    return new F();
}
function creatAnother(original) {
    var clone = object(original);
    clone.sayHi = function() {
        alert("Hi!");
    };
    return clone;
}

var student1 = {
    school: "HNU",
    saySchool: function() {
        return this.school;
    }
};

var student2 = creatAnother(student1);

原型式继承和寄生式继承用于创建与已有对象类似的实例对象。

转载请注明出处:https://segmentfault.com/a/1190000004559437

文章不定期更新完善,如果能对你有一点点启发,我将不胜荣幸。

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/78829.html

相关文章

  • 学Java编程需要注意地方

    摘要:学编程真的不是一件容易的事不管你多喜欢或是多会编程,在学习和解决问题上总会碰到障碍。熟练掌握核心内容,特别是和多线程初步具备面向对象设计和编程的能力掌握基本的优化策略。   学Java编程真的不是一件容易的事,不管你多喜欢或是多会Java编程,在学习和解决问题上总会碰到障碍。工作的时间越久就越能明白这个道理。不过这倒是一个让人进步的机会,因为你要一直不断的学习才能很好的解决你面前的难题...

    leanxi 评论0 收藏0
  • 全面理解JavaScriptthis

    摘要:与方法这两个方法的第一个参数是一个对象,是给准备的,接着在调用函数时将其绑定到。作为事件处理函数当函数被用作事件处理函数时,它的会指向触发事件的元素。总结在任何函数体外部,无论是否在严格模式,都指代全局对象。 this在JavaScript中似乎可以视而不见,但不去正视它学到的就只是残缺的JS。抛开这些形而上的意义不说,从实用性及代码简洁展示this的不可或缺的地位,举例子: func...

    Pines_Cheng 评论0 收藏0
  • JavaScript原型、原型链、继承理解

    摘要:构造函数实例和原型的概念和关系每个函数都属于对象,都会有一个属性叫。这也是继承的依据。这样一来,原型找不到构造函数,这是非常蛋疼的事情,违反了原型链的定义啊。所以现在子对象原型和父对象原型是就建立关系了。 构造函数、实例和原型的概念和关系 每个函数都属于对象,都会有一个属性叫prototype。这个属性指向一个对象,我们把他叫做当前函数的原型对象。原型对象下面有个属性叫construc...

    Aceyclee 评论0 收藏0
  • 【译】每个JavaScript 开发者应该了解10个面试题

    摘要:避免脆弱的基类问题。红牌警告没有提到上述任何问题。单向数据流意味着模型是单一的事实来源。单向数据流是确定性的,而双向绑定可能导致更难以遵循和理解的副作用。原文地址 1. 你能说出两种对 JavaScript 应用开发者而言的编程范式吗? 希望听到: 2. 什么是函数编程? 希望听到: 3. 类继承和原型继承的不同? 希望听到 4. 函数式编程和面向对象编程的优缺点? ...

    mykurisu 评论0 收藏0
  • 复习Javascript专题(三):面向对象对象创建继承,原型原型链)

    摘要:在创建子类实例时,不能向超类型的构造函数中传递参数。构造函数继承子类传进的值是基本思想是在子类构造函数的内部调用超类或父类型构造函数。继承保证构造函数指针指向如果想同时继承多个,还可使用添加属性的方式类继承, OOP:Object Oriented Programming 面向对象编程。 题外话:面向对象的范围实在太大,先把这些大的东西理解理解。 1.什么是对象? 根据高程和权威指南上...

    testHs 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<