什么是JS的原型链继承?

原型链是JavaScript实现继承的核心机制,通过对象的[[Prototype]]链接形成查找链。当访问对象属性时,若自身不存在,则沿原型链向上搜索直至null。每个构造函数的prototype属性为其实例的共同原型,实例通过__proto__指向它,从而实现属性和方法的共享。ES6的class语法是原型继承的语法糖,class使用extends实现继承,底层仍基于原型链,使代码更清晰但不改变继承本质。区分自身与继承属性可用hasOwnProperty()方法,该方法仅检测对象自身的属性,不包括原型链上的属性。常见陷阱包括原型上引用类型属性被所有实例共享导致的数据污染,应将引用类型属性定义在构造函数中以确保独立性。此外,属性遮蔽会覆盖原型属性,需注意赋值时的影响范围。

什么是js的原型链继承?

JavaScript 的原型链继承,说白了,就是对象之间共享属性和方法的一种机制。它不是传统意义上类(Class)那种基于蓝图的继承,而更像是一种“如果你没有,就去问你爸妈要”的链式查找过程。当你想访问一个对象的某个属性或方法时,JS 会先看看这个对象自己有没有。如果没有,它就会沿着一条特殊的链条,去它的“原型”(prototype)对象上找,如果原型对象也没有,就继续往上找原型的原型,直到找到或者找到链条的尽头——

null

为止。这条链,就是原型链。

解决方案

理解原型链继承,我们得从几个核心概念入手。每个 JavaScript 对象(除了少数特例外,比如

Object.prototype

[[Prototype]]

null

)都有一个内部属性

[[Prototype]]

,在许多环境中可以通过

__proto__

访问到,它指向这个对象的原型。同时,函数(作为构造函数使用时)会有一个

prototype

属性,这个

prototype

属性指向一个对象,这个对象就是通过该构造函数创建的所有实例的共同原型。

当使用

new

关键字创建一个新对象时,比如

let obj = new MyConstructor();

,新创建的

obj

[[Prototype]]

就会被设置为

MyConstructor.prototype

。这就是继承的起点。当你试图访问

obj.someProperty

时:

JavaScript 会先在

obj

自身查找

someProperty

。如果

obj

没有,它就会沿着

obj

[[Prototype]]

(也就是

MyConstructor.prototype

)去查找。如果

MyConstructor.prototype

也没有,它会继续沿着

MyConstructor.prototype

[[Prototype]]

向上查找,这个通常会是

Object.prototype

。如果

Object.prototype

还没有,那就继续向上,直到最终的

null

。如果到

null

还没找到,就返回

undefined

这个查找过程,就是原型链在发挥作用。它使得我们可以在原型对象上定义共享的属性和方法,所有实例都可以访问到,从而实现代码复用。

function Person(name) {    this.name = name;}// 在Person的原型上添加方法Person.prototype.sayHello = function() {    console.log(`你好,我是 ${this.name}`);};Person.prototype.species = '人类'; // 共享属性let person1 = new Person('张三');let person2 = new Person('李四');person1.sayHello(); // 你好,我是 张三person2.sayHello(); // 你好,我是 李四console.log(person1.species); // 人类console.log(person2.species); // 人类// 验证原型链console.log(person1.__proto__ === Person.prototype); // trueconsole.log(Person.prototype.__proto__ === Object.prototype); // trueconsole.log(Object.prototype.__proto__); // null

你看,

sayHello

species

这两个东西,我们只在

Person.prototype

上定义了一次,但

person1

person2

都能用。这就是原型链的魅力。

原型链继承和ES6的class继承有什么区别

这个问题经常被问到,尤其是在现代JavaScript开发中。在我看来,ES6 的

class

语法其实就是 JavaScript 原型链继承的一种“语法糖”,它并没有引入一种全新的继承机制,而是提供了一种更接近传统面向对象语言(比如 Java 或 C++)的、更易读、更直观的方式来组织和实现基于原型的继承。

你可以把

class

看作是给那些习惯了类式继承的开发者提供的一层“友好包装”。它用

class

关键字定义构造函数,用

extends

实现继承,用

super

调用父类的构造函数或方法,这些都让代码看起来更像我们熟悉的“类”。

比如,用 ES6

class

实现上面的

Person

例子:

class Person {    constructor(name) {        this.name = name;    }    sayHello() {        console.log(`你好,我是 ${this.name}`);    }    static species = '人类'; // ES6 class field, 也可以写在原型上}class Student extends Person {    constructor(name, school) {        super(name); // 调用父类构造函数        this.school = school;    }    study() {        console.log(`${this.name} 正在 ${this.school} 学习。`);    }}let student1 = new Student('王五', '清华大学');student1.sayHello(); // 你好,我是 王五student1.study(); // 王五 正在 清华大学 学习。// 尽管是class语法,底层依然是原型链console.log(student1.__proto__ === Student.prototype); // trueconsole.log(Student.prototype.__proto__ === Person.prototype); // trueconsole.log(Person.prototype.__proto__ === Object.prototype); // true

从上面的代码就能看出来,

Student.prototype

的原型指向了

Person.prototype

,这正是原型链的体现。所以,

class

语法更多的是改善了开发者体验,让代码结构更清晰,但它并没有改变 JavaScript 对象模型的核心——原型。在我个人看来,理解了原型链,再去学

class

,你会发现很多东西豁然开朗,而不是停留在表面的语法层面。

如何判断一个属性是对象自身的还是继承来的?

在实际开发中,我们经常会遇到需要区分一个属性是直接定义在对象实例上的,还是通过原型链继承而来的情况。这对于避免一些潜在的副作用,或者进行精确的对象操作非常重要。JavaScript 提供了一个非常实用的方法来解决这个问题:

hasOwnProperty()

hasOwnProperty()

Object.prototype

上的一个方法,因此几乎所有对象都可以调用它(除非你手动覆盖或删除了它)。它的作用就是检查对象自身是否拥有某个特定的属性,而不会去查找原型链。如果属性存在于对象自身,它就返回

true

;否则(无论是继承来的还是不存在),都返回

false

function Vehicle(type) {    this.type = type;}Vehicle.prototype.wheels = 4;Vehicle.prototype.getColor = function() {    return 'red';};let car = new Vehicle('轿车');car.brand = '奔驰'; // 自身属性console.log(car.hasOwnProperty('brand')); // true (自身属性)console.log(car.hasOwnProperty('type'));  // true (自身属性,在构造函数中定义)console.log(car.hasOwnProperty('wheels')); // false (继承自原型)console.log(car.hasOwnProperty('getColor')); // false (继承自原型)console.log(car.hasOwnProperty('toString')); // false (继承自Object.prototype)console.log(car.hasOwnProperty('nonExistent')); // false (不存在)// 另一个相关的操作符是 `in`// `in` 操作符会检查属性是否存在于对象自身或其原型链上console.log('brand' in car); // trueconsole.log('wheels' in car); // trueconsole.log('getColor' in car); // trueconsole.log('nonExistent' in car); // false

通过对比

hasOwnProperty()

in

操作符,你可以清晰地看到它们的区别。

hasOwnProperty()

严格只看“自家”的,而

in

则会“问遍祖宗十八代”。在遍历对象属性时,比如使用

for...in

循环,通常会配合

hasOwnProperty()

来过滤掉继承来的属性,只处理对象自身的属性,这是一种很常见的最佳实践,能有效避免遍历到不必要的原型链上的属性。

原型链继承有哪些常见的陷阱或注意事项?

原型链继承虽然强大且是JS的核心,但它也确实有一些需要注意的“坑”,或者说,是需要我们特别留心的地方。在我看来,最让人头疼的莫过于对引用类型属性的误解。

1. 引用类型属性的共享陷阱:这是最常见也最容易犯错的地方。如果在原型上定义了一个引用类型的属性(比如数组或对象),那么所有通过该原型创建的实例都会共享这同一个引用类型属性。这意味着,如果你通过一个实例去修改这个引用类型属性,所有其他实例也会受到影响。

function Gadget(name) {    this.name = name;}Gadget.prototype.features = []; // 在原型上定义一个数组(引用类型)let phone = new Gadget('手机');let laptop = new Gadget('笔记本');phone.features.push('拍照');console.log(phone.features); // ["拍照"]console.log(laptop.features); // ["拍照"] // 坑!laptop的features也被修改了// 如果你在实例上直接赋值,则会创建自身的属性,从而“遮蔽”原型上的同名属性phone.features = ['打电话', '上网']; // 现在phone有了自己的features数组console.log(phone.features); // ["打电话", "上网"]console.log(laptop.features); // ["拍照"] // laptop的features仍然是共享的那个

要避免这个陷阱,通常的做法是在构造函数中定义引用类型的属性,这样每个实例都会有自己独立的副本。

function GadgetSafe(name) {    this.name = name;    this.features = []; // 在构造函数中定义,每个实例独有}let phoneSafe = new GadgetSafe('手机');let laptopSafe = new GadgetSafe('笔记本');phoneSafe.features.push('拍照');console.log(phoneSafe.features); // ["拍照"]console.log(laptopSafe.features); // [] // 互不影响,完美!

2. 性能考量(微乎其微但值得一提):理论上,过长的原型链会稍微增加属性查找的时间,因为 JavaScript 引擎需要遍历更多的对象。但在绝大多数实际应用中,这种性能开销是微不足道的,几乎可以忽略不计。除非你真的构建了一个拥有几十甚至上百层继承的原型链,否则不必为此过于担心。

3.

this

指向问题:当原型上的方法被调用时,

this

的指向仍然取决于方法的调用方式。这和普通函数的

this

行为是一致的,而不是因为继承就有所不同。如果一个原型方法被作为对象的方法调用,

this

将指向该对象实例。但如果它被独立提取出来调用,

this

可能就指向全局对象(严格模式下是

undefined

)。

4. 属性遮蔽(Shadowing):当你给一个实例添加一个与原型上同名的属性时,实例上的属性会“遮蔽”原型上的属性。这意味着,访问该属性时,JavaScript 会优先找到实例自身的属性,而不会再向上查找原型链。虽然这通常不是一个“陷阱”,但理解其行为对于避免意外覆盖或访问错误属性至关重要。

function Animal() {}Animal.prototype.name = '通用动物';let dog = new Animal();console.log(dog.name); // 通用动物dog.name = '旺财'; // 在dog实例上创建了一个新的name属性,遮蔽了原型上的nameconsole.log(dog.name); // 旺财delete dog.name; // 删除实例上的name属性console.log(dog.name); // 通用动物 (现在又可以访问到原型上的name了)

总的来说,原型链继承是 JavaScript 的基石,理解它能帮助我们更深入地掌握这门语言。虽然有一些细微之处需要注意,但只要掌握了这些要点,就能更好地利用它的强大功能。

以上就是什么是JS的原型链继承?的详细内容,更多请关注创想鸟其它相关文章!

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

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
JavaScript 中使用类实现动态数组的统计分析工具
上一篇 2025年12月20日 11:31:50
什么是JS的异步编程?
下一篇 2025年12月20日 11:32:02

相关推荐

  • composer require-dev和require有什么不同_Composer Require与Require-Dev区别解析

    require用于声明项目运行必需的依赖,如框架、数据库组件和第三方SDK,这些包会随项目部署到生产环境;2. require-dev用于声明仅在开发和测试阶段需要的工具,如PHPUnit、PHPStan、Faker等,不会默认部署到生产环境;3. 安装时composer install根据环境决定…

    2026年5月10日
    1000
  • 修复Django电商项目中AJAX过滤产品列表图片不显示问题

    在Django电商项目中,当使用AJAX动态加载过滤后的产品列表时,常遇到图片无法正常显示的问题。这通常是由于前端模板中图片加载方式(如data-setbg属性结合JavaScript库)与AJAX动态内容更新机制不兼容所致。解决方案是直接在AJAX返回的HTML中使用标准的标签来渲染图片,确保浏览…

    2026年5月10日
    000
  • 开源免费PHP工具 PHP开发效率提升利器

    推荐开源免费PHP开发工具以提升效率:VS Code、Sublime Text轻量高效,PhpStorm专业强大;调试用Xdebug、Kint、Ray;依赖管理选Composer;代码质量工具包括PHPStan、Psalm、PHP_CodeSniffer;数据库管理可用%ignore_a_1%MyA…

    2026年5月10日
    000
  • Matplotlib 地图中多类型图例的创建与优化

    Matplotlib 地图中多类型图例的创建与优化Matplotlib 地图中多类型图例的创建与优化Matplotlib 地图中多类型图例的创建与优化Matplotlib 地图中多类型图例的创建与优化

    本教程旨在解决matplotlib地图可视化中,如何在一个图例中同时展示颜色块(如区域分类)和自定义标记(如特定兴趣点)的问题。文章详细介绍了当传统`patch`对象无法正确显示标记时,如何利用`matplotlib.lines.line2d`创建标记图例句柄,并将其与颜色块图例句柄合并,从而生成一…

    2026年5月10日 用户投稿
    100
  • Golang JSON序列化:控制敏感字段暴露的最佳实践

    本教程探讨golang中如何高效控制结构体字段在json序列化时的可见性。当需要将包含敏感信息的结构体数组转换为json响应时,通过利用`encoding/json`包提供的结构体标签,特别是`json:”-“`,可以轻松实现对特定字段的忽略,从而避免敏感数据泄露,确保api…

    2026年5月10日
    000
  • 怎么在PHP代码中实现图片上传功能_PHP图片上传功能实现与安全处理教程

    首先创建含enctype的HTML表单,再用PHP接收文件,检查目录、移动临时文件,验证类型与大小,生成唯一文件名,并调整php.ini限制以确保上传成功。 如果您尝试在PHP项目中添加图片上传功能,但服务器无法正确接收或保存文件,则可能是由于表单配置、文件处理逻辑或安全限制的问题。以下是实现该功能…

    2026年5月10日
    100
  • c++中的SFINAE技术是什么_c++模板编程中的SFINAE原理与应用

    SFINAE 是“替换失败不是错误”的原则,指模板实例化时若参数替换导致错误,只要存在其他合法候选,编译器不报错而是继续重载决议。它用于条件启用模板、类型检测等场景,如通过 decltype 或 enable_if 控制函数重载,实现类型特征判断。尽管 C++20 引入 Concepts 简化了部分…

    2026年5月10日
    000
  • RichHandler与Rich Progress集成:解决显示冲突的教程

    在使用rich库的`richhandler`进行日志输出并同时使用`progress`组件时,可能会遇到显示错乱或溢出问题。这通常是由于为`richhandler`和`progress`分别创建了独立的`console`实例导致的。解决方案是确保日志处理器和进度条组件共享同一个`console`实例…

    2026年5月10日
    000
  • 修复点击时按钮抖动:CSS垂直对齐实践

    本文探讨了在Web开发中,交互式按钮(如播放/暂停按钮)在点击时发生意外垂直位移的问题。通过分析CSS样式变化对元素布局的影响,我们发现这是由于按钮不同状态下的边框样式和内边距改变,以及默认的垂直对齐行为共同作用所致。核心解决方案是利用CSS的vertical-align属性,将其设置为middle…

    2026年5月10日
    000
  • 使用 Jupyter Notebook 进行探索性数据分析

    Jupyter Notebook通过单元格实现代码与Markdown结合,支持数据导入(pandas)、清洗(fillna)、探索(matplotlib/seaborn可视化)、统计分析(describe/corr)和特征工程,便于记录与分享分析过程。 Jupyter Notebook 是进行探索性…

    2026年5月10日
    000
  • php常量怎么用_PHP常量(define/const)定义与使用方法

    PHP中可通过define函数和const关键字定义常量,用于存储不可变值。define适用于全局作用域,支持动态名称和条件定义,如define(‘SITE_NAME’, ‘MyWebsite’);const在编译时生效,语法简洁但限制多,只能在类或全…

    2026年5月10日
    000
  • 如何在HTML中插入表单元素_HTML表单控件与输入类型使用指南

    HTML表单通过标签构建,包含action和method属性定义数据提交目标与方式,常用input类型如text、password、email等适配不同输入需求,配合label、required、placeholder提升可用性,结合textarea、select、button等控件实现完整交互,是…

    2026年5月10日
    000
  • 前端缓存策略与JavaScript存储管理

    根据数据特性选择合适的存储方式并制定清晰的读写与清理逻辑,能显著提升前端性能;合理运用Cookie、localStorage、sessionStorage、IndexedDB及Cache API,结合缓存策略与定期清理机制,可在保证用户体验的同时避免安全与性能隐患。 前端缓存和JavaScript存…

    2026年5月10日
    100
  • c#文件怎么打开

    打开 C# 文件有三种方法:Visual Studio:启动 Visual Studio,通过“文件”菜单打开 C# 文件。文本编辑器:使用文本编辑器打开 C# 文件,将其视为普通文本。.NET Core 命令行工具:使用 csc.exe 命令行工具编译 C# 文件,生成可执行文件。 如何打开 C#…

    2026年5月10日
    000
  • HTML5网页如何实现手势操作 HTML5网页移动端交互的处理技巧

    首先利用原生touch事件实现滑动判断,再通过preventDefault解决滚动冲突,接着引入Hammer.js处理复杂手势,最后通过优化点击区域、避免事件冲突和增加视觉反馈提升体验。 在移动端浏览器中,HTML5网页可以通过触摸事件实现手势操作,提升用户体验。虽然原生JavaScript提供了基…

    2026年5月10日
    000
  • 深入理解 Express.js 中 next() 参数的作用与中间件机制

    本文深入探讨 express.js 中间件函数中的 `next()` 参数。它负责将控制权传递给请求-响应周期中的下一个中间件或路由处理程序。文章将详细解释 `next()` 的工作原理、中间件的注册与执行顺序,以及不正确使用 `next()` 可能导致请求挂起的风险,并通过代码示例和实际应用场景,…

    2026年5月10日
    000
  • 使用 WebCodecs VideoDecoder 实现精确逐帧回退

    本文档旨在解决在使用 WebCodecs VideoDecoder 进行视频解码时,实现精确逐帧回退的问题。通过比较帧的时间戳与目标帧的时间戳,可以避免渲染中间帧,从而提高用户体验。本文将提供详细的解决方案和示例代码,帮助开发者实现精确的视频帧控制。 在使用 WebCodecs VideoDecod…

    2026年5月10日
    000
  • JavaScript 闭包:理解闭包原理与内存泄漏问题

    闭包是函数访问其外部作用域变量的能力,即使外部函数已执行完毕。如 inner 函数引用 outer 中的 count,形成闭包,使变量持久存在。闭包本身无害,但可能因延长变量生命周期导致内存泄漏,例如事件监听器引用大对象时。若未及时清理 DOM 事件或定时器,闭包会阻止垃圾回收,造成内存占用过高。解…

    2026年5月10日
    000
  • JavaScript 动态菜单点击高亮效果实现教程

    本教程详细介绍了如何使用 JavaScript 实现动态菜单的点击高亮功能。通过事件委托和状态管理,当用户点击菜单项时,被点击项会高亮显示(绿色),同时其他菜单项恢复默认样式(白色)。这种方法避免了不必要的DOM操作,提高了性能和代码可维护性,确保了无论点击方向如何,功能都能稳定运行。 动态菜单高亮…

    2026年5月10日
    200
  • c++如何实现UDP通信_c++基于UDP的网络通信示例

    UDP通信基于套接字实现,适用于实时性要求高的场景。1. 流程包括创建套接字、绑定地址(接收方)、发送(sendto)与接收(recvfrom)数据、关闭套接字;2. 服务端监听指定端口,接收客户端消息并回传;3. 客户端发送消息至服务端并接收响应;4. 跨平台需处理Winsock初始化与库链接,编…

    2026年5月10日
    000

发表回复

登录后才能评论
关注微信