JS创建对象模式及其对象原型链探究(五):组合使用构造函数模式和原型模式

组合使用构造函数模式和原型模式

构造函数模式用于定义实例属性,原型模式用于定义方法和共享的属性。

创建自定义类型的最常见方式,就是组合使用构造函数模式和原型模式。

1.创建对象

// 组合使用构造函数模式和原型模式
function Person(name, age, job){
     this.name = name;
     this.age = age;
     this.job = job;
     this.friend = ["Jenny", "Court"];
}

Person.prototype = {
     constructor: Person,       //constructor指向Object构造函数,这里要将其constructor恢复为指向Person构造函数。
     Country: "China",
     showName: function(){
          console.log("name = " + this.name);
     }
}

var p1 = new Person("Mike", 20, "student");
var p2 = new Person("Tom", 23, "Teacher");

console.log(p1);
console.log(p2);

结果,每个实例都会有自己的一份实例属性的副本,但同时又共享着对方法的引用,最大限度地节省了内存。

另外,这种混合模式,还支持向构造函数传递参数。

2.不同之处

看上面的代码,你会发现有个不同之处:Person.prototype的constructor属性默认是指向Object构造函数。

从何而知?我们来测试一番。

// 组合使用构造函数模式和原型模式
function Person(name, age, job){
    this.name = name;
    this.age = age;
    this.job = job;
    this.friend = ["Jenny", "Court"];
}
// 未使用原型模式前,Person构造函数的原型对象的constructor指向Person构造函数本身
console.log("Person.prototype.constructor === Person:");
console.log(Person.prototype.constructor === Person);
console.log("-----分割线-----");

Person.prototype = {
    // constructor: Person,     //一般要认为设置,将其constructor恢复为指向Person构造函数,这里为了演示,就没设置。
    Country: "China",
    showName: function(){
        console.log("name = " + this.name);
    }
}

// 使用原型模式,我们将Person.prototype设置为等于一个以对象字面量形式创建的新对象
// 因此,这里constructor属性也就变成了新对象的constructor属性,指向Object构造函数。
// Person构造函数的原型对象的constructor指向Object构造函数
console.log("Person.prototype =");
console.log(Person.prototype);
console.log("-----分割线-----");

console.log("Person.prototype.constructor === Object:");
console.log(Person.prototype.constructor === Object);
console.log("-----分割线-----");

console.log("Person.prototype.__proto__ === Object.prototype:");
console.log(Person.prototype.__proto__ === Object.prototype);

var p1 = new Person("Mike", 20, "student");
var p2 = new Person("Tom", 23, "Teacher");

console.log(p1);
console.log(p2);

未使用原型模式前,Person构造函数的原型对象的constructor指向Person构造函数本身。

使用原型模式,我们将Person.prototype设置为等于一个以对象字面量形式创建的新对象。

因此,这里constructor属性也就变成了新对象的constructor属性,指向Object构造函数。

所以,Person构造函数的原型对象的constructor指向Object构造函数。

3.观察Person构造函数涉及到的原型链

console.log("Person.prototype=");
console.log(Person.prototype);
console.log("-----分割线-----");

console.log("Person.prototype.constructor === Person:");    //true
console.log(Person.prototype.constructor === Person);    //true
console.log("-----分割线-----");

console.log("Person.prototype.__proto__ === Object.prototype:");    //true
console.log(Person.prototype.__proto__ === Object.prototype);    //true
console.log("-----分割线-----");

console.log("Person.__proto__=");
console.log(Person.__proto__);
console.log("Person.__proto__ === Function.prototype:");
console.log(Person.__proto__ === Function.prototype);    //true
console.log("-----分割线-----");

4.观察p1实例对象涉及到的原型链

console.log("p1.prototype=");
console.log(p1.prototype);
console.log("-----分割线-----");

console.log("p1.__proto__=");
console.log(p1.__proto__);
console.log("p1.__proto__ === Person.prototype:");
console.log(p1.__proto__ === Person.prototype);        //true
console.log("-----分割线-----");

console.log("p1.__proto__.constructor === Person:");
console.log(p1.__proto__.constructor === Person);
console.log("-----分割线-----");

console.log("p1.__proto__.__proto__=");
console.log(p1.__proto__.__proto__);
console.log("p1.__proto__.__proto__ === Object.prototype:");
console.log(p1.__proto__.__proto__ === Object.prototype);    //true
console.log("-----分割线-----");

5.观察下p1.showName属性引用的函数

console.log("p1.showName.prototype=");
console.log(p1.showName.prototype);
console.log("-----分割线-----");

console.log("p1.showName.prototype.constructor=");
console.log(p1.showName.prototype.constructor);
console.log("-----分割线-----");

console.log("p1.showName.prototype.__proto__=");
console.log(p1.showName.prototype.__proto__);

console.log("p1.showName.prototype.__proto__ === Object.prototype:");
console.log(p1.showName.prototype.__proto__ === Object.prototype);    //true
console.log("-----分割线-----");

console.log("p1.showName.__proto__=");
console.log(p1.showName.__proto__);
console.log("p1.showName.__proto__ === Function.prototype:");
console.log(p1.showName.__proto__ === Function.prototype);    //true
console.log("-----分割线-----");

原型链图

时间: 2024-10-14 07:26:16

JS创建对象模式及其对象原型链探究(五):组合使用构造函数模式和原型模式的相关文章

JavaScript 原型链学习(三)原型对象存在的问题 与 组合使用构造函数和原型

原型对象也不是没有缺点.首先,它省略了为构造函数传递初始化参数这一环节, 结果所有实例在默认情况下都将取得相同的属性值.虽然这会在某种程度上带来一些不方便, 但还不是原型对象的最大问题.原型对象的最大问题是由其共享的本性所导致的.原型中所有属性是被很多实例共享的,这种共享对于函数非常合适.对于那些包含基本值的属性倒也说得过去,毕竟(如前面的例子所示),通过在实例上添加个同名属性, 可以隐藏原型中的对应属性.然而,对于包含引用类型值的属性来说,问题就比较突出了.来看下面的例子. function

js 原型链(转)

1.1 理解原型链 JavaScript中几乎所有的东西都是对象,我们说数组是对象.DOM节点是对象.函数等也是对象,创建对象的Object也是对象(本身是构造函数),那么有一个重要的问题:对象从哪里来? 这是一句废话,对象当然是通过一定方式创建出来的,根据实际类型不同,对象的创建方式也千差万别.比如函数,我们可以声明函数.使用Function构造函数创建等,比如数组,我们可以直接通过var arr = [] 的方式创建空数组,也可以通过new Array的方式创建,比如普通的对象,我们可以字面

1--面试总结-js深入理解,对象,原型链,构造函数,执行上下文堆栈,执行上下文,变量对象,活动对象,作用域链,闭包,This

参考一手资料:http://dmitrysoshnikov.com/ecmascript/javascript-the-core/中文翻译版本:https://zhuanlan.zhihu.com/p/32042645 Javascript 是一种单线程编程语言,这意味着它只有一个调用栈,call Stack(调用栈 ,,先入后出) 核心:对象,原型链,构造函数,执行上下文堆栈,执行上下文,变量对象,活动对象,作用域链,闭包,This js原型链? 定义 原型对象也是简单的对象并且可以拥有它们自

[js高手之路]原型对象(prototype)与原型链相关属性与方法详解

一,instanceof: instanceof检测左侧的__proto__原型链上,是否存在右侧的prototype原型. 我在之前的两篇文章 [js高手之路]构造函数的基本特性与优缺点 [js高手之路]一步步图解javascript的原型(prototype)对象,原型链 已经分享过了. function CreateObj(uName) {             this.userName = uName;             this.showUserName = function

JS中面向对象的,对象理解、构造函数、原型、原型链

6.1 理解对象 6.1.1 对象属性类型 ECMS属性有两种类型:数据属性和访问器属性 1 数据属性 [[configurable]] 表示能否通过Delete 删除属性从而从新定义属性,能否修改属性的特性,或者能否把属性修改为访问器属性.对象上定义的属性默认值为true [[enumerable]] 表示能否通过for-in循环返回属性.直接在对象上定义的属性,它们的这个特性默认值为true [[writable]] 表示能否修改属性值.像前面例子中那样直接在对象上定义的属性,它们默认值为t

JS如何创建对象和继承对象

JS创建对象的几种方法:工厂模式,构造函数模式,原型模式,混合模式,动态原型模式 1 在工厂模式中,在构造函数内部创建一个新对象,最后返回这个对象.当实例化时,我们不需要用new关键字,就像调用方法一样就可以实例化. 工厂模式的缺点是容易和普通函数混淆,只能通过命名来确认它是一个 构造函数.不推荐使用这种模式. //factory pattern function createPerson(name, age, job){     var o = {};     o.name = name;  

JS原型链的理解

<JavaScript高级程序设计>是一本好书,下面总结一下书中关于原型链的讲解. 本文从对象的创建开始,逐渐引出原型链的背景.作用以及ECMA对原型链的定义1.理解对象两种常用的创建对象的方法之一是new一个Object对象,并为它添加属性和事件var person = new Object();person.name = "wzk";person.age = 24;person.sayName = function(){ alert(this.name);}; 另一种是

关于对象、构造函数、原型、原型链、继承

对象: 在传统的面向过程的程序设计中,会造成函数或变量的冗余.而js中对象的目的是将所有的具有相同属性或行为的代码整合到一起,形成一个集合,这样就会方便管理,例如: var person1={ name:"memphis", age:26, showMessage:function(){ alert("name:"+this.name); } }; person1.showMessage();//输出name:memphis 以上的例子将name.age.showM

深入理解JS继承和原型链

对于那些熟悉基于类的面向对象语言(Java 或者 C++)的开发者来说,JavaScript 的语法是比较怪异的,这是由于 JavaScript 是一门动态语言,而且它没有类的概念( ES6 新增了class 关键字,但只是语法糖,JavaScript 仍旧是基于原型). 涉及到继承这一块,Javascript 只有一种结构,那就是:对象.在 javaScript 中,每个对象都有一个指向它的原型(prototype)对象的内部链接.这个原型对象又有自己的原型,直到某个对象的原型为null 为止