每次创建新函数,就会根据规则为该函数创建一个 prototype 属性,该属性是一个指向函数原型对象的指针。并且原型对象都默认拥有一个 constructor 属性,该属性是一个指向那个新建函数的指针。

1、常规原型对象

function Person(){
}
Person.prototype.name = "Tom";
Person.prototype.job = "software engineer";
Person.prototype.age = 29;
Person.prototype.sayName = function (){
console.log(this.name);
}; var p1 = new Person();
var p2 = new Person(); p1.sayName(); // Tom
p2.sayName(); // Tom console.log(p1.sayName == p2.sayName); // true

下图展示了构造函数、原型对象、对象实例的关系

关于[[prototype]],在创建一个实例后,该实例内部将包含一个指向原型对象的指针[[prototype]],这个属性在有的实现中可以通过对象的 __proto__属性访问(有的浏览器或者执行环境不能)。要注意的是该属性是实例与构造函数的原型对象的联系,而不是实例与构造函数之间的联系。

上述两个对象实例 p1 和 p2 自身是没有属性和方法的,但都能调用sayName()方法。这是因为当读取某个属性时,会执行一次搜索,先从对象实例自身开始,如果本身有给定属性,则返回该属性的值;如果没有找到,则继续从它的原型对象中查找,找到则返回,没找到继续从原型对象的原型对象(如果该实例的原型对象有原型对象的话)中查找。找到为止,直至原型链的末端,如果最后还是没找到则报错。

介绍一些方法

function Person(){
}
Person.prototype.name = "Tom";
Person.prototype.job = "software engineer";
Person.prototype.age = 29;
Person.prototype.sayName = function (){
console.log(this.name);
}; var p1 = new Person();
var p2 = new Person(); p1.sayName(); // Tom
p2.sayName(); // Tom console.log(p1.sayName == p2.sayName); // true //判断对象之间是否是原型对象--实例的关系
console.log(Person.prototype.isPrototypeOf(p1)); //true
console.log(Person.prototype.isPrototypeOf(p2)); //true //Object.getPrototypeOf(),返回实例对象的[[prototype]]的值,即原型对象
console.log(Object.getPrototypeOf(p1) == Person.prototype); // true //hasOwnProperty(),继承自Object,检测对象自身是否存在某个属性
console.log(p1.hasOwnProperty("name")); //false // in 操作符,只要能通过对象访问到属性就返回true
console.log("name" in p1); // true //for-in 遍历所有能够访问到的且可枚举的属性
for(var prop in p1){
console.log(prop); //name,job,age,sayName
} // Object.keys(),返回参数对象自身的所有可枚举属性的字符串数组
var keys = Object.keys(Person.prototype);
console.log(keys); //["name","job","age","sayName"] //Object.getOwnPropertyNames(),返回参数对象自身的所有属性的字符串数组,包含不可枚举属性
keys = Object.getOwnPropertyNames(Person.prototype);
console.log(keys); //["constructor","name","job","age","sayName"] p1.name = "Greg"; console.log(p1.name); // Greg --来自实例
console.log(p2.name); // Tom --来自原型 console.log(p1.hasOwnProperty("name")); //true
console.log(p2.hasOwnProperty("name")); //false console.log("name" in p1); // true
console.log("name" in p2); // true console.log(Object.keys(p1)); //["name"]
keys = Object.getOwnPropertyNames(p1);
console.log(keys); ////["name"] delete p1.name;
console.log(p1.name); //Tom --来自原型
console.log(p1.hasOwnProperty("name")); //false

 

2、对象字面量形式的原型

function Person(){
} Person.prototype = {
name : "Tom",
job : "software engineer",
age : 29,
sayName : function (){
console.log(this.name);
}
}; var p1 = new Person();
var p2 = new Person(); p1.sayName(); // Tom
p2.sayName(); // Tom console.log(p1.sayName == p2.sayName); // true
console.log(p1 instanceof Object); //true
console.log(p1 instanceof Person); //true
console.log(p1.constructor == Person); //false
console.log(p1.constructor == Object); //true

对象字面量形式的原型,实际上是完全重写了默认的 prototype 对象,因此 constructor 属性已经是新对象的constructor属性了,指向Object,不再指向Person函数了。

可以在原型重写时加上constructor属性,手动添加属性,会导致它的[[enumerable]]特性设置为true,即设为可枚举的,而默认的原生constructor 属性是不可枚举的。

function Person(){
} Person.prototype = {
//constructor: Person,
name : "Tom",
job : "software engineer",
age : 29,
sayName : function (){
console.log(this.name);
}
};
//可以特别定义为不可枚举
Object.defineProperty(Person.prototype,"constructor",{
enmuerable:false,
value:Person
});

字面量对象重写原型可能会出现一些错误:

function Person(){
} var p1 = new Person(); Person.prototype = {
constructor: Person,
name : "Tom",
job : "software engineer",
age : 29,
sayName : function (){
console.log(this.name);
}
}; var p2 = new Person(); p2.sayName(); // Tom
p1.sayName(); // error : p1.sayName is not a function

上面例子中,先创建 p1 对象,后重写原型,导致 p1 的原型对象不是新的原型对象,所以执行 p1.sayName() 时,从 p1 的原型中找不到 sayName 方法。

如果支持__proto__属性的话,可以在重写原型对象后,添加语句:p1.__proto__ = Person.prototype; 重新建立实例和原型的关联。

js 中的原型prototype的更多相关文章

  1. js中的原型prototype

    var arr1 = new Array(12,34,98,43,38,79,56,1); arr1.sum=function (){ var result = 0; for(var i=0; i&l ...

  2. JS 中的原型 -- prototype、__proto__ 以及原型链

    原文: 1.深入理解javascript原型和闭包——prototype原型 2.三张图搞懂JavaScript的原型对象与原型链 打开浏览器控制台,任意定义一个对象,打印出来后,会发现有最后一定有一 ...

  3. js中的原型、继承的一些想法

    最近看到一个别人写的js类库,突然对js中的原型及继承产生了一些想法,之前也看过其中的一些内容,但是总不是很清晰,这几天利用空闲时间,对这块理解了一下,感觉还是有不通之处,思路上没那么条理,仅作为分享 ...

  4. 谈谈JS中的原型

    不知道大家对JS中的原型理解的怎么样,我想如果大家对JS中的原型对象以及prototype属性十分熟悉的话对后面原型链以及继承的理解会十分的容易,这里想和大家分享自己对其的理解,请先看下面这段代码O( ...

  5. JavaScript中的原型prototype和__proto__的区别及原型链概念

    问题 初学js的同学,总是搞不清楚js中的原型是什么东西,看着控制台打印出来的一串串__proto__,迷惑不已. 例如我定义一个Person,创建一个实例p,并打印实例. function Pers ...

  6. 说一说js中__proto__和prototype以及原型继承的那些事

    在面试中遇到过,问js如何实现继承,其实最好的方式就是构造函数+原型,今天在讨论中,发现自己以前理解上的一些误区,特地写出来,最近都比较忙,等手上的项目做完,可以来做个总结. 先说我以前没有认识到位的 ...

  7. js中__proto__, property, prototype, 对象自身属性方法和原型中的属性方法的区别

    __proto__: 这个属性是实例对象的属性,每个实例对象都有一个__proto__属性,这个属性指向实例化该实例的构造函数的原型对象(prototype). proterty:这个方法是对象的属性 ...

  8. 【JS】中的原型prototype到底是个啥

    一.什么是原型 原型prototype是函数的一个属性,这个属性是一个指针,指向一个对象(原型对象),这个原型对象的用途是包含可以由特定类型的所有实例共享的属性和方法. 函数也是一种对象.它也是属性的 ...

  9. js中constructor和prototype

    在最开始学习js的时候,我们在讲到原型链和构造函数的时候经常会有一个例子 如果我们定义函数如下: function Foo() { /* .. */ } Foo.prototype.bar = fun ...

随机推荐

  1. 可变,不可变与 id 的关系

    变量名不能使用关键字: 查看关键字 import  keyword keyword.kwlist 可变与不可变: 列表添加元素后,id并不会改变.说明列表可变 元祖添加元素后,id会改变,就不是同一对 ...

  2. SambaJava API

    做一个 backup package net.jnas; import java.io.File; import java.io.FileInputStream; import java.io.Fil ...

  3. 4. mysql 1449 : The user specified as a definer ('test'@'%') does not exist 解决方法

    权限问题,授权 给 root  所有sql 权限 mysql> grant all privileges on *.* to test@"%" identified by & ...

  4. nginx 服务器常见配置以及负载均衡

    # 配置启动用户,用户权限不够会出现访问 403 的情况 user root; # 启动多少个工作进程 worker_processes 1; # 错误日志文件进程文件的保存地址 error_log ...

  5. msimg32.lib不用为绝对路径发愁

    msimg32.lib不用为绝对路径发愁 以前是每个工程添加bcb绝对路径下的 D:\Program Files (x86)\Borland\CBuilder6\Lib\Psdk\msimg32.li ...

  6. windows下Jenkins环境搭建

    Jenkins简介 Jenkins是一个开源软件项目,业界著名的持续集成工具. Jenkins 安装准备 1.      安装java 并且配置jdk环境 2.      到Jenkins官网下载Je ...

  7. C#--构造函数的理解

    说白了构造函数就是用来初始化类的数据成员{因为C#语言具有类型安全的特质-->不能使用没有初始化的变量)} 在这里引用一下别人的总结,我觉得挺好的: 构造函数是一种特殊的成员函数,它主要用于为对 ...

  8. nginx+php 开启https

    nginx 配置如下,配置好重启nginx,不是nginx -s reload,如果还不能访问肯定就是防火墙问题,关闭防火墙再试试. 我遇到的问题是:我服务器是ecs,域名解析到阿里云复杂均衡的,结果 ...

  9. git工作操作步骤

    上班开始,打开电脑,git pull:拉取git上最新的代码: 编辑代码,准备提交时,git stash:将自己编辑的代码暂存起来,防止git pull时与库中的代码起冲突,否则自己的代码就白敲了: ...

  10. 0.1Linux系统开发Angular项目一一首次运行环境的安装(chrome ,terminator,git,node)

    首先,保证你已经安装了虚拟机(虚拟机可以用virturalbox或者VM)并安装了ubuntu镜像! 安装Chrome浏览器 安装terminator(可以多开)代替原来的命令行工具 sudo apt ...