JavaScript深入浅出__proto__和prototype

目录
  • 构造函数和实例
  • prototype
  • constructor
  • 原型对象的原型
  • 原型链
  • 扩展知识
    • 关于 Object 和 Function
  • 总结

首先我们先记住几个知识点:

  • 每个函数都有一个prototype属性
  • 每个对象都有一个__proto__属性(null除外)
  • 函数也是对象

构造函数和实例

首先我们通过下面的例子了解些基本的概念

function Person() {
}
var person1 = new Person()
  • 使用 new 创建对象的函数就是构造函数、创建出的对象就是构造函数的实例对象
  • 本例中:Person 就是一个构造函数,我们使用 new 创建了一个实例对象 person1
  • 构造函数大写只是约定俗成的习惯,实际上任何可以使用 new 运算符的函数都可以是构造函数

prototype

function Person() {
}
Person.prototype.name = 'Person'
var person1 = new Person()
console.log(person1.name) // Person
console.log(Person.prototype)

  • Person 的 prototype 属性指向的是 Person.prototype 的原型对象
  • 在这个例子中可以看出,person1 本身并没有 name 属性,访问的其实是 Person 中prototype 的 name
  • 从这里可以得出必然有一种关系把person1Person关联起来,使得person1可以访问到Person中prototype的属性值

带着上面的疑问我们继续下面的例子

__proto__prototype的关系

function Person() {
}
Person.prototype.name = 'Person'
var person1 = new Person()
console.log(person1)
console.log(person1.__proto__.name === Person.prototype.name) // true
console.log(person1.__proto__ === Person.prototype) // true

一开始我们就提到,每个对象都有一个__proto__属性(null除外),通过这个例子我们发现person1的__proto__属性下有个name属性,正好是Person.prototype.name的值,由此我们可以看出实例person1是通过__proto__访问的构造函数Person的prototype属性

根据上面的结论,我们很容易得出以下关系图

由此我们很容易得出,多个示例对象之间通过__proto__进行关联,可以通过__proto__共享Person.prototype上的属性

constructor

既然构造函数和实例都可以指向原型,那么原型是否有属性指向构造函数或者实例呢? 通过上面的例子,我们发现除了__proto__,还有一个constructor属性

function Person() {
}
Person.prototype.name = 'Person'
var person1 = new Person()
console.log(Person) // ƒ Person() {}
console.log(Person.prototype.constructor) // ƒ Person() {}
console.log(person1.__proto__.constructor) // ƒ Person() {}
// 由此我们发现`constructor`属性指向的是Person构造函数本身,不难得出以下结论
console.log(Person === Person.prototype.constructor) // true

由此我们可以得出以下关系图

原型对象的原型

  • 原型也是一个对象,一开始就提到,每个对象都有一个__proto__属性(null除外),因此原型对象也是有__proto__
  • 实际上原型对象就是通过 Object 构造函数生成的
var prototypeObj = new Object()
console.log(prototypeObj.__proto__ === Object.prototype) // true

到此我们可以得出一个新的关系图

到这里大家可能就有疑惑了,这样不就无限循环了吗,Object.protoType__proto__又是什么呢?

console.log(Object.protoType.__proto__) // null
console.log(Object.protoType.__proto__ === null) // true

所以 Object.prototype 为null,属性查找到这里也就结束了

原型链

由上面的例子我们得知,在查找对象的属性时,优先查找实例对象的属性,查找不到时就会通过__proto__ 查找 prototype原型对象的属性,还查不到会通过prototype的__proto__继续查找,直到Object.protoType.__proto__为止

图中由__proto__组成的红色链路就是我们所说的原型链

扩展知识

关于 Object 和 Function

既然函数也是对象,对象都有 __proto__ ,那么 Object 和 Function 的 __proto__ 属性又是什么呢?

事实上

  • Object对象是由Function构造函数创建的
  • Function的原型对象Function.prototype是由Object构造函数创建的
  • 【按照原型的定义,可以理解为】Function对象是由Function构造函数本身创建

根据原型链的相关知识,实例对象的 __proto__ 指向构造函数的原型对象 prototype

// Object对象由Function构造函数创建
Object.__proto__ === Function.prototype // true
// Function的原型对象`Function.prototype`是由Object创建
Function.prototype.__proto__ === Object.prototype // true
// Function由Function本身创建(按照原型的定义可以简单这么去理解,这里不做深究)
Function.__proto__ === Function.prototype // true
Object.getPrototypeOf(Function) === Function.prototype // true

由此我们可以得出最终的关系图

关于 getPrototypeOfisPrototypeOfinstanceof

function Person() {
}
var person1 = new Person()
// Object.getPrototypeOf(obj) 返回obj实例对象的原型(obj.__proto__)
console.log(Object.getPrototypeOf(person1) === Person.prototype) // true
// Object.isPrototypeOf(obj),如果obj.__proto__和Object.prototype在一条原型链上(或者理解为Object为obj的构造函数或父级构造函数),则返回true
console.log(Object.prototype.isPrototypeOf(person1)) // true
console.log(Object.prototype.isPrototypeOf(Person.prototype)) // true
console.log(Person.prototype.isPrototypeOf(person1)) // true
// (obj instanceof Object) 同isPrototypeOf类似,obj.__proto__和Object.prototype在一条原型链上,则返回true
console.log(person1 instanceof Person) // true
console.log(person1 instanceof Object) // true

总结

  • 每个函数都有一个prototype属性,值是一个原型对象
  • 每个对象都有一个__proto__属性(null除外),指向构造函数的原型prototype
  • 构造函数的原型对象的constructor指向构造函数本身
  • 原型对象是由Object构造函数实例化产生的,所以原型对象的__proto__指向Object的原型对象Object.prototype
  • Object构造函数的原型对象为null
function Person() {}
var person = new Person()
console.log(person.__proto__ === Person.prototype) // true
console.log(Person === Person.prototype.constructor) // true
console.log(Object.prototype.__proto__ === null) // true
// 推导person.constructor等于person.__proto__.constructor等于Person.prototype.constructor
console.log(person.constructor === Person.prototype.constructor) // true

到此这篇关于JavaScript深入浅出__proto__和prototype的文章就介绍到这了,更多相关JS proto 和 prototype内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • JavaScript使用prototype属性实现继承操作示例

    本文实例讲述了JavaScript使用prototype属性实现继承操作.分享给大家供大家参考,具体如下: JS并没有显式的继承语法,在JS中所有的对象都是Object的子类实现, 因而对象之间是平等关系. 尽管如此我们可以通过特殊的方法达到继承的效果. 当然JS也不能直接定义类, 我们通过定义函数可以得到一个同名的类 , 同时这个函数就是这个类的构造器, 在定义函数时以this修饰的变量就是定义的 类的实例中的属性,当这个属性时函数时,  就可以认为这个属性变成了一个实例方法 //定义一个Pe

  • JavaScript原型Prototype详情

    目录 1.概述 1.1原型是什么 1.2获取原型 2.原型属性 2.1利用原型添加属性与方法. 2.2访问原型属性原型方法 3.自有属性与原型属性 3.1检测自有属性或者原型属性 4.isPrototypeOf()方法 5.扩展内置对象 6.结语 1.概述 1.1原型是什么 在JavaScript中,函数是一个包含属性和方法的Function类型的对象.而原型(Prototype )就是Function类型对象的一个属性. 在函数定义是包含了prototype属性,它的初始值是一个空对象 .在J

  • JS数组降维的实现Array.prototype.concat.apply([], arr)

    把多维数组(尤其是二维数组)转化为一维数组是业务开发中的常用逻辑,最近跟着黄轶老师学习Vue2.6.1.1版本源码时,看到源码对二维数组降维的代码,所以这里来写一篇,记录一下,加强印象 二维数组降为一维数组 循环降维 let children = [1, 2, 3, [4, 5, 6], 7, 8, [9, 10]]; function simpleNormalizeChildren(children) { let reduce = []; for (let i = 0; i < childre

  • js String.prototype.trim字符去前后空格的扩展

    最近学习js的时候发现的这个函数,这样很方便地去除前后空格,用正则实现,简单方便.下面我们小编就为大家分享一下几种实现方式. String.Prototype.trim() trim()返回一个字符串两端空白字符被删除的新字符串,不影响原字符串. /*内置对象添加方法:String.prototype.trim(给String添加一个trim方法) *^这个是以什么什么开头 *$这个是以什么什么结尾 *'/s是String /d是数字' *replace(/^\s+/ , "")把以字

  • Javascript之深入浅出prototype

    我们先来讲一个故事,一个大大的池塘,里面有很多鱼.这是属于我们大家的池塘所以里面的鱼我们都可以吃,但是我们也会从集市买一些鱼放在家里,那么放在家里的鱼肯定是属于我们私人的,外人是不会拥有的.那么在js里我们就把这个池塘称为原型对象,池塘里面我们所共享的鱼称为原型中的属性及方法,而我们自己的鱼称为构造函数中的属性及方法,我们是什么呢?对了,我们是对象的实例. 以上是为了让大家能够趣味性的对prototype有一个概念,接下来就通过代码具体总结一下prototype~ 一.理解prototype 我

  • JavaScript中isPrototypeOf函数

    目录 1.isPrototypeOf() 示例1,Object类实例: 示例2,自己定义Human类: 示例3,再来看看Object的原型(prototype)是否是human的原型: 示例4,Object.prototype是否是内置类的原型: 示例5,Object也是函数(类): 2.和 instanceof 的区别 有时看一些框架源码的时候,会碰到 isPrototypeOf() 这个函数,那么这个函数有什么作用呢? 1.isPrototypeOf() isPrototypeOf() 是

  • 一文彻底理解js原生语法prototype,__proto__和constructor

    目录 1 前言 2 前置知识点 2.1 数据类型 2.2 判断是否是自身属性(hasOwnProperty) 3 一点小思考 3.1 修改 constructor 3.1.1 instanceof 3.1.2 isPrototypeOf 3.2 修改__proto__|prototype 4 修改和获取原型对象的方式 4.1 修改 4.1.1 Object.create 4.1.2 Object.setPrototypeOf 4.2 获取 4.2.1 Object.getPrototypeOf

  • JavaScript深入浅出__proto__和prototype

    目录 构造函数和实例 prototype constructor 原型对象的原型 原型链 扩展知识 关于 Object 和 Function 总结 首先我们先记住几个知识点: 每个函数都有一个prototype属性 每个对象都有一个__proto__属性(null除外) 函数也是对象 构造函数和实例 首先我们通过下面的例子了解些基本的概念 function Person() { } var person1 = new Person() 使用 new 创建对象的函数就是构造函数.创建出的对象就是构

  • javascript 中__proto__和prototype详解

    __proto__是内部原型,prototype是构造器原型(构造器其实就是函数) 构造器的原型(prototype)是一个对象 那什么是构造器呢? 要想创建一个对象,首先要有一个对象构造器,就像php里面一样,要想创建一个对象,首先要有一个类 构造器的实质就是一个函数,下面的问题是:如何通过这个构造器来创建一个对象呢? 答案: new 构造器构造的是对象. 一.所有构造器/函数的__proto__都指向Function.prototype,它是一个空函数(Empty function) 复制代

  • JavaScript中__proto__与prototype的关系深入理解

    这里讨论下对象的内部原型(__proto__)和构造器的原型(prototype)的关系. 一.所有构造器/函数的__proto__都指向Function.prototype,它是一个空函数(Empty function) 复制代码 代码如下: Number.__proto__ === Function.prototype // true Boolean.__proto__ === Function.prototype // true String.__proto__ === Function.

  • 深入学习JavaScript中的原型prototype

    javascript 是一种 prototype based programming 的语言, 而与我们通常的 class based programming 有很大 的区别,我列举重要的几点如下: 1.函数是first class object, 也就是说函数与对象具有相同的语言地位 2.没有类,只有对象 3.函数也是一种对象,所谓的函数对象 4.对象是按 引用 来传递的 那么这种 prototype based programming 的语言如何实现继承呢(OO的一大基本要素), 这也便是

  • Javascript学习笔记9 prototype封装继承

    好,那就让我们一步步打造,首先让我们来看下继承原本的写法: 复制代码 代码如下: <script> var Person = function(name, age) { this.name = name; this.age = age; } Person.prototype.SayHello = function () { alert(this.name + "," + this.age); }; var Programmer = function (name, age,

  • javascript学习笔记(九)javascript中的原型(prototype)及原型链的继承方式

    在使用面向对象编程时,对象间的继承关系自然少不了!而原型正是实现javascript继承的很重要的一种方法! 我们首先来看以下代码: 复制代码 代码如下: function person(name, age) { this.name = name; this.age = age; } person.prototype.getInfo = function() { alert("My name is "+this.name+", and I have "+this.a

  • javascript学习小结之prototype

    JS中的prototype是JS中比较难理解的一个部分 本文基于下面几个知识点: 1 原型法设计模式 在.Net中可以使用clone()来实现原型法 原型法的主要思想是,现在有1个类A,我想要创建一个类B,这个类是以A为原型的,并且能进行扩展.我们称B的原型为A. 2 javascript的方法可以分为三类: a -> 类方法 b -> 对象方法 c -> 原型方法 例子: function People(name){ //对象属性 this.name=name; //对象方法 this

  • JavaScript中的原型prototype完全解析

    要理解JS中的prototype, 首先必须弄清楚以下几个概念    1. JS中所有的东西都是对象 2. JS中所有的东西都由Object衍生而来, 即所有东西原型链的终点指向Object.prototype // ["constructor", "toString", "toLocaleString", "valueOf", "hasOwnProperty", "isPrototypeOf&q

  • 关于Javascript 对象(object)的prototype

    Javascript中的每个对象(object)都会有 prototype .试一下: 复制代码 代码如下: var Richard = new Object(); alert(typeof(Richard.prototype)); 结果令人郁闷,浏览器弹出来的是 undefined-- 到底是怎么回事呢? 再看一个例子: 复制代码 代码如下: function Richard(){} alert(typeof(Richard.prototype)); 上面的例子似乎说明,只有 function

  • JavaScript类和继承 prototype属性

    我们已经在第一章中使用prototype属性模拟类和继承的实现. prototype属性本质上还是一个JavaScript对象. 并且每个函数都有一个默认的prototype属性. 如果这个函数被用在创建自定义对象的场景中,我们称这个函数为构造函数. 比如下面一个简单的场景: 复制代码 代码如下: // 构造函数 function Person(name) { this.name = name; } // 定义Person的原型,原型中的属性可以被自定义对象引用 Person.prototype

随机推荐