本文主要讲三个 问题

  1. prototype 和 proto
  2. function 和 object
  3. new 到底发生了什么

prototype 和 proto

首先我们说下在 JS 中,常常让我们感到困惑的地方,就是 prototype 和 __proto__ 到底是干嘛的

1. __proto__ 就是 Javascript中  所谓的原型 (这里,我们还是拿具体的例子来说明吧)
    function A (name) {         // 这里是一个构造函数
thia.name = name
} var Aobj = { // 这里是一个 对对象字面量
name: ''
} // 我们分别打印出来这二个对象看看
console.dir(A) console.dir(Aobj)

这里我们可以很明显的看到
构造函数的 __proto__ 属性 指向了 function() 对象字面量的 __proto__ 属性 指向了 Object 为什么 指向的 是不一样的呢? 思考下: 确实是 不一样的, 因为 构造函数本身也是一个 函数, 所以它 的原型 指向 function() 而对象字面量 是一个 对象, 那么他的 原型肯定是指向 Object 扩展思考,如果 是一个数组 对象, 那么它的 __proto__ 会指向什么呐?
    const arr = [112,22,3333]
console.dir(arr)

没错, 这里的  __proto__ 就指向了 Array[0]

总结 :一个对象的 __proto__ 属性和自己的内部属性[[Prototype]]指向一个相同的值 (通常称这个值为原型)

tips:firefox、chrome等浏览器把对象内部属性 [[Prototype]] 用 __proto__ 的形式暴露了出来.(老版本的IE并不支持 __proto__ ,IE11中已经加上了 __proto__ 属性)

2.  prototype : 看看上面的 截图,你会发现 只有 构造函数 中 有这个玩意儿,  对的。 prototype 确实 是 在 function 中特有的。 别的对象类型中 都不会有的属性。

我们在看这个 function 对象属性的 时候就会发现这么一个 prototype 的属性,它的值是 一个 Object 。 点开这个 obj 我们就会发现啊, 这个 obj 的constructor 属性 指向了 这个构造函数本身。 是不是很神奇,至于为什么会是这样子的。

留一个 思考题吧, 为什么在 javascript 中,函数对象的 prototype 属性的 constructor 指向是 函数本身?

(在下面的介绍中,我们会 回答到这个问题)

function 和 object

同样,我们来先看一个例子。

function B(name) {
this.name = name
this.getName = function() {
console.log(this.name)
} var c = 'test'
console.log(c)
} var b = new B('testb') // test
console.log(b) // B: { name: 'testb',getName: function() {} }
B('testc') // test
看到上面的 输出 是不是觉得又很诧异了。
确实, 为什么 在 new 的时候, 构造函数居然 执行了一次。 同样, 在非严格模式下, 我们直接执行 构造函数, B('testc') 相当于:

// window.name = 'testc'
// window.getName = function() { console.log(this.name) }
思考:
我们的函数B既可以直接执行,又可以new一下返回一个对象。function和object到底是什么关系,new的时候发生了什么?

new 到底发生了什么

还是上面的 问题, 当我们执行 var b = new B('testb') 的时候发生了什么?

MDN 上的介绍是这样的说的:

对于 var b = new B('testb')

// javascript 实际上执行的是: var o = new Object() // 生成一个 新的 对象 b 这里 可以约等于 var b = {} o.__proto__ = B.prototype // 这里就是 函数对象中 独有的 prototype 属性。 // 这个独有的 prototype 属性 包含了一个 constructor 属性方法,指向的就是 构造函数, 也就是 这里的 function B(name) {} B.call(o) // tips :这里 就需要注意了,因为很多同学都搞不清楚 这里是什么意思。 // 由于 call 的使用 将这里this是指向o, 所以就 可以 把什么this.name/getName 强行的绑定到o上。同时,需要注意的一点就是, 这里的 构造函数 执行科一遍, 只不过是 将 this 指向的 属性和方法,都 强行的 给 新创建的 这个 o 对象 绑定了一遍。 var b = o // 把 这个 o 返回给了 b 。 从而完成了 var b = new B('testb') 的过程 // 如果 还是不明白是 什么意思的话。 我们来看看 call 是干嘛用的
    // 关于 call 的使用说明 

    var o1 = {
name: '111',
getName: function() {
console.log(this.name)
}
} var o2 = {
name: '222'
} o1.getName.call(o2) // 222
所以 这个时候,我们反过头来 看看  这个 new 的对象都有哪些 属性 和方法。

我们 可以 来 做一个 小实验,来 证明下,我们以上所说的东西。

function A (name) { // 这里是一个构造函数
this.name = name
} var o = {}
o.__proto__ = A.prototype
A.call(o)
var a = o var b = new A() console.log(a)
console.log(b)

果然 和 我们想象 的一模一样。 

至于 js 为什么要 把 新建 对象的 原型 指向 构造函数的 prototype 属性。

我们可以这样来理解。 因为 通过 new 方法来创建的  obj 。肯定是需要 一个 标记 来找到自己的 构造器函数。
所以 为了让 整个 程序结构看上去 合理。 我们需要 把 新建 对象的 原型 指向 构造函数的 prototype 属性。

`

所以到最后,我们 总结一下 。

在 javascript 中 prototype 和 proto 到底有什么区别。

prototype 是 面向 构造函数,来思考,

proto 是 面向 实例化 后 的对象 来思考就对了。

`

最后再 给一个例子, 是一个,我们经常会在开发中用到的 例子。

var Person = function(){}
Person.prototype.sayName = function() {
alert('my name is xxx')
} Person.prototype.age = 12 var p = new Person() p.sayName() // 当我们 实例化 之后, 在我们 去执行 p.sayName() 的 时候,我们就会去 this 内部去 查找(这里就是 构造函数 Person 内部去找。 可是 没找到啊。只是一个 空函数, 怎么办呢?) // 这个时候 就会沿着 原型链向上追溯, 但是如何 追溯呢? // 这里就要用到 __proto__ 属性 来作为 追溯的 桥梁。 // 因为 实例化对象的 __proto__ 属性 指向的就是 构造函数的 prototype 属性所对应的 对象啊

最后 终于 愉快的 找到了,自己的对象啦~ 单身狗就可能一直找不到,或者看不懂这篇文章。

好了,以上就是 在 看 vuejs 源码 的时候 关于 new 的 一个 知识 扩展。

在 JavaScript 中 prototype 和 __proto__ 有什么区别的更多相关文章

  1. 【前端】JavaScript中prototype和__proto__的区别

    转载请注明出处:http://www.cnblogs.com/shamoyuu/p/prototype.html 经常有小伙伴问我关于prototype和__proto__的问题,觉得有必要写一篇博客 ...

  2. javascript中prototype与__proto__

    1.prototype:构造函数独有的属性: __proto__:每个对象都有一个名为__proto__的属性: 注意:每个构造函数(自带与自创)都有一个prototype的属性,构造函数的proto ...

  3. Javascript中的prototype和__proto__的联系区别

    转载至http://www.cnblogs.com/sinstone/p/5136871.html   一.联系 prototype和__proto__都指向原型对象,任意一个函数(包括构造函数)都有 ...

  4. Javascript中prototype属性详解 (存)

    Javascript中prototype属性详解   在典型的面向对象的语言中,如java,都存在类(class)的概念,类就是对象的模板,对象就是类的实例.但是在Javascript语言体系中,是不 ...

  5. (转载)详解Javascript中prototype属性(推荐)

    在典型的面向对象的语言中,如java,都存在类(class)的概念,类就是对象的模板,对象就是类的实例.但是在Javascript语言体系中,是不存在类(Class)的概念的,javascript中不 ...

  6. 深入理解Javascript中构造函数和原型对象的区别

    在 Javascript中prototype属性的详解 这篇文章中,详细介绍了构造函数的缺点以及原型(prototype),原型链(prototype chain),构造函数(constructor) ...

  7. JavaScript中const、var和let区别浅析

    在JavaScript中有三种声明变量的方式:var.let.const.下文给大家介绍js中三种定义变量的方式const, var, let的区别. 1.const定义的变量不可以修改,而且必须初始 ...

  8. JavaScript中值类型和引用类型的区别

    JavaScript的数据类型分为两类:原始类型和对象类型.其中,原始类型包括:数字.字符串和布尔值.此外,JavaScript中还有两个特殊的原始值:null和undefined,它们既不是数字也不 ...

  9. JavaScript中=、==、===以及!=、!==的区别与联系

    JavaScript中=.==.===以及!=.!==的区别与联系   在JavaScript中,“=”代表赋值操作:“==”先转换类型再比较,“===”先判断类型,如果不是同一类型直接为false. ...

随机推荐

  1. Eclipse Sort Members默认之后恢复的方法

    alt+shift+s -- > m CRLF 默认一种排序之后,对话框就再也出不来了: 使用这样的方法: window - preference - Java | Java dialogs | ...

  2. windows10 subsystem(bash) 如何使用jupter notebook

    按照传统惯例,应该映射端口jupyter notebook --port=15000,然后用浏览器开http://localhost:15000/ 但是当你运行python的时候,居然发现kernel ...

  3. Nancy简单实战之NancyMusicStore(六):写在最后

    前言 由于公司搬家后,住的地方离上班的地方远了N倍,以前是走路十多分钟就可以到公司的,上班时间也从9:00提早到8:30 现在每天上班都是先坐公交,然后再坐地铁,在这段路上比较浪费时间而且每天都是要6 ...

  4. Javascript日期格式化指定格式的字符串实现

    代码部分 TypeScript /** * format a Date object * 将 Date 转化为指定格式的String * @param {Date} date 源日期对象 * @par ...

  5. java版二叉树算法实现

    import java.util.ArrayList; class BinaryTree { private static class TreeNode { int data; TreeNode le ...

  6. W3Cschool学习笔记——XHTML基础教程

    XHTML 是更严格更纯净的 HTML 代码. XHTML 是什么? XHTML 指可扩展超文本标签语言(EXtensible HyperText Markup Language). XHTML 的目 ...

  7. Eclipse / Intellij Idea配置Git+Maven+Jetty开发环境

    作者:鹿丸不会多项式 出处:http://www.cnblogs.com/hechao123  转载请先与我联系. 最近公司给加配了Mac,本想着花一个小时的时间搭好开发环境,最后全部弄好却用了一上午 ...

  8. (一) 从Angular1到Angular2的杂谈

    使用了angular1一年下来,完成了若干项目,承蒙此框架的强大带来了不算差的项目编写体验,但1.*版本的angular,确实是有厉害的地方也有其尴尬的地方,包括较多数据的渲染的性能问题,还有就是可能 ...

  9. jQuery中.bind() .live() .delegate() .on()的区别 和 三种方式写光棒事件 动画

    地狱的镰刀 bind(type,[data],fn) 为每个匹配元素的特定事件绑定事件处理函数. $("a").bind("click",function(){ ...

  10. 简述.jpg .Gif .png-8 .png-24的区别

    最近有很多朋友在开发过程中有时候会遇到图片加载不清晰,透明度失真,或者对图片进行操作之后造成图片损耗的现象,在这里给大家简单介绍一下常用的几种图片格式之间的区别 Gif格式特点: 1.透明性,Gif是 ...