0

0

JS原型链是什么原理

煙雲

煙雲

发布时间:2025-08-16 13:01:01

|

405人浏览过

|

来源于php中文网

原创

原型链是javascript实现继承的核心机制,当对象查找属性时会沿原型链向上查找直至找到或到达null。1. 每个对象的__proto__指向其构造函数的prototype;2. 构造函数的prototype也是对象,其__proto__指向父类原型,最终指向object.prototype;3. object.prototype.__proto__为null,标志链的结束;4. class语法糖本质仍基于原型链,extends实现继承关系,super调用父类构造函数;5. 避免原型链污染应不修改object.prototype、使用object.create(null)创建无原型对象、用object.freeze()冻结对象及严格验证输入。理解原型链有助于掌握继承、避免bug并提升代码安全性,该机制贯穿于js对象属性查找全过程,是面向对象编程的基础。

JS原型链是什么原理

JS原型链,简单来说,就是对象查找属性时的一条向上查找的路径。当一个对象自身没有某个属性时,它会沿着原型链向上查找,直到找到该属性或者到达原型链的顶端(

null
)。

解决方案

原型链是 JavaScript 实现继承的核心机制。每个 JavaScript 对象都有一个指向其原型(prototype)的内部链接,这个原型对象也有自己的原型,以此类推,形成一个链条,这就是原型链。

具体来说:

  1. __proto__
    prototype
    :
    每个 JavaScript 对象(除了
    null
    )都有一个
    __proto__
    属性,指向创建它的构造函数的
    prototype
    属性指向的对象。 而函数对象比较特殊,它除了拥有
    __proto__
    属性之外,还拥有
    prototype
    属性。
  2. 构造函数: 当使用
    new
    关键字调用一个函数时,这个函数就成为了构造函数。 构造函数的作用是创建新的对象,并且会将新对象的
    __proto__
    属性指向构造函数的
    prototype
    属性指向的对象。
  3. Object.prototype
    :
    所有对象的原型链最终都会指向
    Object.prototype
    ,它是所有对象的基类。
    Object.prototype
    __proto__
    指向
    null
    ,标志着原型链的终结。

举个例子:

function Person(name) {
  this.name = name;
}

Person.prototype.greet = function() {
  console.log('Hello, my name is ' + this.name);
};

const john = new Person('John');
john.greet(); // 输出 "Hello, my name is John"

console.log(john.__proto__ === Person.prototype); // true
console.log(Person.prototype.__proto__ === Object.prototype); // true
console.log(Object.prototype.__proto__ === null); // true

在这个例子中:

  • john
    对象的
    __proto__
    指向
    Person.prototype
  • Person.prototype
    对象的
    __proto__
    指向
    Object.prototype
  • Object.prototype
    对象的
    __proto__
    指向
    null

所以,当

john
对象调用
greet()
方法时,它首先在自身查找,没有找到,然后沿着原型链向上查找,在
Person.prototype
中找到了
greet()
方法,于是执行该方法。

原型链的查找过程

蓝色商务公司网站(XDcms内核)1.0
蓝色商务公司网站(XDcms内核)1.0

本系统使用的是XDcms内核,在原来基础上做来相应修改 前台修改调用数据,可以使用{loop catid=栏目ID}{/loop}方式调用 主要功能: A、内容管理模型,自定义字段,更方便扩展功能。自带模型:单页模型、新闻模型、产品模型、招聘模型 B、栏目自定义,便于内容管理 C、内容模块化,二次开发更便捷。自带模块:幻灯片、QQ客服、友情链接、自定义表单(在线留言、简历管理) D、模板管理,后台

下载

当访问一个对象的属性时,JavaScript 引擎会按照以下步骤查找:

  1. 先在对象自身查找该属性。
  2. 如果对象自身没有该属性,则沿着
    __proto__
    向上查找,即在其原型对象中查找。
  3. 如果原型对象中仍然没有该属性,则继续沿着原型对象的
    __proto__
    向上查找,直到找到该属性或者到达原型链的顶端(
    null
    )。
  4. 如果在原型链的顶端仍然没有找到该属性,则返回
    undefined

为什么理解原型链对 JavaScript 开发至关重要?

理解原型链是深入理解 JavaScript 继承机制的关键。 它直接影响着我们如何组织和复用代码。如果不理解原型链,可能会导致一些意想不到的 bug,例如属性查找错误或者原型污染。 此外,掌握原型链有助于我们更好地理解 JavaScript 中的面向对象编程思想,并能更灵活地运用各种设计模式。

如何避免原型链污染?

原型链污染指的是恶意修改对象的原型,从而影响到所有基于该原型创建的对象。 这可能导致安全漏洞。 避免原型链污染的方法包括:

  • 避免直接修改
    Object.prototype
    :
    尽量不要直接修改
    Object.prototype
    ,因为这会影响到所有对象。
  • 使用
    Object.create(null)
    创建对象:
    使用
    Object.create(null)
    创建的对象没有原型,因此不会受到原型链污染的影响。
  • 使用
    Object.freeze()
    冻结对象:
    使用
    Object.freeze()
    可以冻结对象,使其属性不可修改。
  • 进行输入验证: 对用户输入进行严格的验证,防止恶意代码注入。

一个简单的例子,展示了如何使用

Object.create(null)
来创建一个没有原型的对象,避免原型链污染:

const obj = Object.create(null);
obj.hasOwnProperty = function() {
  return false; // 尝试覆盖 hasOwnProperty
};

console.log(obj.hasOwnProperty('name')); // 报错,因为 hasOwnProperty 不是一个函数

// 正常情况下,我们应该这样使用 hasOwnProperty:
const normalObj = {};
console.log(Object.prototype.hasOwnProperty.call(normalObj, 'name')); // false

在这个例子中,由于

obj
没有原型,所以尝试访问
obj.hasOwnProperty
会报错。 这避免了恶意代码覆盖
hasOwnProperty
方法,从而防止原型链污染。

原型链和
class
语法糖有什么关系?

class
语法糖是 ES6 引入的,它提供了一种更简洁的语法来定义类。 但本质上,
class
仍然是基于原型链实现的。
class
只是一个语法糖,它隐藏了原型链的细节,让代码更易读易写。

例如:

class Animal {
  constructor(name) {
    this.name = name;
  }

  speak() {
    console.log(this.name + ' makes a sound.');
  }
}

class Dog extends Animal {
  constructor(name) {
    super(name);
  }

  speak() {
    console.log(this.name + ' barks.');
  }
}

const dog = new Dog('Buddy');
dog.speak(); // 输出 "Buddy barks."

console.log(dog.__proto__ === Dog.prototype); // true
console.log(Dog.prototype.__proto__ === Animal.prototype); // true

在这个例子中,

Dog
类继承自
Animal
类。 虽然我们使用了
class
语法,但实际上
Dog.prototype
__proto__
仍然指向
Animal.prototype
,这体现了原型链的继承关系。
super()
方法实际上调用了父类的构造函数,并在子类的构造函数中设置了
this
上下文。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
es6新特性
es6新特性

es6新特性有:1、块级作用域变量;2、箭头函数;3、模板字符串;4、解构赋值;5、默认参数;6、 扩展运算符;7、 类和继承;8、Promise。本专题为大家提供es6新特性的相关的文章、下载、课程内容,供大家免费下载体验。

106

2023.07.17

es6新特性有哪些
es6新特性有哪些

es6的新特性有:1、块级作用域;2、箭头函数;3、解构赋值;4、默认参数;5、扩展运算符;6、模板字符串;7、类和模块;8、迭代器和生成器;9、Promise对象;10、模块化导入和导出等等。本专题为大家提供es6新特性的相关的文章、下载、课程内容,供大家免费下载体验。

195

2023.08.04

JavaScript ES6新特性
JavaScript ES6新特性

ES6是JavaScript的根本性升级,引入let/const实现块级作用域、箭头函数解决this绑定问题、解构赋值与模板字符串简化数据处理、对象简写与模块化提升代码可读性与组织性。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

222

2025.12.24

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

237

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

458

2024.03.01

go语言 面向对象
go语言 面向对象

本专题整合了go语言面向对象相关内容,阅读专题下面的文章了解更多详细内容。

56

2025.09.05

java面向对象
java面向对象

本专题整合了java面向对象相关内容,阅读专题下面的文章了解更多详细内容。

52

2025.11.27

go语言 面向对象
go语言 面向对象

本专题整合了go语言面向对象相关内容,阅读专题下面的文章了解更多详细内容。

56

2025.09.05

C++ 设计模式与软件架构
C++ 设计模式与软件架构

本专题深入讲解 C++ 中的常见设计模式与架构优化,包括单例模式、工厂模式、观察者模式、策略模式、命令模式等,结合实际案例展示如何在 C++ 项目中应用这些模式提升代码可维护性与扩展性。通过案例分析,帮助开发者掌握 如何运用设计模式构建高质量的软件架构,提升系统的灵活性与可扩展性。

9

2026.01.30

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Git 教程
Git 教程

共21课时 | 3.1万人学习

Django 教程
Django 教程

共28课时 | 3.7万人学习

MySQL 教程
MySQL 教程

共48课时 | 2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号