javascript原型链需要注意的地方的总结

本篇文章给大家带来的内容是关于javascript原型链需要注意的地方的总结,有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。

前言:最近在细读Javascript高级程序设计,对于我而言,中文版,书中很多地方翻译的差强人意,所以用自己所理解的,尝试解读下。如有纰漏或错误,会非常感谢您的指出。文中绝大部分内容引用自《JavaScript高级程序设计第三版

1. 别忘记默认的原型

事实上,前面例子中展示的原型链还少一环。

我们都知道, 所有引用类型默认都继承了Object,而这个继承也是通过原型链实现的。

立即学习“Java免费学习笔记(深入)”;

所有函数的默认原型是Object的实例。因为函数的原型对象也是对象嘛! 对象当然是Object的实例咯!

因此函数的原型都会包含一个内部指针(__proto__), 指向Object.prototype。

这也是所有自定义类型都会继承toString()、valueOf()等默认方法的根本原因。

所以,上篇例子中展示的原型的原型链中还应该包括另外一个继承层次。

以下代码展示了这个完整的原型链。

//完整原型链的伪代码function Object() {}Object.prototype = {    constructor: f Object(),    hasOwnProperty: f hasOwnProperty(),    isPrototypeOf: f isPrototypeOf(),    propertyIsEnumerable: f propertyIsEnumerable(),    toLocaleString: f toLocaleString(),    toString: f toString(),    valueOf: f valueOf()}//SuperType 父类型function SuperType(){    this.property = true;}SuperType.prototype.getSuperProperty = function() {    console.log(this.property);    return this.property;}/*SuperType.prototype = {    constructor: f SuperType(),    getSuperProperty: function() {    console.log(this.property);    return this.property;    },     __proto__ : {        constructor: f Object(),        hasOwnProperty: f hasOwnProperty(),        isPrototypeOf: f isPrototypeOf(),        propertyIsEnumerable: f propertyIsEnumerable(),        toLocaleString: f toLocaleString(),        toString: f toString(),        valueOf: f valueOf()    }}*///SubType 子类型function SubType() {    this.subproperty = false;}//子类型 继承 父类型SubType.prototype = new SuperType();//实际上子类型的原型是这样的。/*SubType.prototype = {    property: true,    __proto__:  {        constructor : SuperType,        getSuperProperty:function() {            console.log(this.property);            return this.property;        }    }}*/SubType.prototype.getSubProperty = function(){    console.log(this.subproperty);    return this.subproperty;}//那么现在子类型的原型对象是这样的/*SubType.prototype = {    property: true,    getSubProperty: function()  {    console.log(this.subproperty);    return this.subproperty;    },    __proto__:  {        constructor : SuperType,        getSuperProperty:function() {            console.log(this.property);            return this.property;        }    }}*/var subInstanceObject = new SubType();console.log(subInstanceObject.getSuperProperty()); // true

登录后复制

一句话,SubType(子类型)继承了SuperType(父类型),

而SuperType(父类型)继承了Object(祖先)。

当调用subInstanceObject.toString()时,实际上调用的是在保存在Object.prototype中的那个方法。

2. 确定原型和实例对象关系

可以通过两种方式来确定原型和实例之间的关系。

第一种方式是使用instanceof操作符,只要检测的实例对象中的原型链包含出现过的构造函数,结果就会返回true。
因为,这说明他们都参与了,实例对象的创建。

console.log(subInstanceObject instanceof Object); // trueconsole.log(subInstanceObject instanceof SuperType); // trueconsole.log(subInstanceObject instanceof SubType); // true

登录后复制

由于原型链的关系, 我们可以说subIntanceObject是Object、SuperType或SubType中任何一个类型的实例。

第二种方式是使用isPrototypeOf()方法。同样,只要是原型链中出现过的原型,都可以说该原型链所派生的实例对象的原型。

console.log(Object.prototype.isPrototypeOf(subInstanceObject)); //trueconsole.log(SuperType.prototype.isPrototypeOf(subIntanceObject)); // trueconsole.log(SubType.prototype.isPrototypeOf(subIntanceObject)); //true

登录后复制

3. 谨慎地定义方法

子类型有时候需要覆盖父类型的某个方法,或者需要添加父类型中不存在的某个方法。

但不管怎么样,给原型添加方法的代码一定要放在替换原型的语句之后。

function SuperType() {    this.property = true;}SuperType.prototype.getSuperValue = function() {    return this.property;}function SubType() {    this.subproperty = false;}//继承了SuperTypeSubType.prototype = new SuperType();//给原型添加方法的代码一定要放在替换原型的语句之后//添加新方法SubType.prototype.getSubValue = function() {    return this.subproperty;}//重写 超类型中 的 方法SubType.prototype.getSuperValue = function() {    return false;}var instance = new SubType();console.log(instance.getSuperValue())

登录后复制

以上代码中,第一个方法getSubValue()被添加到了SubType中。
第二个方法getSuperValue()是原型中已经存在的一个方法。
重写这个方法将会子类的原型会查找到属于自己的getSuperValue()方法。
当通过SuperType的实例对象调用getSuperValue()时, 还会继续调用原来的那个方法。

再次强调,必须在用SuperType的实例对象替换原型之后,再定义两个方法。

还有一点需要提醒,即在通过原型链实现继承时,不能使用对象字面量创建原型方法。这样会重写原型链的。

function SuperType(){    this.property = true;}SuperType.prototype.getSuperValue = function(){    return this.property;}function SubType(){    this.subproperty = false;}//继承SuperTypeSubType.prototype = new SuperType();/* 现在的原型SubType.prototype = {    property: true,    __proto__: {        constructor: SuperType,        getSuperValue: function() {            return this.property;        }    }}*///使用对象字面量语法会改写原型,导致上一行代码无效// SubType.prototype = new Object();SubType.prototype = {    getSubValue: function() {        return this.subproperty;    },    someOtherMethod: function () {        return false;    }    /*,    __proto__ : {        constructor: fn Object(),        .......    }    */}var instance =  new SubType();console.log(instance.getSuperValue()); // error: instance.getSuperValue is not a function

登录后复制

以上代码展示了刚刚把SuperType的实例对象赋值给原型,紧接着又将原型替换成一个对象字面量而导致的问题。

因为SubType的原型其实保存的是一个Object的实例,而非SuperType的实例对象,因此这条链子断了。

4. 原型链的问题

原型链虽然很强大,可以用它来实现继承,但是总有缺点,世界上不存在万全法。

最主要的问题来自包含引用类型值的原型。

包含引用类型值的原型属性会被所有实例对象共享。

而这也正是组合使用原型模式和构造函数模式的原因。
在构造函数模式中定义属性,在原型模式中定义共享的方法。

在通过原型来实现原型继承时,原型实际上会变成另一个类型的实例对象。

原先的实例对象属性,也就变成了现在的原型属性了。

function SuperType() {    this.colors = ['red', 'green', 'blue'];}function SubType() {}// 子类型继承父类型SubType.prototype = new SuperType();/*SubType.prototype = {    colors: ['red', 'green', 'blue'],    __proto__: {        constructor: fn SuperType(),        .....    }}*/var instance1 = new SubType();instance1.colors.push('black');console.log(instance1.colors); // ['red', 'green', 'blue', 'black']var instance2 = new SubType();console.log(instance2.colors); // ['red', 'green', 'blue', 'black']

登录后复制

原型链的第二个问题在于, 没有办法在不影响所有实例对象的情况下,给父类型的构造函数传递参数。

由于上述两个问题的存在,事件中很少会单独使用原型链。

以上就是javascript原型链需要注意的地方的总结的详细内容,更多请关注【创想鸟】其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2738604.html

(0)
上一篇 2025年3月8日 01:52:35
下一篇 2025年3月8日 01:52:45

AD推荐 黄金广告位招租... 更多推荐

相关推荐

  • ES6中对象的新功能与解构赋值的详解(代码示例)

    本篇文章给大家带来的内容是关于es6中对象的新功能与解构赋值的详解(代码示例),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 ES6 通过字面量语法扩展、新增方法、改进原型等多种方式加强对象的使用,并通过解构简化对象的数据…

    编程技术 2025年3月8日
    200
  • 详解javascript浏览器的事件循环机制

    本篇文章给大家带来的内容是关于详解javascript浏览器的事件循环机制,有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 抛在前面的问题: 单线程如何做到异步 事件循环的过程是怎样的 macrotask 和 microta…

    2025年3月8日 编程技术
    200
  • 解析Node.js的事件循环机制

    本篇文章给大家带来的内容是关于解析node.js的事件循环机制,有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 在浏览器篇已经对事件循环机制和一些相关的概念作了详细介绍,但主要是针对浏览器端的研究,Node环境是否也一样呢?…

    2025年3月8日 编程技术
    200
  • Javascript中什么是流程控制(代码实例)

    本篇文章给大家带来的内容是介绍javascript中什么是流程控制(代码实例)。有一定的参考价值,有需要的朋友可以参考一下,希望对你们有所助。 流程控制 JavaScript通过流程语句来执行程序流,程序流有若干语句组成。在正常情况下,程序…

    编程技术 2025年3月8日
    200
  • JavaScript异步编程的详细介绍(附示例)

    本篇文章给大家带来的内容是关于php协成实现的详解(附代码),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 前言 自己着手准备写这篇文章的初衷是觉得如果想要更深入的理解 JS,异步编程则是必须要跨过的一道坎。由于这里面涉及…

    2025年3月8日
    200
  • javascript中原型和原型链的简单介绍

    本篇文章给大家带来的内容是关于javascript中原型和原型链的简单介绍 ,有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 [[Prototype]] 几乎所有对象在创建的时候都会生成[[Prototype]]链,就是人们…

    编程技术 2025年3月8日
    200
  • JavaScript如何给数组添加元素?js数组添加元素的3种方法(代码实例)

    数组是javascrip中中一个比较重要的部分,在学习js数组时,数组元素的操作是不可缺少的部分,那么你知道数组元素如何添加吗?本篇文章就给大家介绍如何往js数组(一维)中添加元素,让大家了解往js数组中添加元素的方法。有一定的参考价值,有…

    2025年3月8日
    200
  • JavaScript判断奇数和偶数的两种方法

    javascript是前端开发中必不可少的一部分,那你知道如何用js判断一个数是奇数还偶数吗?这篇文章就给大家分享js判断奇数和偶数的两种方法,有一定的参考价值,感兴趣的朋友可以参考一下。 方法一:用if函数判断数字能否整除2 描述:用if…

    2025年3月8日
    200
  • 什么是跨域?javascript跨域的四种方式介绍

    本篇文章给大家带来的内容是关于php协成实现的详解(附代码),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 一、什么是跨域 JavaScript出于安全方面的考虑,不允许跨域调用其他页面的对象。那什么是跨域呢,简单地理解就…

    2025年3月8日
    200
  • jQuery中API、事件和多库共存的简单介绍

    本篇文章给大家带来的内容是关于jquery中api、事件和多库共存的简单介绍,有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 API prop() 和 attr() prop() 方法用来改变影响DOM元素的动态状态,而不是…

    编程技术 2025年3月8日
    200

发表回复

登录后才能评论